From 4eaa547b1a57728017b3574b3f8fba2a50114d95 Mon Sep 17 00:00:00 2001 From: Jackson Date: Tue, 9 Jul 2024 16:59:19 -0400 Subject: [PATCH 1/3] Document wallet.ts example --- example/wallet.ts | 408 ++++++++++++++++++++++++++++++++++------------ 1 file changed, 305 insertions(+), 103 deletions(-) diff --git a/example/wallet.ts b/example/wallet.ts index 61dbaea..3d8184a 100644 --- a/example/wallet.ts +++ b/example/wallet.ts @@ -31,179 +31,356 @@ import { } from '../' /** - * A class to create a mock wallet for demonstration a wallet - * implementation compatible with Aptos AIP-62 Wallet Standard + * This class is a template you can modify to implement an AIP-62 Wallet. + * + * Sections of the code which need to be revised will be marked with a "REVISION" comment. + * We recommend using the REVISION comments like a checklist and deleting them as you go. + * Ex. REVISION - Update this section. + * + * Function implementations are for DEMONSTRATION PURPOSES ONLY. Please ensure you rewrite all features + * to use your Wallet as the method of communicating on-chain. + * + * For a working implementation of this example, see the next-js example app here: https://github.com/aptos-labs/aptos-wallet-adapter/tree/main/apps/nextjs-example + * (And more specifically, see https://github.com/aptos-labs/aptos-wallet-adapter/blob/main/apps/nextjs-example/src/utils/standardWallet.ts) */ + +/** + * Interface of a **WalletAccount**, also referred to as an **Account**. + * + * An account is a _read-only data object_ that is provided from the Wallet to the app, authorizing the app to use it. + * + * The app can use an account to display and query information from a chain. + * + * The app can also act using an account by passing it to functions of the Wallet. + * + * Wallets may use or extend {@link "@wallet-standard/wallet".ReadonlyWalletAccount} which implements this interface. + * + */ +// REVISION - Replace the "MyWallet" in "MyWalletAccount" with the name of your wallet. Ex. "PetraAccount" export class MyWalletAccount implements AptosWalletAccount { - address: string + /** Address of the account, corresponding with a public key. */ + address: string; - publicKey: Uint8Array + /** Public key of the account, corresponding with a secret key to use. */ + publicKey: Uint8Array; + /** + * Chains supported by the account. + * + * This must be a subset of ["aptos:devnet", "aptos:testnet", "aptos:localnet", "aptos:mainnet"]. + * + * It is recommended to support at least ["aptos:devnet", "aptos:testnet", and "aptos:mainnet"]. + */ chains: IdentifierArray = APTOS_CHAINS - features: IdentifierArray = [] + /** + * Function names of features that are supported for this Wallet's account object. + */ + features: IdentifierArray; - signingScheme: SigningScheme + /** The signing scheme used for the private key of the address. Ex. SigningScheme.Ed25519 */ + signingScheme: SigningScheme; - label?: string + /** Optional user-friendly descriptive label or name for the account. This may be displayed by the app. */ + label?: string; + /** + * Optional user-friendly icon for the account. This may be displayed by the app. + */ icon?: | `data:image/svg+xml;base64,${string}` | `data:image/webp;base64,${string}` | `data:image/png;base64,${string}` | `data:image/gif;base64,${string}` - | undefined + | undefined; + // REVISION - Update this constructor to use values your wallet supports. constructor(account: Account) { - this.address = account.accountAddress.toString() - this.publicKey = account.publicKey.toUint8Array() - this.signingScheme = account.signingScheme + this.address = account.accountAddress.toString(); + this.publicKey = account.publicKey.toUint8Array(); + // REVISION - Choose which chains your wallet supports. This may only be subset of all Aptos networks. + this.chains = APTOS_CHAINS; // ["aptos:devnet", "aptos:testnet", "aptos:localnet", "aptos:mainnet"] + /** + * REVISION - Ensure this signing scheme matches the encoding used to generate your private key. + */ + this.signingScheme = account.signingScheme; } } +/** + * REVISION - This class needs to be extensively customized to match the details of your wallet. + * + * 1. MyWallet should be renamed to be the name of your wallet. Ex. For Petra, MyWallet should be named "PetraWallet". (Be sure to also update references to "MyWallet" in this file.) + * 2. Update the values of this class to match your Wallet's deatils. + * 3. Implement each of the features below. (Including adding implementations for any additional required features that you can find here in the "AptosFeatures" type: https://github.com/aptos-labs/wallet-standard/blob/main/src/features/index.ts) + */ export class MyWallet implements AptosWallet { - readonly url: string = 'https://aptos.dev' - readonly version = '1.0.0' - readonly name: string = 'Aptos Burner' - readonly icon = - '' - chains = APTOS_CHAINS - accounts: MyWalletAccount[] = [] - - // Local MyWallet class variables - signer: Account - aptos: Aptos + // REVISION - Include the link to create an account using your wallet or your primary website. (Ex. https://chromewebstore.google.com/detail/petra-aptos-wallet/ejjladinnckdgjemekebdpeokbikhfci?hl=en) + readonly url: string = "https://aptos.dev"; + // This should be updated whenever you release a new implementation of "MyWallet" + readonly version = "1.0.0"; + // REVISION - Change the name to the name of your Wallet. (Ex. "Petra") + readonly name: string = "Aptos Burner"; + /** + * REVISION - Set the icon to be a base64 encoding of your Wallet's logo. + * + * The icon data must be of the format: + * 1. "data:image/" + * 2. The icon's file extension, which must be one of: + * - "svg+xml" + * - "webp" + * - "png" + * - "gif" + * 3. ";base64," + * 4. The base64 encoding of the image file. + * + * See the current value of icon for an example of this format. + */ + readonly icon = + ""; + /** + * REVISION - Set the subset of Aptos chains your wallet supports. + * APTOS_CHAINS = ["aptos:devnet", "aptos:testnet", "aptos:localnet", "aptos:mainnet"] + * It is recommended to support at least "aptos:mainnet", "aptos:testnet", and "aptos:devnet". + */ + chains = APTOS_CHAINS; + /** + * The set of accounts that your Wallet has shared information for. These do NOT include private keys. + * This list is normally expanded during `aptos:connect` and reduced during `aptos:disconnect`. + * NOTE: For demonstration purposes, the template initializes a default account in the constructor, + * but that should NOT be carried into your final implementation of this template. + */ + accounts: MyWalletAccount[] = []; + + // Local MyWallet class variables, + /** + * REVISION - These two variables likely should NOT be in your finalized plugin template. + * They are used throughout this example's feature implementations in order to show how you could + * implement each function. + * + * signer - This stores the private keys for an account on-chain. (Example purposes only) + * aptos - This handles the network connection. (Your wallet may have a different way of handling the on-chain connection than this Aptos instance) + * + * Remember: These two variables SHOULD LIKELY BE DELETED after you replace your implementations of each feature with ones that use your Wallet. + */ + signer: Account; + aptos: Aptos; + /** + * REVISION - List all features your wallet supports below. + * You will need to implement how your wallet supports each. + * + * In order to be compatible with the AIP-62 Wallet standard, ensure you are at least supporting all + * currently required features by checking the list of features in the `AptosFeatures` type here: + * https://github.com/aptos-labs/wallet-standard/blob/main/src/features/index.ts + * + * To find the names of features to pass into `this.features` below you can either go into the feature implementations + * and look at the NameSpace variable, or you can import the `AptosFeatures` type and see the names there. + * Ex. See `AptosSignTransactionNamespace` in https://github.com/aptos-labs/wallet-standard/blob/main/src/features/aptosSignTransaction.ts + * + * For additional customization, you may implement optional features. + * For the most support though, you should extend the wallet-standard to support additional features as part of the standard. + */ get features(): AptosFeatures { return { - 'aptos:connect': { - version: '1.0.0', - connect: this.connect + "aptos:connect": { + version: "1.0.0", + connect: this.connect, }, - 'aptos:network': { - version: '1.0.0', - network: this.network + "aptos:network": { + version: "1.0.0", + network: this.network, }, - 'aptos:disconnect': { - version: '1.0.0', - disconnect: this.disconnect + "aptos:disconnect": { + version: "1.0.0", + disconnect: this.disconnect, }, - 'aptos:signTransaction': { - version: '1.0.0', - signTransaction: this.signTransaction + "aptos:signTransaction": { + version: "1.0.0", + signTransaction: this.signTransaction, }, - 'aptos:signMessage': { - version: '1.0.0', - signMessage: this.signMessage + "aptos:signMessage": { + version: "1.0.0", + signMessage: this.signMessage, }, - 'aptos:onAccountChange': { - version: '1.0.0', - onAccountChange: this.onAccountChange + "aptos:onAccountChange": { + version: "1.0.0", + onAccountChange: this.onAccountChange, }, - 'aptos:onNetworkChange': { - version: '1.0.0', - onNetworkChange: this.onNetworkChange + "aptos:onNetworkChange": { + version: "1.0.0", + onNetworkChange: this.onNetworkChange, }, - 'aptos:account': { - version: '1.0.0', - account: this.account - } - } + "aptos:account": { + version: "1.0.0", + account: this.account, + }, + }; } + /** + * REVISION - This constructor should be updated to support your Wallet's implementation of the supported features. + * + * The template code's constructor currently initializes `signer` to act as the private key for an account on-chain, and uses + * `aptos` to handle the on-chain connection. + * + */ constructor() { - // Local MyWallet class variables - this.signer = Account.generate() + // THIS LOGIC SHOULD BE REPLACED. IT IS FOR EXAMPLE PURPOSES ONLY. + // Create a random signer for our stub implementations. + this.signer = Account.generate(); + // We will use DEVNET since we can fund our test account via a faucet there. const aptosConfig = new AptosConfig({ - network: Network.DEVNET - }) - this.aptos = new Aptos(aptosConfig) + network: Network.DEVNET, + }); + // Use the instance Aptos connection to process requests. + this.aptos = new Aptos(aptosConfig); - this.accounts = [new MyWalletAccount(this.signer)] + // Update our Wallet object to know that we are connected to this new signer. + this.accounts = [new MyWalletAccount(this.signer)]; } + /** + * REVISION - Implement this function using your Wallet. + * + * Look up the account info for the currently connected wallet address on the chosen network. + * + * @returns Return account info. + */ account: AptosGetAccountMethod = async (): Promise => { const account = new AccountInfo({ address: this.signer.accountAddress, - publicKey: this.signer.publicKey - }) - return Promise.resolve(account) - } + publicKey: this.signer.publicKey, + }); + return Promise.resolve(account); + }; - connect: AptosConnectMethod = async (): Promise> => { + /** + * REVISION - Implement this function using your Wallet. + * + * Connect an account using this Wallet. + * This must wait for the user to sign in to the Wallet provider and confirm they are ok sharing + * details with the dapp. + * + * For demonstration purposes, this template example assumes the user is using the account generated in `signer` + * and assumes the user approved letting the dapp use the account information. + * + * Your implmentation should include a way to track which account was just connected. This likely will involve + * setting the `this.accounts` variable. + * + * @returns Whether the user approved connecting their account, and account info. + * @throws Error when unable to connect to the Wallet provider. + */ + connect: AptosConnectMethod = async (): Promise< + UserResponse + > => { + // THIS LOGIC SHOULD BE REPLACED. IT IS FOR EXAMPLE PURPOSES ONLY. try { await this.aptos.fundAccount({ accountAddress: this.signer.accountAddress, - amount: 1_000_000 - }) + amount: 1_000_000, + }); const account = new AccountInfo({ address: this.signer.accountAddress, - publicKey: this.signer.publicKey - }) + publicKey: this.signer.publicKey, + }); return { status: UserResponseStatus.APPROVED, - args: account - } + args: account, + }; } catch (e) { return { status: UserResponseStatus.REJECTED - } + } } - } + }; + /** + * REVISION - Implement this function using your Wallet. + * + * Return the name, chainId, and url of the network connection your wallet is using to connect to the Aptos chain. + * + * @returns Which network the connected Wallet is pointing to. + */ network: AptosGetNetworkMethod = async (): Promise => { - const network = await this.aptos.getLedgerInfo() + // THIS LOGIC SHOULD BE REPLACED. IT IS FOR EXAMPLE PURPOSES ONLY. + // You may use getLedgerInfo() to determine which ledger your Wallet is connected to. + const network = await this.aptos.getLedgerInfo(); return { + // REVISION - Ensure the name and url match the chain_id your wallet responds with. name: Network.DEVNET, chainId: network.chain_id, - url: 'https://fullnode.devnet.aptoslabs.com/v1' - } - } + url: "https://fullnode.devnet.aptoslabs.com/v1", + }; + }; + /** + * REVISION - Implement this function using your Wallet. + * + * Remove the permission of the Wallet class to access the account that was connected. + * + * @returns Resolves when done cleaning up. + */ disconnect: AptosDisconnectMethod = async (): Promise => { - return Promise.resolve() - } + // THIS LOGIC SHOULD BE REPLACED. IT IS FOR EXAMPLE PURPOSES ONLY. + return Promise.resolve(); + }; + /** + * REVISION - Implement this function using your Wallet. + * + * @param transaction - A transaction that the user should have the ability to sign if they choose to. + * @param asFeePayer - Optionally, another this signature is acting as a fee-payer for the transaction being signed. + * @returns The result of whether the user chose to sign the transaction or not. + */ signTransaction: AptosSignTransactionMethod = async ( transaction: AnyRawTransaction, - asFeePayer?: boolean + asFeePayer?: boolean, ): Promise> => { + // THIS LOGIC SHOULD BE REPLACED. IT IS FOR EXAMPLE PURPOSES ONLY. if (asFeePayer) { const senderAuthenticator = this.aptos.transaction.signAsFeePayer({ signer: this.signer, - transaction - }) + transaction, + }); return Promise.resolve({ status: UserResponseStatus.APPROVED, - args: senderAuthenticator - }) + args: senderAuthenticator, + }); } const senderAuthenticator = this.aptos.transaction.sign({ signer: this.signer, - transaction - }) + transaction, + }); return Promise.resolve({ status: UserResponseStatus.APPROVED, - args: senderAuthenticator - }) - } + args: senderAuthenticator, + }); + }; + /** + * REVISION - Implement this function using your Wallet. + * + * @param input - A message to sign with the private key of the connected account. + * @returns A user response either with a signed message, or the user rejecting to sign. + */ signMessage: AptosSignMessageMethod = async ( - input: AptosSignMessageInput + input: AptosSignMessageInput, ): Promise> => { + // THIS LOGIC SHOULD BE REPLACED. IT IS FOR EXAMPLE PURPOSES ONLY. // 'Aptos' + application + address + nonce + chainId + message const messageToSign = `Aptos demoAdapter ${this.signer.accountAddress.toString()} ${input.nonce} ${input.chainId ?? (await this.network()).chainId} - ${input.message}` + ${input.message}`; - const encodedMessageToSign = new TextEncoder().encode(messageToSign) + const encodedMessageToSign = new TextEncoder().encode(messageToSign); - const signature = this.signer.sign(encodedMessageToSign) + const signature = this.signer.sign(encodedMessageToSign); return Promise.resolve({ status: UserResponseStatus.APPROVED, @@ -212,25 +389,50 @@ export class MyWallet implements AptosWallet { fullMessage: messageToSign, message: input.message, nonce: input.nonce, - prefix: 'APTOS', - signature: signature - } - }) - } + prefix: "APTOS", + signature: signature, + }, + }); + }; + /** + * REVISION - Implement this function using your Wallet. + * + * An event which will be triggered anytime an Account changes. + * + * @returns when the logic is resolved. + */ onAccountChange: AptosOnAccountChangeMethod = async (): Promise => { - return Promise.resolve() - } + // THIS LOGIC SHOULD BE REPLACED. IT IS FOR EXAMPLE PURPOSES ONLY. + return Promise.resolve(); + }; + /** + * REVISION - Implement this function using your Wallet. + * + * When users indicate a Network change should occur, update your Wallet accordingly. + * + * @returns when the logic is resolved. + */ onNetworkChange: AptosOnNetworkChangeMethod = async (): Promise => { - return Promise.resolve() - } -} - -// a wallet call to register itself -function register() { - const myWallet = new MyWallet() - registerWallet(myWallet) + // THIS LOGIC SHOULD BE REPLACED. IT IS FOR EXAMPLE PURPOSES ONLY. + return Promise.resolve(); + }; } -register() +/** + * REVISION - This section is ONLY for Browser Extension Wallets. + * + * Ensure that you import ant call registerWallet on pageload in your Wallet's logic. + * This will enable any dapps that are using the AIP-62 Wallet Adapter package to detect + * your wallet and connect to it. + * + * The below registration is for example purposes only, and likely should not occur in the + * same file as your "MyWallet" implementation in practice since registerWallet must be called + * when the page is loaded. + */ +(function () { + if (typeof window === "undefined") return; + const myWallet = new MyWallet(); + registerWallet(myWallet); +})(); From 2630d4bc41db1bda1ff80f5f5880a8bba0a9b87d Mon Sep 17 00:00:00 2001 From: Jackson <163934542+jmintuitive@users.noreply.github.com> Date: Tue, 9 Jul 2024 18:44:03 -0400 Subject: [PATCH 2/3] Update example/wallet.ts Co-authored-by: Maayan --- example/wallet.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/example/wallet.ts b/example/wallet.ts index 3d8184a..e7c5f28 100644 --- a/example/wallet.ts +++ b/example/wallet.ts @@ -423,7 +423,7 @@ export class MyWallet implements AptosWallet { /** * REVISION - This section is ONLY for Browser Extension Wallets. * - * Ensure that you import ant call registerWallet on pageload in your Wallet's logic. + * Ensure that you import and call registerWallet on pageload in your Wallet's logic. * This will enable any dapps that are using the AIP-62 Wallet Adapter package to detect * your wallet and connect to it. * From f3de6bd8eba86e30560d9c5cf47de93765e6552f Mon Sep 17 00:00:00 2001 From: Jackson Date: Tue, 9 Jul 2024 18:46:15 -0400 Subject: [PATCH 3/3] Add note about skipping network call. --- example/wallet.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/example/wallet.ts b/example/wallet.ts index e7c5f28..bbc121b 100644 --- a/example/wallet.ts +++ b/example/wallet.ts @@ -309,6 +309,7 @@ export class MyWallet implements AptosWallet { return { // REVISION - Ensure the name and url match the chain_id your wallet responds with. name: Network.DEVNET, + // REVISION - For mainnet and testnet is not recommended to make the getLedgerInfo() network call as the chain_id is fixed for those networks. chainId: network.chain_id, url: "https://fullnode.devnet.aptoslabs.com/v1", };