Testing
Developing Decentralized Applications (DApps) on the Cosmos network often involves integrating with the Keplr wallet. Testing is key for ensuring users can flawlessly interact with your DApp through the Keplr wallet.
@agoric/synpress is an end-to-end (e2e) testing framework that simplifies the testing process for Keplr-based DApps. This framework automates testing how your DApp interacts with the Keplr wallet, simulating real user experiences. @agoric/synpress is built upon synthetixio/synpress, a framework designed for Metamask-based DApps.
Since @agoric/synpress is based on Cypress, the official Cypress documentation has a lot of information that you can use with @agoric/synpress.
Installation
Before you start testing your DApp with Keplr, you'll need to install the @agoric/synpress in your project using the following command:
yarn add -D @agoric/synpressProject structure
We recommend the following project structure for keeping your e2e tests clean and manageable:
project_dir
└── src
└── tests
└── e2e
└── synpress.config.js
└── support.js
└── specs
└── test.spec.jsConfigure synpress.config.js
Before you start writing your E2E tests, you'll need to create a configuration file named synpress.config.js in the tests/e2e directory of your project.
Inside synpress.config.js, you'll set the baseUrl property within the e2e configuration. This tells Cypress where to find your DApp during testing. Here's an example configuration:
const baseConfig = require('@agoric/synpress/synpress.config');
const { defineConfig } = require('cypress');
module.exports = defineConfig({
...baseConfig,
e2e: {
...baseConfig.e2e,
baseUrl: 'http://localhost:5173'
}
});In this example, baseUrl is set to http://localhost:5173. Make sure to replace this with the actual URL where your DApp is running.
By default, if you don't create a synpress.config.js file, @agoric/synpress assumes your DApp runs on port 3000 of your local machine (http://localhost:3000). Specifying the correct baseUrl ensures your tests interact with the right instance of your DApp.
Since you have a separate synpress.config.js in the tests/e2e folder, you need to specify its location when running your tests. Use the --configFile flag with the synpress run command:
EXTENSION=keplr synpress run --configFile=tests/e2e/synpress.config.jsCreating a Support File
Navigate to your project's tests/e2e directory and create a new file named support.js. This file can be used to create reusable commands that encapsulate common testing actions. For example, you can create a custom command to navigate to a specific page within your application. You can read more about it over here.
After creating your support.js file, make sure to include the following import statement:
import '@agoric/synpress/support/index';This import is essential because it brings in the necessary functionalities from @agoric/synpress to interact with the Keplr wallet within your e2e tests. Without it, your tests won't be able to leverage the features provided by @agoric/synpress for Keplr integration.
Example Tests
With the environment set up, let's write end-to-end (e2e) tests to test your DApp's functionality. In the tests/e2e/specs folder, create a new file with a descriptive name that reflects what it tests (e.g., user_login.spec.js or token_transfer.spec.js).
Test Structure
You use describe blocks to group related tests together, and it blocks to define individual test cases within those groups.
describe('User Login', () => {
it('should login with valid credentials', () => {
// Test steps for login functionality
});
});Test 1: Setting Up Keplr Wallet
it('should setup a Keplr wallet', () => {
cy.setupWallet({
secretWords: 'KEPLR_MNEMONIC'
});
cy.visit('/');
});This test case simulates setting up a Keplr wallet for your tests, using the cy.setupWallet method. Make sure to replace KEPLR_MNEMONIC with a 24-word mnemonic phrase. The setupWallet method creates a wallet based on the provided mnemonic phrase, which can then be used throughout your test suite.
After setting up the wallet, we visit the root path (/) of the DApp using cy.visit('/').
Test 2: Connecting Keplr Wallet
it('should accept connection with wallet', () => {
cy.contains('Connect Wallet').click();
cy.acceptAccess();
});This test simulates a user connecting their Keplr wallet to your DApp. cy.contains('Connect Wallet') searches for an element containing the text Connect Wallet on the webpage and triggers a click event. cy.acceptAccess simulates accepting Keplr wallet access for your DApp.
Test 3: Signing a Transaction
it('should confirm make an offer transaction', () => {
cy.contains('Make an Offer').click();
cy.confirmTransaction();
});This test simulates transaction Signing on your DApp. cy.contains('Make an Offer') searches for an element containing the text Make an Offer and triggers a click event. cy.confirmTransaction simulates confirming a transaction.
Running the Tests
You can now trigger the tests by running this command:
EXTENSION=keplr synpress run --configFile=tests/e2e/synpress.config.jsProjects Using @agoric/synpress
Some examples projects utilizing @agoric/synpress for e2e tests:
@agoric/synpress Commands
The @agoric/synpress commands are custom Cypress commands designed for use with Keplr. They provide ready-made functions to interact with Keplr-specific elements and workflows.
setupWallet
Initializes a Keplr wallet for testing. You can either create a new one or import an existing one using the provided options.
Arguments:
(All arguments are optional)
The command takes a single object as an argument with the following keys:
secretWords(string): The secret words for the wallet. (default is a predefined mnemonic phrase that represents a wallet address on the Agoric chain)privateKey(string): The private key for the wallet.password(string): The password for the wallet. (default isTest1234).walletName(string): The name of the wallet. (default isMy Wallet).selectedChains(array): The chains to select. (default is an empty array,[]).createNewWallet(boolean): Whether to create a new wallet (default isfalse).
Returns:
- (boolean):
trueif the wallet was set up successfully,falseotherwise.
acceptAccess
Handles the authorization process for connecting the Keplr wallet to a DApp. DApps provide a Connect Wallet option to initiate this process.
Returns:
- (boolean):
trueif access was accepted successfully,falseotherwise.
confirmTransaction
Handles signing transactions within a DApp using the Keplr wallet. This occurs when a user initiates a transaction, such as sending tokens or interacting with a smart contract.
Returns:
- (boolean):
trueif transaction was approved successfully,falseotherwise.
disconnectWalletFromDapp
Disconnects the Keplr wallet from all previously connected DApps.
Returns:
- (boolean):
trueif the disconnection was successful,falseotherwise.
getWalletAddress
Retrieves the wallet address of the currently active wallet associated with a specified blockchain.
Arguments:
chainName(string): The name of the blockchain for which you want to obtain the wallet address.
Returns:
- (string): The wallet address for the specified blockchain.
switchWallet
Switches the active wallet to the specified wallet by name.
Arguments:
walletName(string): The name of the wallet you want to switch to. This should match the name used when the wallet was created using thesetupWalletcommand.
Returns:
- (boolean):
trueif the wallet switch was successful,falseotherwise.
addNewTokensFound
Adds all the new tokens discovered by the Keplr extension to the user's token portfolio when it connects to a new blockchain network.
Returns:
- (boolean):
trueif the tokens were successfully added to the user's token portfoliofalseotherwise.
getTokenAmount
Retrieves the balance of a specified token in the wallet.
Arguments:
tokenName(string): The name of the token for which you want to retrieve the balance.
Returns:
- (number): The amount of the specified token currently held in the wallet.
isExtensionWindowActive
Determines whether the Keplr extension window is the currently active tab.
Returns:
- (boolean):
trueif the Keplr extension window is active,falseotherwise.
switchToExtensionWindow
Switches the focus to the Keplr extension window tab. Keplr window is used for managing wallet related operations.
Returns:
- (boolean):
trueif the focus change was successful,falseotherwise.
isCypressWindowActive
Determines whether the Cypress window, where the DApp is running, is the currently active tab.
Returns:
- (boolean):
trueif the Cypress window is active,falseotherwise.
switchToCypressWindow
Switches the focus to the Cypress window tab.
Returns:
- (boolean):
trueif the focus change was successful,falseotherwise.
For a full list of commands, you can refer to the Keplr plugin file.