diff --git a/src/components/vm/VmInitializer.tsx b/src/components/vm/VmInitializer.tsx index 1fe323dde..4cfb455fd 100644 --- a/src/components/vm/VmInitializer.tsx +++ b/src/components/vm/VmInitializer.tsx @@ -34,6 +34,8 @@ import { recordWalletConnect, reset as resetSegment } from '@/utils/analytics'; import { networkId, signInContractId } from '@/utils/config'; import { KEYPOM_OPTIONS } from '@/utils/keypom-options'; +import { setupFastAuthWallet } from '../../lib/selector/fast-auth-wallet'; + export default function VmInitializer() { const [signedIn, setSignedIn] = useState(false); const [signedAccountId, setSignedAccountId] = useState(null); @@ -68,14 +70,15 @@ export default function VmInitializer() { }), setupNightly(), setupWelldoneWallet(), - setupFastAuth({ - networkId, - signInContractId, - relayerUrl: - networkId === 'testnet' - ? 'http://34.70.226.83:3030/relay' - : 'https://near-relayer-mainnet.api.pagoda.co/relay', - }) as any, // TODO: Refactor setupFastAuth() to TS + // setupFastAuth({ + // networkId, + // signInContractId, + // relayerUrl: + // networkId === 'testnet' + // ? 'http://34.70.226.83:3030/relay' + // : 'https://near-relayer-mainnet.api.pagoda.co/relay', + // }) as any, // TODO: Refactor setupFastAuth() to TS + setupFastAuthWallet(), setupKeypom({ trialAccountSpecs: { url: diff --git a/src/lib/selector/fast-auth-icon.ts b/src/lib/selector/fast-auth-icon.ts new file mode 100644 index 000000000..a4b9c7965 --- /dev/null +++ b/src/lib/selector/fast-auth-icon.ts @@ -0,0 +1,2 @@ +/* eslint-disable import/no-anonymous-default-export */ +export default `` \ No newline at end of file diff --git a/src/lib/selector/fast-auth-wallet.ts b/src/lib/selector/fast-auth-wallet.ts new file mode 100644 index 000000000..358179a64 --- /dev/null +++ b/src/lib/selector/fast-auth-wallet.ts @@ -0,0 +1,239 @@ +import type { + Account, + BrowserWallet, + Network, + Optional, + Transaction, + WalletBehaviourFactory, + WalletModuleFactory, +} from "@near-wallet-selector/core"; +import { createAction } from "@near-wallet-selector/wallet-utils"; +import * as nearAPI from "near-api-js"; + +import icon from "./fast-auth-icon"; +import { FastAuthWalletConnection } from "./fastAuthWalletConnection"; + +export interface FastAuthWalletParams { + walletUrl?: string; + iconUrl?: string; + deprecated?: boolean; + successUrl?: string; + failureUrl?: string; +} + +interface FastAuthWalletState { + wallet: nearAPI.WalletConnection; + keyStore: nearAPI.keyStores.BrowserLocalStorageKeyStore; +} + +interface FastAuthWalletExtraOptions { + walletUrl: string; +} + +const resolveWalletUrl = (network: Network, walletUrl?: string) => { + if (walletUrl) { + return walletUrl; + } + + switch (network.networkId) { + case "mainnet": + return "http://localhost:3000"; + case "testnet": + return "http://localhost:3000"; + default: + throw new Error("Invalid wallet url"); + } +}; + +const setupWalletState = async ( + params: FastAuthWalletExtraOptions, + network: Network +): Promise => { + const keyStore = new nearAPI.keyStores.BrowserLocalStorageKeyStore(); + + const near = await nearAPI.connect({ + keyStore, + walletUrl: params.walletUrl, + ...network, + headers: {}, + }); + + const wallet = new FastAuthWalletConnection(near, "near_app"); + + return { + wallet, + keyStore, + }; +}; + +const FastAuthWallet: WalletBehaviourFactory< + BrowserWallet, + { params: FastAuthWalletExtraOptions } +> = async ({ metadata, options, store, params, logger }) => { + const _state = await setupWalletState(params, options.network); + const getAccounts = async (): Promise> => { + const accountId = _state.wallet.getAccountId(); + const account = _state.wallet.account(); + + if (!accountId || !account) { + return []; + } + + const publicKey = await account.connection.signer.getPublicKey( + account.accountId, + options.network.networkId + ); + return [ + { + accountId, + publicKey: publicKey ? publicKey.toString() : "", + }, + ]; + }; + + const transformTransactions = async ( + transactions: Array> + ) => { + const account = _state.wallet.account(); + const { networkId, signer, provider } = account.connection; + + const localKey = await signer.getPublicKey(account.accountId, networkId); + + return Promise.all( + transactions.map(async (transaction, index) => { + const actions = transaction.actions.map((action) => + createAction(action) + ); + const accessKey = await account.accessKeyForTransaction( + transaction.receiverId, + actions, + localKey + ); + + if (!accessKey) { + throw new Error( + `Failed to find matching key for transaction sent to ${transaction.receiverId}` + ); + } + + const block = await provider.block({ finality: "final" }); + + return nearAPI.transactions.createTransaction( + account.accountId, + nearAPI.utils.PublicKey.from(accessKey.public_key), + transaction.receiverId, + accessKey.access_key.nonce + index + 1, + actions, + nearAPI.utils.serialize.base_decode(block.header.hash) + ); + }) + ); + }; + + return { + async signIn({ contractId, methodNames, successUrl, failureUrl }) { + const existingAccounts = await getAccounts(); + + if (existingAccounts.length) { + return existingAccounts; + } + + await _state.wallet.requestSignIn({ + contractId, + methodNames, + successUrl, + failureUrl, + }); + + return getAccounts(); + }, + + async signOut() { + if (_state.wallet.isSignedIn()) { + _state.wallet.signOut(); + } + }, + + async getAccounts() { + return getAccounts(); + }, + + async verifyOwner() { + throw new Error(`Method not supported by ${metadata.name}`); + }, + + async signAndSendTransaction({ + signerId, + receiverId, + actions, + callbackUrl, + }) { + logger.log("signAndSendTransaction", { + signerId, + receiverId, + actions, + callbackUrl, + }); + + const { contract } = store.getState(); + + if (!_state.wallet.isSignedIn() || !contract) { + throw new Error("Wallet not signed in"); + } + + const account = _state.wallet.account(); + + return account["signAndSendTransaction"]({ + receiverId: receiverId || contract.contractId, + actions: actions.map((action) => createAction(action)), + walletCallbackUrl: callbackUrl, + }); + }, + + async signAndSendTransactions({ transactions, callbackUrl }) { + logger.log("signAndSendTransactions", { transactions, callbackUrl }); + + if (!_state.wallet.isSignedIn()) { + throw new Error("Wallet not signed in"); + } + + return _state.wallet.requestSignTransactions({ + transactions: await transformTransactions(transactions), + callbackUrl, + }); + }, + }; +}; + +export function setupFastAuthWallet({ + walletUrl, + iconUrl = icon, + deprecated = false, + successUrl = "", + failureUrl = "", +}: FastAuthWalletParams = {}): WalletModuleFactory { + return async (moduleOptions) => { + return { + id: "fast-auth-wallet", + type: "browser", + metadata: { + name: "FastAuthWallet", + description: null, + iconUrl, + deprecated, + available: true, + successUrl, + failureUrl, + walletUrl: resolveWalletUrl(moduleOptions.options.network, walletUrl), + }, + init: (options) => { + return FastAuthWallet({ + ...options, + params: { + walletUrl: resolveWalletUrl(options.options.network, walletUrl), + }, + }); + }, + }; + }; +} \ No newline at end of file diff --git a/src/lib/selector/fastAuthWalletConnection.ts b/src/lib/selector/fastAuthWalletConnection.ts new file mode 100644 index 000000000..d031c2b83 --- /dev/null +++ b/src/lib/selector/fastAuthWalletConnection.ts @@ -0,0 +1,263 @@ +import { serialize } from 'borsh'; +import type { InMemorySigner,keyStores, Near} from 'near-api-js'; +import { KeyPair, transactions } from 'near-api-js'; +const { SCHEMA } = transactions +import { ConnectedWalletAccount } from 'near-api-js'; + +const LOGIN_WALLET_URL_SUFFIX = '/login/'; +const LOCAL_STORAGE_KEY_SUFFIX = '_wallet_auth_key'; +const PENDING_ACCESS_KEY_PREFIX = 'pending_key'; // browser storage key for a pending access key (i.e. key has been generated but we are not sure it was added yet) + +interface SignInOptions { + contractId?: string; + methodNames?: string[]; + // TODO: Replace following with single callbackUrl + successUrl?: string; + failureUrl?: string; +} + +/** + * Information to send NEAR wallet for signing transactions and redirecting the browser back to the calling application + */ +interface RequestSignTransactionsOptions { + /** list of transactions to sign */ + transactions: transactions.Transaction[]; + /** url NEAR Wallet will redirect to after transaction signing is complete */ + callbackUrl?: string; + /** meta information NEAR Wallet will send back to the application. `meta` will be attached to the `callbackUrl` as a url search param */ + meta?: string; +} + +export class FastAuthWalletConnection { + /** @hidden */ + _walletBaseUrl: string; + + /** @hidden */ + _authDataKey: string; + + /** @hidden */ + _keyStore: keyStores.KeyStore; + + /** @hidden */ + _authData: { accountId?: string; allKeys?: string[] }; + + /** @hidden */ + _networkId: string; + + /** @hidden */ + // _near: Near; + _near: Near; + + /** @hidden */ + _connectedAccount: ConnectedWalletAccount; + + /** @hidden */ + _completeSignInPromise: Promise; + + /** @hidden */ + _iframe: HTMLIFrameElement; + + constructor(near: Near, appKeyPrefix: string) { + if (typeof(appKeyPrefix) !== 'string') { + throw new Error('Please define a clear appKeyPrefix for this WalletConnection instance as the second argument to the constructor'); + } + + if (typeof window === 'undefined') { + return new Proxy(this, { + get(target, property) { + if(property === 'isSignedIn') { + return () => false; + } + if(property === 'getAccountId') { + return () => ''; + } + if(target[property] && typeof target[property] === 'function') { + return () => { + throw new Error('No window found in context, please ensure you are using WalletConnection on the browser'); + }; + } + return target[property]; + } + }); + } + const iframe = document.createElement("iframe"); + iframe.style.display = "none"; + iframe.allow = "publickey-credentials-create *; publickey-credentials-get *; clipboard-write" + document.body.appendChild(iframe); + this._iframe = iframe; + this._near = near; + const authDataKey = appKeyPrefix + LOCAL_STORAGE_KEY_SUFFIX; + const authData = JSON.parse(window.localStorage.getItem(authDataKey) as string); + this._networkId = near.config.networkId; + this._walletBaseUrl = near.config.walletUrl; + appKeyPrefix = appKeyPrefix || near.config.contractName || 'default'; + this._keyStore = (near.connection.signer as InMemorySigner).keyStore; + this._authData = authData || { allKeys: [] }; + this._authDataKey = authDataKey; + if (!this.isSignedIn()) { + this._completeSignInPromise = this._completeSignInWithAccessKey(); + } + } + + /** + * Returns true, if this WalletConnection is authorized with the wallet. + * @example + * ```js + * const wallet = new WalletConnection(near, 'my-app'); + * wallet.isSignedIn(); + * ``` + */ + isSignedIn() { + return !!this._authData.accountId; + } + + /** + * Returns promise of completing signing in after redirecting from wallet + * @example + * ```js + * // on login callback page + * const wallet = new WalletConnection(near, 'my-app'); + * wallet.isSignedIn(); // false + * await wallet.isSignedInAsync(); // true + * ``` + */ + async isSignedInAsync() { + if (!this._completeSignInPromise) { + return this.isSignedIn(); + } + + await this._completeSignInPromise; + return this.isSignedIn(); + } + + /** + * Returns authorized Account ID. + * @example + * ```js + * const wallet = new WalletConnection(near, 'my-app'); + * wallet.getAccountId(); + * ``` + */ + getAccountId() { + return this._authData.accountId || ''; + } + + /** + * Redirects current page to the wallet authentication page. + * @param options An optional options object + * @param options.contractId The NEAR account where the contract is deployed + * @param options.successUrl URL to redirect upon success. Default: current url + * @param options.failureUrl URL to redirect upon failure. Default: current url + * + * @example + * ```js + * const wallet = new WalletConnection(near, 'my-app'); + * // redirects to the NEAR Wallet + * wallet.requestSignIn({ contractId: 'account-with-deploy-contract.near' }); + * ``` + */ + async requestSignIn({ contractId, methodNames, successUrl, failureUrl }: SignInOptions) { + const currentUrl = new URL(window.location.href); + const newUrl = new URL(this._walletBaseUrl + LOGIN_WALLET_URL_SUFFIX); + newUrl.searchParams.set('success_url', successUrl || currentUrl.href); + newUrl.searchParams.set('failure_url', failureUrl || currentUrl.href); + if (contractId) { + /* Throws exception if contract account does not exist */ + const contractAccount = await this._near.account(contractId); + await contractAccount.state(); + + newUrl.searchParams.set('contract_id', contractId); + const accessKey = KeyPair.fromRandom('ed25519'); + newUrl.searchParams.set('public_key', accessKey.getPublicKey().toString()); + await this._keyStore.setKey(this._networkId, PENDING_ACCESS_KEY_PREFIX + accessKey.getPublicKey(), accessKey); + } + + if (methodNames) { + methodNames.forEach(methodName => { + newUrl.searchParams.append('methodNames', methodName); + }); + } + + this._iframe.src = newUrl.toString(); + this._iframe.style.display = "block"; + } + + /** + * Requests the user to quickly sign for a transaction or batch of transactions by redirecting to the NEAR wallet. + */ + async requestSignTransactions({ transactions, meta, callbackUrl }: RequestSignTransactionsOptions): Promise { + const currentUrl = new URL(window.location.href); + const newUrl = new URL('sign', this._walletBaseUrl); + + newUrl.searchParams.set('transactions', transactions + .map(transaction => serialize(SCHEMA, transaction)) + .map(serialized => Buffer.from(serialized).toString('base64')) + .join(',')); + newUrl.searchParams.set('callbackUrl', callbackUrl || currentUrl.href); + if (meta) newUrl.searchParams.set('meta', meta); + + this._iframe.src = newUrl.toString(); + this._iframe.style.display = "block"; + } + + /** + * @hidden + * Complete sign in for a given account id and public key. To be invoked by the app when getting a callback from the wallet. + */ + async _completeSignInWithAccessKey() { + const currentUrl = new URL(window.location.href); + const publicKey = currentUrl.searchParams.get('public_key') || ''; + const allKeys = (currentUrl.searchParams.get('all_keys') || '').split(','); + const accountId = currentUrl.searchParams.get('account_id') || ''; + // TODO: Handle errors during login + if (accountId) { + const authData = { + accountId, + allKeys + }; + window.localStorage.setItem(this._authDataKey, JSON.stringify(authData)); + if (publicKey) { + await this._moveKeyFromTempToPermanent(accountId, publicKey); + } + this._authData = authData; + } + currentUrl.searchParams.delete('public_key'); + currentUrl.searchParams.delete('all_keys'); + currentUrl.searchParams.delete('account_id'); + currentUrl.searchParams.delete('meta'); + currentUrl.searchParams.delete('transactionHashes'); + + window.history.replaceState({}, document.title, currentUrl.toString()); + } + + /** + * @hidden + * @param accountId The NEAR account owning the given public key + * @param publicKey The public key being set to the key store + */ + async _moveKeyFromTempToPermanent(accountId: string, publicKey: string) { + const keyPair = await this._keyStore.getKey(this._networkId, PENDING_ACCESS_KEY_PREFIX + publicKey); + await this._keyStore.setKey(this._networkId, accountId, keyPair); + await this._keyStore.removeKey(this._networkId, PENDING_ACCESS_KEY_PREFIX + publicKey); + } + + /** + * Sign out from the current account + * @example + * walletConnection.signOut(); + */ + signOut() { + this._authData = {}; + window.localStorage.removeItem(this._authDataKey); + } + + /** + * Returns the current connected wallet account + */ + account() { + if (!this._connectedAccount) { + this._connectedAccount = new ConnectedWalletAccount(this, this._near.connection, this._authData.accountId as string); + } + return this._connectedAccount; + } +} \ No newline at end of file