From 86a063e31350bc6f647f70112f5479703f154711 Mon Sep 17 00:00:00 2001 From: Matt Date: Tue, 4 Apr 2023 14:52:32 -0700 Subject: [PATCH 1/6] wip: add fastauth-metamask project --- package.json | 1 + packages/fastauth-metamask/.eslintrc.json | 18 + packages/fastauth-metamask/.gitignore | 3 + packages/fastauth-metamask/README.md | 69 + packages/fastauth-metamask/jest.config.js | 14 + packages/fastauth-metamask/jest.config.ts | 15 + packages/fastauth-metamask/package.json | 27 + packages/fastauth-metamask/project.json | 54 + .../fastauth-metamask/src/assets/icons.ts | 1 + packages/fastauth-metamask/src/index.ts | 25 + .../src/lib/fastauth-metamask.ts | 178 +++ .../fastauth-metamask/src/lib/neth-lib.ts | 1297 +++++++++++++++++ packages/fastauth-metamask/tsconfig.json | 22 + packages/fastauth-metamask/tsconfig.lib.json | 10 + packages/fastauth-metamask/tsconfig.spec.json | 15 + packages/fastauth-metamask/yarn.lock | 653 +++++++++ workspace.json | 1 + 17 files changed, 2403 insertions(+) create mode 100644 packages/fastauth-metamask/.eslintrc.json create mode 100644 packages/fastauth-metamask/.gitignore create mode 100644 packages/fastauth-metamask/README.md create mode 100644 packages/fastauth-metamask/jest.config.js create mode 100644 packages/fastauth-metamask/jest.config.ts create mode 100644 packages/fastauth-metamask/package.json create mode 100644 packages/fastauth-metamask/project.json create mode 100644 packages/fastauth-metamask/src/assets/icons.ts create mode 100644 packages/fastauth-metamask/src/index.ts create mode 100644 packages/fastauth-metamask/src/lib/fastauth-metamask.ts create mode 100644 packages/fastauth-metamask/src/lib/neth-lib.ts create mode 100644 packages/fastauth-metamask/tsconfig.json create mode 100644 packages/fastauth-metamask/tsconfig.lib.json create mode 100644 packages/fastauth-metamask/tsconfig.spec.json create mode 100644 packages/fastauth-metamask/yarn.lock diff --git a/package.json b/package.json index 878ba3e71..944dc82db 100644 --- a/package.json +++ b/package.json @@ -51,6 +51,7 @@ "build:my-near-wallet": "nx run-many --target=build --projects=my-near-wallet --configuration=production", "build:sender": "nx run-many --target=build --projects=sender --configuration=production", "build:neth": "nx run-many --target=build --projects=neth --configuration=production", + "build:fastauth-metamask": "nx run-many --target=build --projects=fastauth-metamask --configuration=production", "build:nearfi": "nx run-many --target=build --projects=nearfi --configuration=production", "build:nightly": "nx run-many --target=build --projects=nightly --configuration=production", "build:meteor-wallet": "nx run-many --target=build --projects=meteor-wallet --configuration=production", diff --git a/packages/fastauth-metamask/.eslintrc.json b/packages/fastauth-metamask/.eslintrc.json new file mode 100644 index 000000000..9d9c0db55 --- /dev/null +++ b/packages/fastauth-metamask/.eslintrc.json @@ -0,0 +1,18 @@ +{ + "extends": ["../../.eslintrc.json"], + "ignorePatterns": ["!**/*"], + "overrides": [ + { + "files": ["*.ts", "*.tsx", "*.js", "*.jsx"], + "rules": {} + }, + { + "files": ["*.ts", "*.tsx"], + "rules": {} + }, + { + "files": ["*.js", "*.jsx"], + "rules": {} + } + ] +} diff --git a/packages/fastauth-metamask/.gitignore b/packages/fastauth-metamask/.gitignore new file mode 100644 index 000000000..c20a41dca --- /dev/null +++ b/packages/fastauth-metamask/.gitignore @@ -0,0 +1,3 @@ + +# dependencies +/node_modules diff --git a/packages/fastauth-metamask/README.md b/packages/fastauth-metamask/README.md new file mode 100644 index 000000000..dedafaab6 --- /dev/null +++ b/packages/fastauth-metamask/README.md @@ -0,0 +1,69 @@ +# @near-wallet-selector/fastauth-metamask + +This is the [fastauth-metamask](https://fastauth-metamask.app) package for NEAR Wallet Selector. + +## Installation and Usage + +The easiest way to use this package is to install it from the NPM registry, this package requires `near-api-js` v0.44.2 or above: + +```bash +# Using Yarn +yarn add near-api-js + +# Using NPM. +npm install near-api-js +``` +```bash +# Using Yarn +yarn add @near-wallet-selector/fastauth-metamask + +# Using NPM. +npm install @near-wallet-selector/fastauth-metamask +``` + +Then use it in your dApp: + +```ts +import { setupWalletSelector } from "@near-wallet-selector/core"; +import { setupfastauth-metamask } from "@near-wallet-selector/fastauth-metamask"; + +// fastauth-metamask for Wallet Selector can be setup without any params or it can take one optional param. +const fastauth-metamask = setupfastauth-metamask({ + // default fastauth-metamask icon included + iconUrl?: string; + // default 200 Tgas - for each fastauth-metamask transaction (bundling can include multiple "inner" transactions) + gas?: string; + // default false - cover screen with rgba(0, 0, 0, 0.5) mask while signing and awaiting transaction outcome + useModalCover?: boolean; + // default true - signAndSendTransactions will be bundled into 1 fastauth-metamask TX + bundle?: boolean, + // default false + deprecated?: boolean, +}); + +const selector = await setupWalletSelector({ + network: "testnet", + modules: [fastauth-metamask], +}); +``` + +## Options + +Setup options are described in comments above + +## Assets + +Assets such as icons can be found in the `/assets` directory of the package. Below is an example using Webpack: + +```ts +import { setupfastauth-metamask } from "@near-wallet-selector/fastauth-metamask"; +import { nearWalletIcon } from "@near-wallet-selector/fastauth-metamask/assets/icons"; + +const fastauth-metamask = setupfastauth-metamask({ + iconUrl: nearWalletIcon +}); +``` + +## License + +This repository is distributed under the terms of both the MIT license and the Apache License (Version 2.0). diff --git a/packages/fastauth-metamask/jest.config.js b/packages/fastauth-metamask/jest.config.js new file mode 100644 index 000000000..8cb98f49c --- /dev/null +++ b/packages/fastauth-metamask/jest.config.js @@ -0,0 +1,14 @@ +module.exports = { + displayName: "fastauth-metamask", + preset: "../../jest.preset.js", + globals: { + "ts-jest": { + tsconfig: "/tsconfig.spec.json", + }, + }, + transform: { + "^.+\\.[tj]sx?$": "ts-jest", + }, + moduleFileExtensions: ["ts", "tsx", "js", "jsx"], + coverageDirectory: "../../coverage/packages/fastauth-metamask", +}; diff --git a/packages/fastauth-metamask/jest.config.ts b/packages/fastauth-metamask/jest.config.ts new file mode 100644 index 000000000..9c3d812d4 --- /dev/null +++ b/packages/fastauth-metamask/jest.config.ts @@ -0,0 +1,15 @@ +/* eslint-disable */ +export default { + displayName: "fastauth-metamask", + preset: "../../jest.preset.js", + globals: { + "ts-jest": { + tsconfig: "/tsconfig.spec.json", + }, + }, + transform: { + "^.+\\.[tj]s$": "ts-jest", + }, + moduleFileExtensions: ["ts", "tsx", "js", "jsx"], + coverageDirectory: "../../coverage/packages/fastauth-metamask", +}; diff --git a/packages/fastauth-metamask/package.json b/packages/fastauth-metamask/package.json new file mode 100644 index 000000000..1af84fbd6 --- /dev/null +++ b/packages/fastauth-metamask/package.json @@ -0,0 +1,27 @@ +{ + "name": "@near-wallet-selector/fastauth-metamask", + "version": "7.9.2", + "description": "Control NEAR accounts with ETH accounts", + "author": "mattlockyer", + "keywords": [ + "near", + "blockchain", + "wallets", + "dapps", + "near-protocol", + "near-blockchain", + "wallet selector", + "fastauth-metamask" + ], + "repository": { + "type": "git", + "url": "https://github.com/near/wallet-selector.git" + }, + "bugs": { + "url": "https://github.com/near/wallet-selector/issues" + }, + "homepage": "https://github.com/near/wallet-selector/tree/main/packages/fastauth-metamask", + "peerDependencies": { + "near-api-js": "^0.44.2 || ^1.0.0" + } +} diff --git a/packages/fastauth-metamask/project.json b/packages/fastauth-metamask/project.json new file mode 100644 index 000000000..89c80e5fb --- /dev/null +++ b/packages/fastauth-metamask/project.json @@ -0,0 +1,54 @@ +{ + "$schema": "../../node_modules/nx/schemas/project-schema.json", + "sourceRoot": "packages/fastauth-metamask/src", + "projectType": "library", + "targets": { + "build": { + "executor": "@nrwl/web:rollup", + "outputs": ["{options.outputPath}"], + "options": { + "outputPath": "dist/packages/fastauth-metamask", + "tsConfig": "packages/fastauth-metamask/tsconfig.lib.json", + "project": "packages/fastauth-metamask/package.json", + "entryFile": "packages/fastauth-metamask/src/index.ts", + "buildableProjectDepsInPackageJsonType": "dependencies", + "compiler": "babel", + "format": ["esm", "cjs"], + "assets": [ + { + "glob": "packages/fastauth-metamask/README.md", + "input": ".", + "output": "." + }, + { + "glob": "packages/fastauth-metamask/assets/*", + "input": ".", + "output": "assets" + } + ] + } + }, + "lint": { + "executor": "@nrwl/linter:eslint", + "outputs": ["{options.outputFile}"], + "options": { + "lintFilePatterns": ["packages/fastauth-metamask/**/*.ts"] + } + }, + "test": { + "executor": "@nrwl/jest:jest", + "outputs": ["coverage/packages/fastauth-metamask"], + "options": { + "jestConfig": "packages/fastauth-metamask/jest.config.ts", + "passWithNoTests": true + } + }, + "deploy": { + "executor": "ngx-deploy-npm:deploy", + "options": { + "access": "public" + } + } + }, + "tags": ["injected-wallet"] +} diff --git a/packages/fastauth-metamask/src/assets/icons.ts b/packages/fastauth-metamask/src/assets/icons.ts new file mode 100644 index 000000000..3c03fb11a --- /dev/null +++ b/packages/fastauth-metamask/src/assets/icons.ts @@ -0,0 +1 @@ +export const FastAuthMetaMaskIcon = `` \ No newline at end of file diff --git a/packages/fastauth-metamask/src/index.ts b/packages/fastauth-metamask/src/index.ts new file mode 100644 index 000000000..208e5828c --- /dev/null +++ b/packages/fastauth-metamask/src/index.ts @@ -0,0 +1,25 @@ +export type { FastAuthMetaMaskParams } from "./lib/fastauth-metamask"; + +export { setupFastAuthMetaMask } from "./lib/fastauth-metamask"; +export { + getNear, + getConnection, + getEthereum, + handleCreate, + accountExists, + switchEthereum, + handleCheckAccount, + handleCancelFunding, + verifyOwner, + getNearMap, + handleDisconnect, + handleUpdateContract, + handleRefreshAppKey, + hasAppKey, + signIn, + signOut, + isSignedIn, + signAndSendTransactions, + initConnection, + MIN_NEW_ACCOUNT_ASK, +} from "./lib/neth-lib"; diff --git a/packages/fastauth-metamask/src/lib/fastauth-metamask.ts b/packages/fastauth-metamask/src/lib/fastauth-metamask.ts new file mode 100644 index 000000000..11d59ca25 --- /dev/null +++ b/packages/fastauth-metamask/src/lib/fastauth-metamask.ts @@ -0,0 +1,178 @@ +import type { + WalletModuleFactory, + InjectedWallet, + Action, + FunctionCallAction, + WalletBehaviourFactory, +} from "@near-wallet-selector/core"; +import detectEthereumProvider from "@metamask/detect-provider"; +import { FastAuthMetaMaskIcon } from "../assets/icons"; +import { + getNear, + signIn, + signOut, + verifyOwner, + isSignedIn, + signAndSendTransactions, + initConnection, + METAMASK_URL, +} from "./neth-lib"; +import isMobile from "is-mobile"; +export { initConnection } from "./neth-lib"; + +declare global { + interface Window { + contractPath: string | null; + ethereum: { + chainId: string; + // eslint-disable-next-line + request: any; + }; + } +} + +export interface FastAuthMetaMaskParams { + // default FastAuthMetaMask icon included + iconUrl?: string; + // default false + deprecated?: boolean; +} + +const isInstalled = async () => { + await detectEthereumProvider({ timeout: 100 }); + return !!window.ethereum; +}; + +const FastAuthMetaMask: WalletBehaviourFactory = async ({ + metadata, + logger, + store, + options, + provider, +}) => { + + const isValidActions = ( + actions: Array + ): actions is Array => { + return actions.every((x) => x.type === "FunctionCall"); + }; + + const transformActions = (actions: Array) => { + const validActions = isValidActions(actions); + + if (!validActions) { + throw new Error( + `Only 'FunctionCall' actions types are supported by ${metadata.name}` + ); + } + + return actions.map((x) => x.params); + }; + + const signTransactions = async (transactions) => { + logger.log("FastAuthMetaMask:signAndSendTransactions", { transactions }); + + const { contract } = store.getState(); + + if (!(await isSignedIn()) || !contract) { + throw new Error("Wallet not signed in"); + } + + const transformedTxs = transactions.map(({ receiverId, actions }) => ({ + receiverId: receiverId || contract.contractId, + actions: transformActions(actions), + })); + + let res; + try { + res = await signAndSendTransactions({ + transactions: transformedTxs, + }); + } catch (e) { + /// "user rejected signing" or near network error + logger.log("FastAuthMetaMask:signAndSendTransactions Error", e); + throw e; + } + + return res; + }; + + // return the wallet interface for wallet-selector + return { + async signIn() { + let account; + try { + account = await signIn(); + if (!account) { + return []; + } + } catch (e) { + if (!/not connected/.test((e as object).toString())) { + throw e; + } + // console.log(e); + } + return [account]; + }, + + async signOut() { + await signOut(); + }, + + async verifyOwner({ message }) { + logger.log("FastAuthMetaMask:verifyOwner", { message }); + return verifyOwner({ message, provider, account: null }); + }, + + async getAccounts() { + const { accountId, account } = await getNear(); + return [ + { + accountId, + publicKey: ( + await account.connection.signer.getPublicKey( + account.accountId, + options.network.networkId + ) + ).toString(), + }, + ]; + }, + + signAndSendTransaction: async ({ receiverId, actions }) => + signTransactions([{ receiverId, actions }]), + + signAndSendTransactions: async ({ transactions }) => + signTransactions(transactions), + }; +}; + +export function setupFastAuthMetaMask({ + iconUrl = FastAuthMetaMaskIcon, + deprecated = false, +}: FastAuthMetaMaskParams = {}): WalletModuleFactory { + return async () => { + + const mobile = isMobile(); + if (mobile) { + return null; + } + + const installed = await isInstalled(); + + return { + id: "fastauth-metamask", + type: "injected", + metadata: { + name: "MetaMask", + description: null, + iconUrl, + downloadUrl: METAMASK_URL, + deprecated: false, + available: installed, + }, + deprecated, + init: FastAuthMetaMask, + }; + }; +} diff --git a/packages/fastauth-metamask/src/lib/neth-lib.ts b/packages/fastauth-metamask/src/lib/neth-lib.ts new file mode 100644 index 000000000..2e8d454d3 --- /dev/null +++ b/packages/fastauth-metamask/src/lib/neth-lib.ts @@ -0,0 +1,1297 @@ +/*eslint @typescript-eslint/no-use-before-define: 1*/ +/*eslint @typescript-eslint/no-explicit-any: 1*/ +/*eslint @typescript-eslint/naming-convention: 1*/ +// @ts-nocheck + +import { ethers } from "ethers"; +import detectEthereumProvider from "@metamask/detect-provider"; +import * as nearAPI from "near-api-js"; +import { generateSeedPhrase } from "near-seed-phrase"; +import BN from "bn.js"; +import type { + KeyPair as NearKeyPair, + Account as NearAccount, +} from "near-api-js"; + +const { + Near, + Account, + KeyPair, + keyStores: { BrowserLocalStorageKeyStore }, + transactions: { addKey, deleteKey, functionCallAccessKey }, + utils: { + PublicKey, + format: { parseNearAmount }, + }, +} = nearAPI; + +export const METAMASK_URL = "https://chrome.google.com/webstore/detail/metamask/nkbihfbeogaeaoehlefnkodbefgpgknn"; +export const PREV_NETH_SITE_URL = "neardefi.github.io/neth"; + +const NETWORK = { + testnet: { + FUNDING_ACCOUNT_ID: "neth.testnet", + MAP_ACCOUNT_ID: "map.neth.testnet", + ROOT_ACCOUNT_ID: "testnet", + }, + mainnet: { + MAP_ACCOUNT_ID: "nethmap.near", + ROOT_ACCOUNT_ID: "near", + }, +}; + +const WS_STORAGE_NAMESPACE = "near-wallet-selector:neth:"; +const REFRESH_MSG = `Please refresh the page and try again.`; +const TX_ARGS_ATTEMPT = "__TX_ARGS_ATTEMPT"; +const ATTEMPT_SECRET_KEY = "__ATTEMPT_SECRET_KEY"; +const ATTEMPT_ACCOUNT_ID = "__ATTEMPT_ACCOUNT_ID"; +const ATTEMPT_ETH_ADDRESS = "__ATTEMPT_ETH_ADDRESS"; +const APP_KEY_SECRET = "__APP_KEY_SECRET"; +const APP_KEY_ACCOUNT_ID = "__APP_KEY_ACCOUNT_ID"; +const defaultGas = "200000000000000"; +const halfGas = "50000000000000"; +/// this is the new account amount 0.21 for account name, keys, contract and 0.01 for mapping contract storage cost +const MIN_NEW_ACCOUNT = parseNearAmount("0.4"); +const MIN_NEW_ACCOUNT_THRESH = parseNearAmount("0.49"); +export const MIN_NEW_ACCOUNT_ASK = parseNearAmount("0.5"); +const FUNDING_CHECK_TIMEOUT = 5000; +/// lkmfawl + +const attachedDepositMapping = parseNearAmount("0.05"); + +/// Helpers +const defaultStorage = (prefix = "") => ({ + getItem: (k) => { + const v = localStorage.getItem(prefix + k); + if (v?.charAt(0) !== "{") { + return v; + } + try { + return JSON.parse(v); + } catch (e) { + // logger.log(e); + } + }, + setItem: (k, v) => + localStorage.setItem( + prefix + k, + typeof v === "string" ? v : JSON.stringify(v) + ), + removeItem: (k) => localStorage.removeItem(prefix + k), +}); +const defaultLogger = () => ({ + // eslint-disable-next-line + log: (args) => console.log(...args), +}); + +/// NEAR setup +let near, + gas, + keyStore, + logger, + storage, + connection, + networkId, + contractAccount, + accountSuffix; + +export const initConnection = ({ + network, + gas: _gas = defaultGas, + logger: _logger = defaultLogger(), + storage: _storage = defaultStorage(), +}) => { + gas = _gas; + logger = _logger; + storage = _storage; + + keyStore = new BrowserLocalStorageKeyStore(); + near = new Near({ + ...network, + keyStore, + }); + connection = near.connection; + networkId = network.networkId; + contractAccount = new Account( + connection, + networkId === "mainnet" ? "near" : networkId + ); + accountSuffix = networkId === "mainnet" ? ".near" : "." + networkId; + + const cover = document.createElement("div"); + cover.style.display = "none"; + cover.style.width = "100%"; + cover.style.height = "100vh"; + cover.style.zIndex = "999999"; + cover.style.position = "fixed"; + cover.style.top = "0"; + cover.style.background = "rgba(0, 0, 0, 0.5)"; + document.body.appendChild(cover); + + /// recovery from unbundled TXs that haven't been broadcast yet + broadcastTXs(); + + return cover; +}; + +export const getConnection = () => { + return { + near, + connection, + keyStore, + networkId, + contractAccount, + accountSuffix, + }; +}; + +/// helpers + +export const accountExists = async (accountId, ethAddress = null) => { + try { + const account = new nearAPI.Account(connection, accountId); + await account.state(); + + if (ethAddress) { + const mapAccountId = await getNearMap(ethAddress); + if (mapAccountId) { + return true; + } + } + + return true; + } catch (e: any) { + if (!/no such file|does not exist/.test(e.toString())) { + throw e; + } + return false; + } +}; + +const buf2hex = (buf) => ethers.utils.hexlify(buf).substring(2); +const pub2hex = (publicKey) => + ethers.utils.hexlify(PublicKey.fromString(publicKey).data).substring(2); + +/// account creation and connection flow + +export const handleCreate = async ( + signer, + ethAddress, + newAccountId, + fundingAccountCB, + fundingErrorCB, + postFundingCB +) => { + if ( + (networkId === "testnet" && newAccountId.indexOf(".near") > -1) || + (networkId === "mainnet" && newAccountId.indexOf(".testnet") > -1) + ) { + return alert( + "Invalid account name. You do not need to add any .near or .testnet. Please try again." + ); + } + + /// get keypair from eth sig entropy for the near-eth account + const { publicKey: fundingAccountPubKey, secretKey: new_secret_key } = + await keyPairFromEthSig(signer, fundingKeyPayload()); + /// store attempt in localStorage so we can recover and retry / resume contract deployment + await storage.setItem(ATTEMPT_ACCOUNT_ID, newAccountId); + await storage.setItem(ATTEMPT_SECRET_KEY, new_secret_key); + await storage.setItem(ATTEMPT_ETH_ADDRESS, ethAddress); + + return await createAccount({ + signer, + newAccountId, + fundingAccountPubKey, + fundingAccountCB, + fundingErrorCB, + postFundingCB, + }); +}; + +const createAccount = async ({ + signer, + newAccountId, + fundingAccountPubKey, + fundingAccountCB, + fundingErrorCB, + postFundingCB, +}) => { + // const { publicKey, secretKey } = parseSeedPhrase(process.env.REACT_APP_FUNDING_SEED_PHRASE); + /// assumes implicit is funded, otherwise will warn and cycle here + + const implicitAccountId = Buffer.from( + PublicKey.from(fundingAccountPubKey).data + ).toString("hex"); + if (fundingAccountCB) { + fundingAccountCB(implicitAccountId); + } + + /// wait for implicit funding here and then continue to createAccount + const checkImplicitFunded = async () => { + logger.log("checking for funding of implicit account", implicitAccountId); + const account = new Account(connection, implicitAccountId); + try { + const balance = await account.getAccountBalance(); + const { available } = balance; + const diff = new BN(available).sub( + new BN(MIN_NEW_ACCOUNT_THRESH as string) + ); + if (diff.lt(new BN("0"))) { + // alert(`There is not enough NEAR (${formatNearAmount(MIN_NEW_ACCOUNT_ASK, 4)} minimum) to create a new account and deploy NETH contract. Please deposit more and try again.`) + if (fundingErrorCB) { + fundingErrorCB(implicitAccountId, diff.abs().toString()); + } + await new Promise((r) => setTimeout(r, FUNDING_CHECK_TIMEOUT)); + return await checkImplicitFunded(); + } + } catch (e: any) { + if (!/does not exist/gi.test(e.toString())) { + throw e; + } + logger.log("not funded, checking again"); + await new Promise((r) => setTimeout(r, FUNDING_CHECK_TIMEOUT)); + return await checkImplicitFunded(); + } + return true; + }; + /// if not funded properly, return and reload + if (!(await checkImplicitFunded())) { + return window.location.reload(); + } + logger.log("implicit account funded", implicitAccountId); + + if (postFundingCB) { + postFundingCB(); + } + + const { account, ethAddress } = await setupFromStorage(implicitAccountId); + + /// final checks, last chance to cancel funding if they fail + if (await accountExists(newAccountId, ethAddress)) { + alert(`${newAccountId} already exists. Please try another.`); + return await handleCancelFunding(implicitAccountId); + } + + /// create account now + + /// get keypair from eth sig entropy for the near-eth account + const { publicKey: new_public_key, secretKey: new_secret_key } = + await keyPairFromEthSig( + signer, + unlimitedKeyPayload(newAccountId, ethAddress) + ); + await storage.setItem(ATTEMPT_SECRET_KEY, new_secret_key); + // remove any existing app key + await storage.removeItem(APP_KEY_ACCOUNT_ID); + await storage.removeItem(APP_KEY_SECRET); + + try { + await account.functionCall({ + contractId: NETWORK[networkId].ROOT_ACCOUNT_ID, + methodName: "create_account", + args: { + new_account_id: newAccountId, + new_public_key, + }, + gas, + attachedDeposit: new BN(MIN_NEW_ACCOUNT as string), + }); + } catch (e) { + if (!/be created by/.test(JSON.stringify(e))) { + throw e; + } + return handleCancelFunding(implicitAccountId); + } + /// check + if (!(await accountExists(newAccountId))) { + return logger.log( + `Account ${newAccountId} could NOT be created. Please refresh the page and try again.` + ); + } + logger.log(`Account ${newAccountId} created successfully.`); + /// drain implicit + await account.deleteAccount(newAccountId); + + return await handleMapping(); +}; + +export const handleCancelFunding = async (fundingAccountId) => { + const { account } = await setupFromStorage(fundingAccountId); + const refundAccountId = window.prompt( + `There was an error creating the account. You need to refund and try again. Please enter the account you funded from. MAKE SURE IT IS CORRECT. THIS CANNOT BE UNDONE.` + ); + /// drain implicit + try { + await account.deleteAccount(refundAccountId as string); + } catch (e) { + logger.log("Cannot delete implicit"); + } finally { + /// delete attempt + await storage.removeItem(ATTEMPT_ACCOUNT_ID); + await storage.removeItem(ATTEMPT_SECRET_KEY); + await storage.removeItem(ATTEMPT_ETH_ADDRESS); + } +}; + +export const handleMapping = async () => { + const { account, ethAddress } = await setupFromStorage(); + try { + await account.functionCall({ + contractId: NETWORK[networkId].MAP_ACCOUNT_ID, + methodName: "set", + args: { eth_address: ethAddress }, + gas, + attachedDeposit: new BN(attachedDepositMapping as string), + }); + logger.log(`Account mapping successful`); + } catch (e) { + logger.log(e); + return logger.log(`Account mapping failed`); + } + return await handleDeployContract(); +}; + +export const handleDeployContract = async () => { + const { account } = await setupFromStorage(); + + const contractPath = window?.contractPath; + // logger.log(contractPath) + const ab = await fetch(contractPath as string).then((res) => + res.arrayBuffer() + ); + const contractBytes = new Uint8Array(ab); + // logger.log("contractBytes.length", contractBytes.length); + try { + await account.deployContract(contractBytes); + logger.log(`Contract deployed successfully.`); + } catch (e) { + logger.log(e); + return logger.log(`Contract deployment failed. ${REFRESH_MSG}`); + } + + return await handleSetupContract(); +}; + +export const handleSetupContract = async () => { + const { account, ethAddress } = await setupFromStorage(); + + try { + await account.functionCall({ + contractId: account.accountId, + methodName: "setup", + args: { eth_address: ethAddress }, + gas, + }); + logger.log(`Contract setup successfully.`); + } catch (e) { + logger.log(e); + return logger.log(`Contract setup failed. ${REFRESH_MSG}`); + } + + return await handleKeys(); +}; + +export const handleKeys = async () => { + const { account, newAccountId, ethAddress } = await setupFromStorage(); + const accessKeys = await account.getAccessKeys(); + // keys are done + if ( + accessKeys.length !== 1 || + accessKeys[0]?.access_key?.permission !== "FullAccess" + ) { + return; + } + const publicKey = PublicKey.from(accessKeys[0].public_key); + const actions = [ + // delete the full access key + deleteKey(publicKey), + // limited to execute, unlimited allowance + addKey(publicKey, functionCallAccessKey(newAccountId, ["execute"])), + ]; + try { + const res = await (account as any).signAndSendTransaction({ + receiverId: newAccountId, + actions, + }); + if (res?.status?.SuccessValue !== "") { + return logger.log(`Key rotation failed. ${REFRESH_MSG}`); + } + logger.log(`Key rotation successful.`); + } catch (e) { + logger.log(e); + return logger.log(`Key rotation failed. ${REFRESH_MSG}`); + } + return await handleCheckAccount({ ethAddress }); +}; + +/// waterfall check everything about account and fill in missing pieces + +export const handleCheckAccount = async ({ + signer = null, + ethAddress = null, + fundingAccountCB = null, + fundingErrorCB = null, + postFundingCB = null, +}) => { + const setup = await setupFromStorage(); + let { newAccountId } = setup; + const { newSecretKey } = setup; + + const mapAccountId = await getNearMap(ethAddress); + if (!mapAccountId) { + // alert("create account first"); + logger.log("No account mapping exists."); + } else { + newAccountId = mapAccountId; + } + + logger.log("Checking account created."); + if (!(await accountExists(newAccountId))) { + const keyPair = KeyPair.fromString(newSecretKey); + return createAccount({ + signer, + newAccountId, + fundingAccountPubKey: keyPair.getPublicKey().toString(), + fundingAccountCB, + fundingErrorCB, + postFundingCB, + }); + } + + const account = new Account(connection, newAccountId); + + logger.log("Checking account address mapping."); + const mapRes = await account.viewFunction( + NETWORK[networkId].MAP_ACCOUNT_ID, + "get_eth", + { + account_id: newAccountId, + } + ); + if (mapRes === null) { + return handleMapping(); + } + + logger.log("Checking contract deployed."); + const state = await account.state(); + if (state.code_hash === "11111111111111111111111111111111") { + return handleDeployContract(); + } + + logger.log("Checking contract setup."); + try { + const ethRes = await account.viewFunction(newAccountId, "get_address"); + // any reason the address wasn't set properly + if (!ethRes || !ethRes.length) { + return handleSetupContract(); + } + } catch (e) { + // not set at all (wasm error unreachable storage value) + logger.log(e); + return handleSetupContract(); + } + + logger.log("Checking access keys."); + const accessKeys = await account.getAccessKeys(); + if ( + accessKeys.length === 1 && + accessKeys[0]?.access_key?.permission === "FullAccess" + ) { + return handleKeys(); + } + + logger.log("Account created."); + logger.log("Contract deployed and setup."); + logger.log("Mapping added."); + logger.log("Keys rotated."); + + await storage.removeItem(ATTEMPT_ACCOUNT_ID); + await storage.removeItem(ATTEMPT_SECRET_KEY); + await storage.removeItem(ATTEMPT_ETH_ADDRESS); + + return { account }; +}; + +/// on same domain as setup + +export const hasAppKey = (accessKeys) => + accessKeys.some((k) => { + const functionCallPermission = k?.access_key?.permission?.FunctionCall; + return ( + functionCallPermission.allowance !== null && + functionCallPermission.method_names[0] === "execute" + ); + }); + +export const handleRefreshAppKey = async (signer, ethAddress) => { + const { account, accountId } = await getUnlimitedKeyAccount( + signer, + ethAddress + ); + + // now refresh app key + const nonce = parseInt( + await account.viewFunction(accountId, "get_nonce"), + 16 + ).toString(); + // new public key based on current nonce which will become the app_key_nonce in contract after this TX + const { publicKey, secretKey } = await keyPairFromEthSig( + signer, + appKeyPayload(accountId, nonce) + ); + // logger.log(publicKey); + const public_key = pub2hex(publicKey); + const actions: Array = [ + { + type: "AddKey", + public_key, + allowance: parseNearAmount("1"), + receiver_id: accountId, + method_names: "execute", + }, + ]; + /// check keys, find old app key, delete that first + const accessKeys = await account.getAccessKeys(); + if (hasAppKey(accessKeys)) { + // old public key based on current app_key_nonce + const appKeyNonce = parseInt( + await account.viewFunction(accountId, "get_app_key_nonce"), + 16 + ).toString(); + const { publicKey: oldPublicKey } = await keyPairFromEthSig( + signer, + appKeyPayload(accountId, appKeyNonce) + ); + const oldPublicKeyHex = pub2hex(oldPublicKey); + actions.unshift({ + type: "DeleteKey", + public_key: oldPublicKeyHex, + }); + } + /// get args for execute call + const args = await ethSignJson(signer, { + nonce, + receivers: [accountId], + transactions: [ + { + actions, + }, + ], + }); + const res = await account.functionCall({ + contractId: accountId, + methodName: "execute", + args, + gas, + }); + + if (res?.status?.SuccessValue !== "") { + return logger.log(`App key rotation unsuccessful. ${REFRESH_MSG}`); + } + await storage.removeItem(APP_KEY_SECRET); + await storage.removeItem(APP_KEY_ACCOUNT_ID); + return { publicKey: public_key, secretKey }; +}; + +export const handleUpdateContract = async (signer, ethAddress) => { + const { account, accountId } = await getUnlimitedKeyAccount( + signer, + ethAddress + ); + + const contractPath = window?.contractPath; + const ab = await fetch(contractPath as string).then((res) => + res.arrayBuffer() + ); + const contractBytes = new Uint8Array(ab); + const actions = [ + { + type: "DeployContract", + code: buf2hex(contractBytes), + }, + ]; + const nonce = parseInt( + await account.viewFunction(accountId, "get_nonce"), + 16 + ).toString(); + const args = await ethSignJson(signer, { + nonce, + receivers: [accountId], + transactions: [ + { + actions, + }, + ], + }); + const res = await account.functionCall({ + contractId: accountId, + methodName: "execute", + args, + gas, + }); + if (res?.status?.SuccessValue !== "") { + return logger.log(`Redeply contract unsuccessful. ${REFRESH_MSG}`); + } +}; + +/// account disconnecting flow + +export const handleDisconnect = async (signer, ethAddress) => { + const { account, accountId, secretKey } = await getUnlimitedKeyAccount( + signer, + ethAddress + ); + + const { + seedPhrase, + publicKey, + secretKey: newSecretKey, + } = generateSeedPhrase(); + const _seedPhrase = window.prompt( + "Copy this down and keep it safe!!! This is your new seed phrase!!!", + seedPhrase + ); + if (seedPhrase !== _seedPhrase) { + return alert( + "There was an error copying seed phrase. Nothing has been done. Please try again." + ); + } + const oldUnlimitedKey = KeyPair.fromString(secretKey); + + const actions = [ + { + type: "DeleteKey", + public_key: pub2hex(oldUnlimitedKey.getPublicKey().toString()), + }, + { + type: "AddKey", + public_key: pub2hex(publicKey), + // special case will add full access key + allowance: "0", + }, + { + type: "FunctionCall", + method_name: "remove_storage", + args: "", + amount: "0", + gas: halfGas, + }, + { + type: "DeployContract", + code: "", + }, + ]; + /// check keys, find old app key, delete that first + const accessKeys = await account.getAccessKeys(); + if ( + accessKeys.some((k) => { + const functionCallPermission = k?.access_key?.permission?.FunctionCall; + return ( + functionCallPermission?.allowance !== null && + functionCallPermission?.method_names[0] === "execute" + ); + }) + ) { + const appKeyNonce = parseInt( + await account.viewFunction(accountId, "get_app_key_nonce"), + 16 + ).toString(); + const { publicKey: oldPublicKey } = await keyPairFromEthSig( + signer, + appKeyPayload(accountId, appKeyNonce) + ); + const oldPublicKeyHex = pub2hex(oldPublicKey); + actions.unshift({ + type: "DeleteKey", + public_key: oldPublicKeyHex, + }); + } + + /// get args for execute call + const nonce = parseInt( + await account.viewFunction(accountId, "get_nonce"), + 16 + ).toString(); + const args = await ethSignJson(signer, { + nonce, + receivers: [accountId], + transactions: [ + { + actions, + }, + ], + }); + const res = await account.functionCall({ + contractId: accountId, + methodName: "execute", + args, + gas, + }); + + if (res?.status?.SuccessValue !== "") { + return logger.log("app key rotation unsuccessful"); + } + + // remove the mapping (can do this later if user has FAK) + + keyStore.setKey(networkId, accountId, newSecretKey); + try { + const mapRes = await account.functionCall({ + contractId: NETWORK[networkId].MAP_ACCOUNT_ID, + methodName: "del", + args: {}, + gas, + }); + logger.log(mapRes); + if (mapRes?.status?.SuccessValue !== "") { + logger.log("account mapping removal failed"); + } + } catch (e) { + logger.log(e); + } + + return { account }; +}; + +/// helpers for account creation and connection domain + +const setupFromStorage = async (accountId = "") => { + const newAccountId = + accountId.length > 0 + ? accountId + : await storage.getItem(ATTEMPT_ACCOUNT_ID); + const newSecretKey = await storage.getItem(ATTEMPT_SECRET_KEY); + const ethAddress = await storage.getItem(ATTEMPT_ETH_ADDRESS); + const account = new Account(connection, newAccountId); + let keyPair; + if (newSecretKey) { + keyPair = KeyPair.fromString(newSecretKey); + keyStore.setKey(networkId, newAccountId, keyPair); + } + return { newAccountId, newSecretKey, ethAddress, account, keyPair }; +}; + +const getUnlimitedKeyAccount = async ( + signer, + ethAddress, + tryPrevUrl = false +) => { + let accountId, + secretKey = await storage.getItem(ATTEMPT_SECRET_KEY); + // if unlimited allowance access key is not in localStorage user will have to sign to generate it + if (!secretKey) { + // TODO remove dep on near-utils + // use any random near account to check mapping + accountId = await getNearMap(ethAddress); + const { secretKey: _secretKey } = await keyPairFromEthSig( + signer, + unlimitedKeyPayload(accountId, tryPrevUrl) + ); + secretKey = _secretKey; + } else { + accountId = await storage.getItem(ATTEMPT_ACCOUNT_ID); + } + const account = new Account(connection, accountId); + const keyPair = KeyPair.fromString(secretKey); + const publicKey = keyPair.getPublicKey().toString(); + /// check if access key matches + const accessKeys = await account.getAccessKeys(); + if (!accessKeys.some(({ public_key }) => publicKey === public_key)) { + return await getUnlimitedKeyAccount(signer, ethAddress, true); + } + + keyStore.setKey(networkId, accountId, keyPair); + return { account, accountId, secretKey }; +}; + +/** + * The access key payloads, unlimited and limited + */ + +const appKeyPayload = (accountId, appKeyNonce) => ({ + WARNING: `Creating key for: ${accountId}`, + nonce: appKeyNonce, + description: `ONLY sign this on apps you trust! This key CAN use up to 1 N for transactions.`, +}); + +const unlimitedKeyPayload = (accountId, tryPrevUrl) => ({ + WARNING: `Creates a key with access to your (new) paired NEAR Account: ${accountId}`, + description: `ONLY sign this message on this website: ${ + tryPrevUrl ? PREV_NETH_SITE_URL : NETH_SITE_URL + }`, +}); + +const fundingKeyPayload = () => ({ + WARNING: `This creates a full access key in your localStorage to a funding account you will be sending NEAR to.`, + description: `ONLY sign this message on this website: ${NETH_SITE_URL}`, +}); + +/** + * main domain, types and eth signTypedData method + */ + +const domain = { + name: "NETH", + version: "1", + // chainId: 1, // aurora + chainId: 1313161554, // aurora +}; + +const HEADER_OFFSET = "NETH"; +const HEADER_PAD = 8; +const RECEIVER_MARKER = "|~-_NETH~-_-~RECEIVER_-~|"; +const PREFIX = "|NETH_"; +const SUFFIX = "_NETH|"; + +const pack = (elements) => + elements + .map((el) => { + const str = + typeof el === "string" + ? el + : Object.entries(el) + .map( + ([k, v]) => + `${PREFIX}${k}:${ + typeof v === "string" ? v : JSON.stringify(v) + }${SUFFIX}` + ) + .join(""); + + const len = str.length.toString().padStart(HEADER_PAD, "0"); + + return HEADER_OFFSET + len + "__" + str; + }) + .join(""); + +const ethSignJson = async (signer, json) => { + const Transaction: Array = []; + const types = { Transaction }; + Object.entries(json).forEach(([k]) => { + types.Transaction.push({ + type: "string", + name: k, + }); + }); + /// convenience for devs so they can pass in JSON + + /// hoist any functionCall args containing receiver|account in their key to top level receivers + /// replaces value with marker, contract fills in marker + + if (json.transactions) { + Object.values(json.transactions).forEach((tx: any, i) => { + tx.actions.forEach((action) => { + if (!action.args) { + return; + } + if (Buffer.isBuffer(action.args)) { + action.args = "0x" + action.args.toString("hex"); + return; + } + Object.entries(action.args).forEach(([key, value]) => { + /// TODO include check on value to determine valid account_id to be replaced + + if (/receiver_id|account_id/g.test(key)) { + action.args[key] = RECEIVER_MARKER; + json.receivers.splice(i + 1, 0, value); + } + }); + }); + }); + + json.transactions = pack( + json.transactions.map(({ actions }) => pack(actions)) + ); + } + if (json.receivers) { + const numReceivers = json.receivers.length.toString(); + json.receivers = + HEADER_OFFSET + + json.receivers.join(",").length.toString().padStart(HEADER_PAD, "0") + + "__" + + json.receivers.join(","); + json.receivers = + json.receivers.substring(0, 4) + + numReceivers.padStart(3, "0") + + json.receivers.substring(7); + } + + const sig = await signer._signTypedData(domain, types, json); + + const args = { + sig, + msg: json, + }; + // logger.log('\nargs\n', JSON.stringify(args, null, 4), '\n'); + return args; +}; + +const keyPairFromEthSig = async (signer, json) => { + const { sig } = await ethSignJson(signer, json); + const sigHash = ethers.utils.id(sig); + /// use 32 bytes of entropy from hash of signature to create NEAR keyPair + return generateSeedPhrase(sigHash.substring(2, 34)); +}; + +/** + * Used by apps to signIn and signAndSendTransactions + */ + +/// ethereum + +export const getEthereum = async () => { + const provider = await detectEthereumProvider(); + + if (!provider) { + return alert("Please install/activate MetaMask and try again."); + } + + try { + await window.ethereum.request({ + method: "wallet_switchEthereumChain", + params: [{ chainId: "0x" + domain.chainId.toString(16) }], + }); + } catch (e: any) { + logger.log(e); + const code = e?.code || e?.data?.originalError?.code; + if (code !== 4902) { + throw e; + } + + try { + await window.ethereum.request({ + method: "wallet_addEthereumChain", + params: [ + { + chainId: "0x" + domain.chainId.toString(16), + chainName: "Aurora Mainnet", + nativeCurrency: { + name: "Ethereum", + symbol: "ETH", + decimals: 18, + }, + blockExplorerUrls: ["https://explorer.mainnet.aurora.dev/"], + rpcUrls: ["https://mainnet.aurora.dev"], + }, + ], + }); + } catch (e2) { + alert( + 'Error adding chain. Please click "Choose Ethereum Account" and add the Aurora Network to continue.' + ); + throw e2; + } + } + + const ethersProvider = new ethers.providers.Web3Provider(window.ethereum); + const accounts = await ethersProvider.listAccounts(); + if (accounts.length === 0) { + await ethersProvider.send("eth_requestAccounts", []); + } + const signer = ethersProvider.getSigner(); + + return { signer, ethAddress: await signer.getAddress() }; +}; + +export const switchEthereum = async () => { + const provider = (await detectEthereumProvider()) as any; + await provider.send("wallet_requestPermissions", [{ eth_accounts: {} }]); + const ethersProvider = new ethers.providers.Web3Provider(window.ethereum); + const signer = ethersProvider.getSigner(); + return { signer, ethAddress: await signer.getAddress() }; +}; + +/// near + +export const getNearMap = async (eth_address) => { + return contractAccount.viewFunction( + NETWORK[networkId].MAP_ACCOUNT_ID, + "get_near", + { eth_address } + ); +}; + +export const getNear = async (): { + account: NearAccount; + accountId: string; + keyPair: NearKeyPair; + secretKey: string; +} => { + const secretKey = await storage.getItem(APP_KEY_SECRET); + const accountId = await storage.getItem(APP_KEY_ACCOUNT_ID); + if (!secretKey || !accountId) { + const ethRes: any = await getEthereum(); + const res = await getAppKey(ethRes); + if (!res) { + return false; + } + return await getNear(); + } + const account = new Account(connection, accountId); + const keyPair = KeyPair.fromString(secretKey); + keyStore.setKey(networkId, accountId, keyPair); + + return { account, accountId, keyPair, secretKey }; +}; + +export const signIn = getNear; + +export const signOut = async () => { + const accountId = await storage.getItem(APP_KEY_ACCOUNT_ID); + if (!accountId) { + return logger.log("already signed out"); + } + await storage.removeItem(APP_KEY_SECRET); + await storage.removeItem(APP_KEY_ACCOUNT_ID); + return { accountId }; +}; + +export const verifyOwner = async ({ message, provider, account }) => { + let accountId; + if (!account) { + ({ account, accountId } = await getNear()); + } else { + ({ accountId } = account); + } + + if (!account) { + throw new Error("Wallet not signed in"); + } + + const pubKey = await account.connection.signer.getPublicKey( + accountId, + networkId + ); + const publicKey = Buffer.from(pubKey.data).toString("base64"); + const block = await provider.block({ finality: "final" }); + const blockId = block.header.hash; + + const data = { + accountId, + message, + blockId, + publicKey, + keyType: pubKey.keyType, + }; + const encoded = JSON.stringify(data); + + const signed = await account.connection.signer.signMessage( + new Uint8Array(Buffer.from(encoded)), + accountId, + networkId + ); + + return { + ...data, + signature: Buffer.from(signed.signature).toString("base64"), + }; +}; + +export const isSignedIn = async () => { + /// init defaultStorage here because it's not initialized until initConnection + const tempStorage = defaultStorage(WS_STORAGE_NAMESPACE); + return ( + !!(await tempStorage.getItem(APP_KEY_SECRET)) || + !!(await tempStorage.getItem(APP_KEY_ACCOUNT_ID)) + ); +}; + +// const promptValidAccountId = async (msg) => { +// const newAccountId = window.prompt(msg); +// if (!newAccountId) { +// throw new Error("NETH Error: failed to pick valid NEAR account name"); +// } +// if ( +// newAccountId.length < 2 || +// newAccountId.indexOf(".") > -1 || +// !ACCOUNT_REGEX.test(newAccountId) || +// newAccountId.length > 64 +// ) { +// return promptValidAccountId( +// `account is invalid (a-z, 0-9 and -,_ only; min 2; max 64; ${accountSuffix} applied automatically)` +// ); +// } +// if (await accountExists(newAccountId)) { +// return promptValidAccountId(`account already exists`); +// } +// return newAccountId; +// }; + +export const getAppKey = async ({ signer, ethAddress: eth_address }) => { + const accountId = await getNearMap(eth_address); + + if (!accountId) { + const tryAgain = window.confirm( + `Ethereum account ${eth_address} is not connected to a NETH account. Would you like to try another Ethereum account?` + ); + + if (tryAgain) { + try { + const { signer: _signer, ethAddress } = await switchEthereum(); + return await getAppKey({ signer: _signer, ethAddress }); + } catch (e) { + logger.log(e); + } + return; + } + + const nethURL = `${NETH_SITE_URL}/${ + networkId === "testnet" ? "?network=testnet" : "" + }`; + window.prompt( + `We couldn't find a NETH account. To set up a NETH account visit`, + nethURL + ); + + // throw new Error(`Ethereum account is not connected to a NETH account. To set up a NETH account visit: ${nethURL}`) + // /// prompt for near account name and auto deploy + // const newAccountId = await promptValidAccountId( + // `The Ethereum address ${eth_address} is not connected to a NEAR account yet. Select a NEAR account name and we'll create and connect one for you.`, + // ); + // const { account } = await handleCreate(signer, eth_address, newAccountId + accountSuffix); + // accountId = account.accountId; + } + const appKeyNonce = parseInt( + await contractAccount.viewFunction(accountId, "get_app_key_nonce"), + 16 + ).toString(); + const { publicKey, secretKey } = await keyPairFromEthSig( + signer, + appKeyPayload(accountId, appKeyNonce) + ); + const account = new Account(connection, accountId); + // check that app key exists on account + const accessKeys = await account.getAccessKeys(); + if (!hasAppKey(accessKeys)) { + await handleRefreshAppKey(signer, eth_address); + } + const keyPair = KeyPair.fromString(secretKey); + keyStore.setKey(networkId, accountId, keyPair); + await storage.setItem(APP_KEY_SECRET, secretKey); + await storage.setItem(APP_KEY_ACCOUNT_ID, account.accountId); + return { publicKey, secretKey, account }; +}; + +const broadcastTXs = async () => { + const { account, accountId } = await getNear(); + const args = await storage.getItem(TX_ARGS_ATTEMPT); + if (!args || args.length === 0) { + return; + } + + const res: Array = []; + while (args.length > 0) { + const currentArgs = args.shift(); + logger.log("NETH: broadcasting tx", currentArgs); + try { + const tx = await account.functionCall({ + contractId: accountId, + methodName: "execute", + args: currentArgs, + gas, + }); + await storage.setItem(TX_ARGS_ATTEMPT, args); + res.push(tx); + } catch (e) { + logger.log("NETH: ERROR broadcasting tx", e); + } + } + await storage.removeItem(TX_ARGS_ATTEMPT); + return res; +}; + +export const signAndSendTransactions = async ({ transactions, bundle = false }) => { + const ethRes: any = await getEthereum(); + const { signer } = ethRes; + const { account, accountId } = await getNear(); + + const receivers = transactions.map(({ receiverId }) => receiverId); + const transformedTxs = transactions.map(({ receiverId, actions }) => ({ + actions: convertActions(actions, accountId, receiverId), + })); + + const nonce = parseInt( + await account.viewFunction(accountId, "get_nonce"), + 16 + ); + const args: Array = []; + if (!bundle) { + for (let i = 0; i < transformedTxs.length; i++) { + args.push( + await ethSignJson(signer, { + nonce: (nonce + i).toString(), + receivers: [receivers[i]], + transactions: [transformedTxs[i]], + }) + ); + } + } else { + args.push( + await ethSignJson(signer, { + nonce: nonce.toString(), + receivers, + transactions: transformedTxs, + }) + ); + } + + await storage.setItem(TX_ARGS_ATTEMPT, args); + + const res = await broadcastTXs(); + + return res; +}; + +/// helpers + +export const convertActions = (actions, accountId, receiverId) => + actions.map((_action) => { + const { enum: type } = _action; + const { + gas: _gas, + publicKey, + methodName, + args, + deposit, + accessKey, + code, + } = _action[type] || _action; + + const action: any = { + type: (type && type[0].toUpperCase() + type.substr(1)) || "FunctionCall", + gas: (_gas && _gas.toString()) || undefined, + public_key: (publicKey && pub2hex(publicKey)) || undefined, + method_name: methodName, + args: args || undefined, + code: code || undefined, + amount: (deposit && deposit.toString()) || undefined, + permission: undefined, + }; + + Object.keys(action).forEach((k) => { + if (action[k] === undefined) { + delete action[k]; + } + }); + + if (accessKey) { + if (receiverId === accountId) { + action.allowance = parseNearAmount("1"); + action.method_names = "execute"; + action.receiver_id = accountId; + } else if (accessKey.permission.enum === "functionCall") { + const { + receiverId: _receiverId, + methodNames, + allowance, + } = accessKey.permission.functionCall; + action.receiver_id = _receiverId; + action.allowance = + (allowance && allowance.toString()) || parseNearAmount("0.25"); + action.method_names = methodNames.join(","); + } + } + + return action; + }); diff --git a/packages/fastauth-metamask/tsconfig.json b/packages/fastauth-metamask/tsconfig.json new file mode 100644 index 000000000..f5667adda --- /dev/null +++ b/packages/fastauth-metamask/tsconfig.json @@ -0,0 +1,22 @@ +{ + "extends": "../../tsconfig.base.json", + "compilerOptions": { + "forceConsistentCasingInFileNames": true, + "strict": true, + "noImplicitReturns": true, + "noFallthroughCasesInSwitch": true, + "allowJs": true, + "noImplicitAny": false, + }, + "files": [], + "include": [], + "references": [ + { + "path": "./tsconfig.lib.json" + }, + { + "path": "./tsconfig.spec.json" + } + ] + } + \ No newline at end of file diff --git a/packages/fastauth-metamask/tsconfig.lib.json b/packages/fastauth-metamask/tsconfig.lib.json new file mode 100644 index 000000000..24b2269e5 --- /dev/null +++ b/packages/fastauth-metamask/tsconfig.lib.json @@ -0,0 +1,10 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "../../dist/out-tsc", + "declaration": true, + "types": [] + }, + "include": ["**/*.ts", "src/lib/fastauth-metamask-lib.ts"], + "exclude": ["jest.config.ts", "**/*.spec.ts", "**/*.test.ts"] +} diff --git a/packages/fastauth-metamask/tsconfig.spec.json b/packages/fastauth-metamask/tsconfig.spec.json new file mode 100644 index 000000000..a176d1bc4 --- /dev/null +++ b/packages/fastauth-metamask/tsconfig.spec.json @@ -0,0 +1,15 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "../../dist/out-tsc", + "module": "commonjs", + "types": [ + "jest", + "node" + ] + }, + "include": [ + "jest.config.ts", + "**/*.spec.ts" + ] +} diff --git a/packages/fastauth-metamask/yarn.lock b/packages/fastauth-metamask/yarn.lock new file mode 100644 index 000000000..6192f4420 --- /dev/null +++ b/packages/fastauth-metamask/yarn.lock @@ -0,0 +1,653 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +"@ethersproject/abi@5.7.0", "@ethersproject/abi@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/abi/-/abi-5.7.0.tgz#b3f3e045bbbeed1af3947335c247ad625a44e449" + integrity sha512-351ktp42TiRcYB3H1OP8yajPeAQstMW/yCFokj/AthP9bLHzQFPlOrxOcwYEDkUAICmOHljvN4K39OMTMUa9RA== + dependencies: + "@ethersproject/address" "^5.7.0" + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/constants" "^5.7.0" + "@ethersproject/hash" "^5.7.0" + "@ethersproject/keccak256" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/strings" "^5.7.0" + +"@ethersproject/abstract-provider@5.7.0", "@ethersproject/abstract-provider@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/abstract-provider/-/abstract-provider-5.7.0.tgz#b0a8550f88b6bf9d51f90e4795d48294630cb9ef" + integrity sha512-R41c9UkchKCpAqStMYUpdunjo3pkEvZC3FAwZn5S5MGbXoMQOHIdHItezTETxAO5bevtMApSyEhn9+CHcDsWBw== + dependencies: + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/networks" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/transactions" "^5.7.0" + "@ethersproject/web" "^5.7.0" + +"@ethersproject/abstract-signer@5.7.0", "@ethersproject/abstract-signer@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz#13f4f32117868452191a4649723cb086d2b596b2" + integrity sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ== + dependencies: + "@ethersproject/abstract-provider" "^5.7.0" + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + +"@ethersproject/address@5.7.0", "@ethersproject/address@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/address/-/address-5.7.0.tgz#19b56c4d74a3b0a46bfdbb6cfcc0a153fc697f37" + integrity sha512-9wYhYt7aghVGo758POM5nqcOMaE168Q6aRLJZwUmiqSrAungkG74gSSeKEIR7ukixesdRZGPgVqme6vmxs1fkA== + dependencies: + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/keccak256" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/rlp" "^5.7.0" + +"@ethersproject/base64@5.7.0", "@ethersproject/base64@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/base64/-/base64-5.7.0.tgz#ac4ee92aa36c1628173e221d0d01f53692059e1c" + integrity sha512-Dr8tcHt2mEbsZr/mwTPIQAf3Ai0Bks/7gTw9dSqk1mQvhW3XvRlmDJr/4n+wg1JmCl16NZue17CDh8xb/vZ0sQ== + dependencies: + "@ethersproject/bytes" "^5.7.0" + +"@ethersproject/basex@5.7.0", "@ethersproject/basex@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/basex/-/basex-5.7.0.tgz#97034dc7e8938a8ca943ab20f8a5e492ece4020b" + integrity sha512-ywlh43GwZLv2Voc2gQVTKBoVQ1mti3d8HK5aMxsfu/nRDnMmNqaSJ3r3n85HBByT8OpoY96SXM1FogC533T4zw== + dependencies: + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + +"@ethersproject/bignumber@5.7.0", "@ethersproject/bignumber@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/bignumber/-/bignumber-5.7.0.tgz#e2f03837f268ba655ffba03a57853e18a18dc9c2" + integrity sha512-n1CAdIHRWjSucQO3MC1zPSVgV/6dy/fjL9pMrPP9peL+QxEg9wOsVqwD4+818B6LUEtaXzVHQiuivzRoxPxUGw== + dependencies: + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + bn.js "^5.2.1" + +"@ethersproject/bytes@5.7.0", "@ethersproject/bytes@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/bytes/-/bytes-5.7.0.tgz#a00f6ea8d7e7534d6d87f47188af1148d71f155d" + integrity sha512-nsbxwgFXWh9NyYWo+U8atvmMsSdKJprTcICAkvbBffT75qDocbuggBU0SJiVK2MuTrp0q+xvLkTnGMPK1+uA9A== + dependencies: + "@ethersproject/logger" "^5.7.0" + +"@ethersproject/constants@5.7.0", "@ethersproject/constants@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/constants/-/constants-5.7.0.tgz#df80a9705a7e08984161f09014ea012d1c75295e" + integrity sha512-DHI+y5dBNvkpYUMiRQyxRBYBefZkJfo70VUkUAsRjcPs47muV9evftfZ0PJVCXYbAiCgght0DtcF9srFQmIgWA== + dependencies: + "@ethersproject/bignumber" "^5.7.0" + +"@ethersproject/contracts@5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/contracts/-/contracts-5.7.0.tgz#c305e775abd07e48aa590e1a877ed5c316f8bd1e" + integrity sha512-5GJbzEU3X+d33CdfPhcyS+z8MzsTrBGk/sc+G+59+tPa9yFkl6HQ9D6L0QMgNTA9q8dT0XKxxkyp883XsQvbbg== + dependencies: + "@ethersproject/abi" "^5.7.0" + "@ethersproject/abstract-provider" "^5.7.0" + "@ethersproject/abstract-signer" "^5.7.0" + "@ethersproject/address" "^5.7.0" + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/constants" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/transactions" "^5.7.0" + +"@ethersproject/hash@5.7.0", "@ethersproject/hash@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/hash/-/hash-5.7.0.tgz#eb7aca84a588508369562e16e514b539ba5240a7" + integrity sha512-qX5WrQfnah1EFnO5zJv1v46a8HW0+E5xuBBDTwMFZLuVTx0tbU2kkx15NqdjxecrLGatQN9FGQKpb1FKdHCt+g== + dependencies: + "@ethersproject/abstract-signer" "^5.7.0" + "@ethersproject/address" "^5.7.0" + "@ethersproject/base64" "^5.7.0" + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/keccak256" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/strings" "^5.7.0" + +"@ethersproject/hdnode@5.7.0", "@ethersproject/hdnode@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/hdnode/-/hdnode-5.7.0.tgz#e627ddc6b466bc77aebf1a6b9e47405ca5aef9cf" + integrity sha512-OmyYo9EENBPPf4ERhR7oj6uAtUAhYGqOnIS+jE5pTXvdKBS99ikzq1E7Iv0ZQZ5V36Lqx1qZLeak0Ra16qpeOg== + dependencies: + "@ethersproject/abstract-signer" "^5.7.0" + "@ethersproject/basex" "^5.7.0" + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/pbkdf2" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/sha2" "^5.7.0" + "@ethersproject/signing-key" "^5.7.0" + "@ethersproject/strings" "^5.7.0" + "@ethersproject/transactions" "^5.7.0" + "@ethersproject/wordlists" "^5.7.0" + +"@ethersproject/json-wallets@5.7.0", "@ethersproject/json-wallets@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/json-wallets/-/json-wallets-5.7.0.tgz#5e3355287b548c32b368d91014919ebebddd5360" + integrity sha512-8oee5Xgu6+RKgJTkvEMl2wDgSPSAQ9MB/3JYjFV9jlKvcYHUXZC+cQp0njgmxdHkYWn8s6/IqIZYm0YWCjO/0g== + dependencies: + "@ethersproject/abstract-signer" "^5.7.0" + "@ethersproject/address" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/hdnode" "^5.7.0" + "@ethersproject/keccak256" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/pbkdf2" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/random" "^5.7.0" + "@ethersproject/strings" "^5.7.0" + "@ethersproject/transactions" "^5.7.0" + aes-js "3.0.0" + scrypt-js "3.0.1" + +"@ethersproject/keccak256@5.7.0", "@ethersproject/keccak256@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/keccak256/-/keccak256-5.7.0.tgz#3186350c6e1cd6aba7940384ec7d6d9db01f335a" + integrity sha512-2UcPboeL/iW+pSg6vZ6ydF8tCnv3Iu/8tUmLLzWWGzxWKFFqOBQFLo6uLUv6BDrLgCDfN28RJ/wtByx+jZ4KBg== + dependencies: + "@ethersproject/bytes" "^5.7.0" + js-sha3 "0.8.0" + +"@ethersproject/logger@5.7.0", "@ethersproject/logger@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/logger/-/logger-5.7.0.tgz#6ce9ae168e74fecf287be17062b590852c311892" + integrity sha512-0odtFdXu/XHtjQXJYA3u9G0G8btm0ND5Cu8M7i5vhEcE8/HmF4Lbdqanwyv4uQTr2tx6b7fQRmgLrsnpQlmnig== + +"@ethersproject/networks@5.7.1", "@ethersproject/networks@^5.7.0": + version "5.7.1" + resolved "https://registry.yarnpkg.com/@ethersproject/networks/-/networks-5.7.1.tgz#118e1a981d757d45ccea6bb58d9fd3d9db14ead6" + integrity sha512-n/MufjFYv3yFcUyfhnXotyDlNdFb7onmkSy8aQERi2PjNcnWQ66xXxa3XlS8nCcA8aJKJjIIMNJTC7tu80GwpQ== + dependencies: + "@ethersproject/logger" "^5.7.0" + +"@ethersproject/pbkdf2@5.7.0", "@ethersproject/pbkdf2@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/pbkdf2/-/pbkdf2-5.7.0.tgz#d2267d0a1f6e123f3771007338c47cccd83d3102" + integrity sha512-oR/dBRZR6GTyaofd86DehG72hY6NpAjhabkhxgr3X2FpJtJuodEl2auADWBZfhDHgVCbu3/H/Ocq2uC6dpNjjw== + dependencies: + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/sha2" "^5.7.0" + +"@ethersproject/properties@5.7.0", "@ethersproject/properties@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/properties/-/properties-5.7.0.tgz#a6e12cb0439b878aaf470f1902a176033067ed30" + integrity sha512-J87jy8suntrAkIZtecpxEPxY//szqr1mlBaYlQ0r4RCaiD2hjheqF9s1LVE8vVuJCXisjIP+JgtK/Do54ej4Sw== + dependencies: + "@ethersproject/logger" "^5.7.0" + +"@ethersproject/providers@5.7.2": + version "5.7.2" + resolved "https://registry.yarnpkg.com/@ethersproject/providers/-/providers-5.7.2.tgz#f8b1a4f275d7ce58cf0a2eec222269a08beb18cb" + integrity sha512-g34EWZ1WWAVgr4aptGlVBF8mhl3VWjv+8hoAnzStu8Ah22VHBsuGzP17eb6xDVRzw895G4W7vvx60lFFur/1Rg== + dependencies: + "@ethersproject/abstract-provider" "^5.7.0" + "@ethersproject/abstract-signer" "^5.7.0" + "@ethersproject/address" "^5.7.0" + "@ethersproject/base64" "^5.7.0" + "@ethersproject/basex" "^5.7.0" + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/constants" "^5.7.0" + "@ethersproject/hash" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/networks" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/random" "^5.7.0" + "@ethersproject/rlp" "^5.7.0" + "@ethersproject/sha2" "^5.7.0" + "@ethersproject/strings" "^5.7.0" + "@ethersproject/transactions" "^5.7.0" + "@ethersproject/web" "^5.7.0" + bech32 "1.1.4" + ws "7.4.6" + +"@ethersproject/random@5.7.0", "@ethersproject/random@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/random/-/random-5.7.0.tgz#af19dcbc2484aae078bb03656ec05df66253280c" + integrity sha512-19WjScqRA8IIeWclFme75VMXSBvi4e6InrUNuaR4s5pTF2qNhcGdCUwdxUVGtDDqC00sDLCO93jPQoDUH4HVmQ== + dependencies: + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + +"@ethersproject/rlp@5.7.0", "@ethersproject/rlp@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/rlp/-/rlp-5.7.0.tgz#de39e4d5918b9d74d46de93af80b7685a9c21304" + integrity sha512-rBxzX2vK8mVF7b0Tol44t5Tb8gomOHkj5guL+HhzQ1yBh/ydjGnpw6at+X6Iw0Kp3OzzzkcKp8N9r0W4kYSs9w== + dependencies: + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + +"@ethersproject/sha2@5.7.0", "@ethersproject/sha2@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/sha2/-/sha2-5.7.0.tgz#9a5f7a7824ef784f7f7680984e593a800480c9fb" + integrity sha512-gKlH42riwb3KYp0reLsFTokByAKoJdgFCwI+CCiX/k+Jm2mbNs6oOaCjYQSlI1+XBVejwH2KrmCbMAT/GnRDQw== + dependencies: + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + hash.js "1.1.7" + +"@ethersproject/signing-key@5.7.0", "@ethersproject/signing-key@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/signing-key/-/signing-key-5.7.0.tgz#06b2df39411b00bc57c7c09b01d1e41cf1b16ab3" + integrity sha512-MZdy2nL3wO0u7gkB4nA/pEf8lu1TlFswPNmy8AiYkfKTdO6eXBJyUdmHO/ehm/htHw9K/qF8ujnTyUAD+Ry54Q== + dependencies: + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + bn.js "^5.2.1" + elliptic "6.5.4" + hash.js "1.1.7" + +"@ethersproject/solidity@5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/solidity/-/solidity-5.7.0.tgz#5e9c911d8a2acce2a5ebb48a5e2e0af20b631cb8" + integrity sha512-HmabMd2Dt/raavyaGukF4XxizWKhKQ24DoLtdNbBmNKUOPqwjsKQSdV9GQtj9CBEea9DlzETlVER1gYeXXBGaA== + dependencies: + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/keccak256" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/sha2" "^5.7.0" + "@ethersproject/strings" "^5.7.0" + +"@ethersproject/strings@5.7.0", "@ethersproject/strings@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/strings/-/strings-5.7.0.tgz#54c9d2a7c57ae8f1205c88a9d3a56471e14d5ed2" + integrity sha512-/9nu+lj0YswRNSH0NXYqrh8775XNyEdUQAuf3f+SmOrnVewcJ5SBNAjF7lpgehKi4abvNNXyf+HX86czCdJ8Mg== + dependencies: + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/constants" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + +"@ethersproject/transactions@5.7.0", "@ethersproject/transactions@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/transactions/-/transactions-5.7.0.tgz#91318fc24063e057885a6af13fdb703e1f993d3b" + integrity sha512-kmcNicCp1lp8qanMTC3RIikGgoJ80ztTyvtsFvCYpSCfkjhD0jZ2LOrnbcuxuToLIUYYf+4XwD1rP+B/erDIhQ== + dependencies: + "@ethersproject/address" "^5.7.0" + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/constants" "^5.7.0" + "@ethersproject/keccak256" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/rlp" "^5.7.0" + "@ethersproject/signing-key" "^5.7.0" + +"@ethersproject/units@5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/units/-/units-5.7.0.tgz#637b563d7e14f42deeee39245275d477aae1d8b1" + integrity sha512-pD3xLMy3SJu9kG5xDGI7+xhTEmGXlEqXU4OfNapmfnxLVY4EMSSRp7j1k7eezutBPH7RBN/7QPnwR7hzNlEFeg== + dependencies: + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/constants" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + +"@ethersproject/wallet@5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/wallet/-/wallet-5.7.0.tgz#4e5d0790d96fe21d61d38fb40324e6c7ef350b2d" + integrity sha512-MhmXlJXEJFBFVKrDLB4ZdDzxcBxQ3rLyCkhNqVu3CDYvR97E+8r01UgrI+TI99Le+aYm/in/0vp86guJuM7FCA== + dependencies: + "@ethersproject/abstract-provider" "^5.7.0" + "@ethersproject/abstract-signer" "^5.7.0" + "@ethersproject/address" "^5.7.0" + "@ethersproject/bignumber" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/hash" "^5.7.0" + "@ethersproject/hdnode" "^5.7.0" + "@ethersproject/json-wallets" "^5.7.0" + "@ethersproject/keccak256" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/random" "^5.7.0" + "@ethersproject/signing-key" "^5.7.0" + "@ethersproject/transactions" "^5.7.0" + "@ethersproject/wordlists" "^5.7.0" + +"@ethersproject/web@5.7.1", "@ethersproject/web@^5.7.0": + version "5.7.1" + resolved "https://registry.yarnpkg.com/@ethersproject/web/-/web-5.7.1.tgz#de1f285b373149bee5928f4eb7bcb87ee5fbb4ae" + integrity sha512-Gueu8lSvyjBWL4cYsWsjh6MtMwM0+H4HvqFPZfB6dV8ctbP9zFAO73VG1cMWae0FLPCtz0peKPpZY8/ugJJX2w== + dependencies: + "@ethersproject/base64" "^5.7.0" + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/strings" "^5.7.0" + +"@ethersproject/wordlists@5.7.0", "@ethersproject/wordlists@^5.7.0": + version "5.7.0" + resolved "https://registry.yarnpkg.com/@ethersproject/wordlists/-/wordlists-5.7.0.tgz#8fb2c07185d68c3e09eb3bfd6e779ba2774627f5" + integrity sha512-S2TFNJNfHWVHNE6cNDjbVlZ6MgE17MIxMbMg2zv3wn+3XSJGosL1m9ZVv3GXCf/2ymSsQ+hRI5IzoMJTG6aoVA== + dependencies: + "@ethersproject/bytes" "^5.7.0" + "@ethersproject/hash" "^5.7.0" + "@ethersproject/logger" "^5.7.0" + "@ethersproject/properties" "^5.7.0" + "@ethersproject/strings" "^5.7.0" + +"@metamask/detect-provider@^2.0.0": + version "2.0.0" + resolved "https://registry.yarnpkg.com/@metamask/detect-provider/-/detect-provider-2.0.0.tgz#4bc2795e5e6f7d8b84b2e845058d2f222c99917d" + integrity sha512-sFpN+TX13E9fdBDh9lvQeZdJn4qYoRb/6QF2oZZK/Pn559IhCFacPMU1rMuqyXoFQF3JSJfii2l98B87QDPeCQ== + +"@types/node@11.11.6": + version "11.11.6" + resolved "https://registry.yarnpkg.com/@types/node/-/node-11.11.6.tgz#df929d1bb2eee5afdda598a41930fe50b43eaa6a" + integrity sha512-Exw4yUWMBXM3X+8oqzJNRqZSwUAaS4+7NdvHqQuFi/d+synz++xmX3QIf+BFqneW8N31R8Ky+sikfZUXq07ggQ== + +aes-js@3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/aes-js/-/aes-js-3.0.0.tgz#e21df10ad6c2053295bcbb8dab40b09dbea87e4d" + integrity sha512-H7wUZRn8WpTq9jocdxQ2c8x2sKo9ZVmzfRE13GiNJXfp7NcKYEdvl3vspKjXox6RIG2VtaRe4JFvxG4rqp2Zuw== + +base-x@^3.0.2: + version "3.0.9" + resolved "https://registry.yarnpkg.com/base-x/-/base-x-3.0.9.tgz#6349aaabb58526332de9f60995e548a53fe21320" + integrity sha512-H7JU6iBHTal1gp56aKoaa//YUxEaAOUiydvrV/pILqIHXTtqxSkATOnDA2u+jZ/61sD+L/412+7kzXRtWukhpQ== + dependencies: + safe-buffer "^5.0.1" + +base-x@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/base-x/-/base-x-4.0.0.tgz#d0e3b7753450c73f8ad2389b5c018a4af7b2224a" + integrity sha512-FuwxlW4H5kh37X/oW59pwTzzTKRzfrrQwhmyspRM7swOEZcHtDZSCt45U6oKgtuFE+WYPblePMVIPR4RZrh/hw== + +bech32@1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/bech32/-/bech32-1.1.4.tgz#e38c9f37bf179b8eb16ae3a772b40c356d4832e9" + integrity sha512-s0IrSOzLlbvX7yp4WBfPITzpAU8sqQcpsmwXDiKwrG4r491vwCO/XpejasRNl0piBMe/DvP4Tz0mIS/X1DPJBQ== + +bip39-light@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/bip39-light/-/bip39-light-1.0.7.tgz#06a72f251b89389a136d3f177f29b03342adc5ba" + integrity sha512-WDTmLRQUsiioBdTs9BmSEmkJza+8xfJmptsNJjxnoq3EydSa/ZBXT6rm66KoT3PJIRYMnhSKNR7S9YL1l7R40Q== + dependencies: + create-hash "^1.1.0" + pbkdf2 "^3.0.9" + +bip39@3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/bip39/-/bip39-3.0.2.tgz#2baf42ff3071fc9ddd5103de92e8f80d9257ee32" + integrity sha512-J4E1r2N0tUylTKt07ibXvhpT2c5pyAFgvuA5q1H9uDy6dEGpjV8jmymh3MTYJDLCNbIVClSB9FbND49I6N24MQ== + dependencies: + "@types/node" "11.11.6" + create-hash "^1.1.0" + pbkdf2 "^3.0.9" + randombytes "^2.0.1" + +bn.js@^4.11.9: + version "4.12.0" + resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.12.0.tgz#775b3f278efbb9718eec7361f483fb36fbbfea88" + integrity sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA== + +bn.js@^5.2.1: + version "5.2.1" + resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-5.2.1.tgz#0bc527a6a0d18d0aa8d5b0538ce4a77dccfa7b70" + integrity sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ== + +brorand@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/brorand/-/brorand-1.1.0.tgz#12c25efe40a45e3c323eb8675a0a0ce57b22371f" + integrity sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w== + +bs58@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/bs58/-/bs58-4.0.1.tgz#be161e76c354f6f788ae4071f63f34e8c4f0a42a" + integrity sha512-Ok3Wdf5vOIlBrgCvTq96gBkJw+JUEzdBgyaza5HLtPm7yTHkjRy8+JzNyHF7BHa0bNWOQIp3m5YF0nnFcOIKLw== + dependencies: + base-x "^3.0.2" + +bs58@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/bs58/-/bs58-5.0.0.tgz#865575b4d13c09ea2a84622df6c8cbeb54ffc279" + integrity sha512-r+ihvQJvahgYT50JD05dyJNKlmmSlMoOGwn1lCcEzanPglg7TxYjioQUYehQ9mAR/+hOSd2jRc/Z2y5UxBymvQ== + dependencies: + base-x "^4.0.0" + +cipher-base@^1.0.1, cipher-base@^1.0.3: + version "1.0.4" + resolved "https://registry.yarnpkg.com/cipher-base/-/cipher-base-1.0.4.tgz#8760e4ecc272f4c363532f926d874aae2c1397de" + integrity sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q== + dependencies: + inherits "^2.0.1" + safe-buffer "^5.0.1" + +create-hash@^1.1.0, create-hash@^1.1.2: + version "1.2.0" + resolved "https://registry.yarnpkg.com/create-hash/-/create-hash-1.2.0.tgz#889078af11a63756bcfb59bd221996be3a9ef196" + integrity sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg== + dependencies: + cipher-base "^1.0.1" + inherits "^2.0.1" + md5.js "^1.3.4" + ripemd160 "^2.0.1" + sha.js "^2.4.0" + +create-hmac@1.1.7, create-hmac@^1.1.4: + version "1.1.7" + resolved "https://registry.yarnpkg.com/create-hmac/-/create-hmac-1.1.7.tgz#69170c78b3ab957147b2b8b04572e47ead2243ff" + integrity sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg== + dependencies: + cipher-base "^1.0.3" + create-hash "^1.1.0" + inherits "^2.0.1" + ripemd160 "^2.0.0" + safe-buffer "^5.0.1" + sha.js "^2.4.8" + +elliptic@6.5.4: + version "6.5.4" + resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.4.tgz#da37cebd31e79a1367e941b592ed1fbebd58abbb" + integrity sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ== + dependencies: + bn.js "^4.11.9" + brorand "^1.1.0" + hash.js "^1.0.0" + hmac-drbg "^1.0.1" + inherits "^2.0.4" + minimalistic-assert "^1.0.1" + minimalistic-crypto-utils "^1.0.1" + +ethers@^5.7.2: + version "5.7.2" + resolved "https://registry.yarnpkg.com/ethers/-/ethers-5.7.2.tgz#3a7deeabbb8c030d4126b24f84e525466145872e" + integrity sha512-wswUsmWo1aOK8rR7DIKiWSw9DbLWe6x98Jrn8wcTflTVvaXhAMaB5zGAXy0GYQEQp9iO1iSHWVyARQm11zUtyg== + dependencies: + "@ethersproject/abi" "5.7.0" + "@ethersproject/abstract-provider" "5.7.0" + "@ethersproject/abstract-signer" "5.7.0" + "@ethersproject/address" "5.7.0" + "@ethersproject/base64" "5.7.0" + "@ethersproject/basex" "5.7.0" + "@ethersproject/bignumber" "5.7.0" + "@ethersproject/bytes" "5.7.0" + "@ethersproject/constants" "5.7.0" + "@ethersproject/contracts" "5.7.0" + "@ethersproject/hash" "5.7.0" + "@ethersproject/hdnode" "5.7.0" + "@ethersproject/json-wallets" "5.7.0" + "@ethersproject/keccak256" "5.7.0" + "@ethersproject/logger" "5.7.0" + "@ethersproject/networks" "5.7.1" + "@ethersproject/pbkdf2" "5.7.0" + "@ethersproject/properties" "5.7.0" + "@ethersproject/providers" "5.7.2" + "@ethersproject/random" "5.7.0" + "@ethersproject/rlp" "5.7.0" + "@ethersproject/sha2" "5.7.0" + "@ethersproject/signing-key" "5.7.0" + "@ethersproject/solidity" "5.7.0" + "@ethersproject/strings" "5.7.0" + "@ethersproject/transactions" "5.7.0" + "@ethersproject/units" "5.7.0" + "@ethersproject/wallet" "5.7.0" + "@ethersproject/web" "5.7.1" + "@ethersproject/wordlists" "5.7.0" + +hash-base@^3.0.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/hash-base/-/hash-base-3.1.0.tgz#55c381d9e06e1d2997a883b4a3fddfe7f0d3af33" + integrity sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA== + dependencies: + inherits "^2.0.4" + readable-stream "^3.6.0" + safe-buffer "^5.2.0" + +hash.js@1.1.7, hash.js@^1.0.0, hash.js@^1.0.3: + version "1.1.7" + resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.7.tgz#0babca538e8d4ee4a0f8988d68866537a003cf42" + integrity sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA== + dependencies: + inherits "^2.0.3" + minimalistic-assert "^1.0.1" + +hmac-drbg@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz#d2745701025a6c775a6c545793ed502fc0c649a1" + integrity sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg== + dependencies: + hash.js "^1.0.3" + minimalistic-assert "^1.0.0" + minimalistic-crypto-utils "^1.0.1" + +inherits@^2.0.1, inherits@^2.0.3, inherits@^2.0.4: + version "2.0.4" + resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" + integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== + +js-sha3@0.8.0: + version "0.8.0" + resolved "https://registry.yarnpkg.com/js-sha3/-/js-sha3-0.8.0.tgz#b9b7a5da73afad7dedd0f8c463954cbde6818840" + integrity sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q== + +md5.js@^1.3.4: + version "1.3.5" + resolved "https://registry.yarnpkg.com/md5.js/-/md5.js-1.3.5.tgz#b5d07b8e3216e3e27cd728d72f70d1e6a342005f" + integrity sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg== + dependencies: + hash-base "^3.0.0" + inherits "^2.0.1" + safe-buffer "^5.1.2" + +minimalistic-assert@^1.0.0, minimalistic-assert@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz#2e194de044626d4a10e7f7fbc00ce73e83e4d5c7" + integrity sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A== + +minimalistic-crypto-utils@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz#f6c00c1c0b082246e5c4d99dfb8c7c083b2b582a" + integrity sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg== + +near-hd-key@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/near-hd-key/-/near-hd-key-1.2.1.tgz#f508ff15436cf8a439b543220f3cc72188a46756" + integrity sha512-SIrthcL5Wc0sps+2e1xGj3zceEa68TgNZDLuCx0daxmfTP7sFTB3/mtE2pYhlFsCxWoMn+JfID5E1NlzvvbRJg== + dependencies: + bip39 "3.0.2" + create-hmac "1.1.7" + tweetnacl "1.0.3" + +near-seed-phrase@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/near-seed-phrase/-/near-seed-phrase-0.2.0.tgz#fb7cf89682112b1160ab68abb50dc821f49be18a" + integrity sha512-NpmrnejpY1AdlRpDZ0schJQJtfBaoUheRfiYtQpcq9TkwPgqKZCRULV5L3hHmLc0ep7KRtikbPQ9R2ztN/3cyQ== + dependencies: + bip39-light "^1.0.7" + bs58 "^4.0.1" + near-hd-key "^1.2.1" + tweetnacl "^1.0.2" + +pbkdf2@^3.0.9: + version "3.1.2" + resolved "https://registry.yarnpkg.com/pbkdf2/-/pbkdf2-3.1.2.tgz#dd822aa0887580e52f1a039dc3eda108efae3075" + integrity sha512-iuh7L6jA7JEGu2WxDwtQP1ddOpaJNC4KlDEFfdQajSGgGPNi4OyDc2R7QnbY2bR9QjBVGwgvTdNJZoE7RaxUMA== + dependencies: + create-hash "^1.1.2" + create-hmac "^1.1.4" + ripemd160 "^2.0.1" + safe-buffer "^5.0.1" + sha.js "^2.4.8" + +randombytes@^2.0.1: + version "2.1.0" + resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a" + integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ== + dependencies: + safe-buffer "^5.1.0" + +readable-stream@^3.6.0: + version "3.6.0" + resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.0.tgz#337bbda3adc0706bd3e024426a286d4b4b2c9198" + integrity sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA== + dependencies: + inherits "^2.0.3" + string_decoder "^1.1.1" + util-deprecate "^1.0.1" + +ripemd160@^2.0.0, ripemd160@^2.0.1: + version "2.0.2" + resolved "https://registry.yarnpkg.com/ripemd160/-/ripemd160-2.0.2.tgz#a1c1a6f624751577ba5d07914cbc92850585890c" + integrity sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA== + dependencies: + hash-base "^3.0.0" + inherits "^2.0.1" + +safe-buffer@^5.0.1, safe-buffer@^5.1.0, safe-buffer@^5.1.2, safe-buffer@^5.2.0, safe-buffer@~5.2.0: + version "5.2.1" + resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" + integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== + +scrypt-js@3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/scrypt-js/-/scrypt-js-3.0.1.tgz#d314a57c2aef69d1ad98a138a21fe9eafa9ee312" + integrity sha512-cdwTTnqPu0Hyvf5in5asVdZocVDTNRmR7XEcJuIzMjJeSHybHl7vpB66AzwTaIg6CLSbtjcxc8fqcySfnTkccA== + +sha.js@^2.4.0, sha.js@^2.4.8: + version "2.4.11" + resolved "https://registry.yarnpkg.com/sha.js/-/sha.js-2.4.11.tgz#37a5cf0b81ecbc6943de109ba2960d1b26584ae7" + integrity sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ== + dependencies: + inherits "^2.0.1" + safe-buffer "^5.0.1" + +string_decoder@^1.1.1: + version "1.3.0" + resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e" + integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA== + dependencies: + safe-buffer "~5.2.0" + +tweetnacl@1.0.3, tweetnacl@^1.0.2, tweetnacl@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-1.0.3.tgz#ac0af71680458d8a6378d0d0d050ab1407d35596" + integrity sha512-6rt+RN7aOi1nGMyC4Xa5DdYiukl2UWCbcJft7YhxReBGQD7OAM8Pbxw6YMo4r2diNEA8FEmu32YOn9rhaiE5yw== + +util-deprecate@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" + integrity sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw== + +ws@7.4.6: + version "7.4.6" + resolved "https://registry.yarnpkg.com/ws/-/ws-7.4.6.tgz#5654ca8ecdeee47c33a9a4bf6d28e2be2980377c" + integrity sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A== diff --git a/workspace.json b/workspace.json index 35d6e7d0c..68a01e66d 100644 --- a/workspace.json +++ b/workspace.json @@ -19,6 +19,7 @@ "narwallets": "packages/narwallets", "near-wallet": "packages/near-wallet", "neth": "packages/neth", + "fastauth-metamask": "packages/fastauth-metamask", "nightly": "packages/nightly", "nightly-connect": "packages/nightly-connect", "opto-wallet": "packages/opto-wallet", From 6ead683402f60d466711b4e2719a08a89a86bf8d Mon Sep 17 00:00:00 2001 From: Matt Date: Mon, 10 Apr 2023 12:49:46 -0700 Subject: [PATCH 2/6] completed basic --- .../react/contexts/WalletSelectorContext.tsx | 2 + .../src/lib/fastauth-metamask.ts | 72 ++-- packages/fastauth-metamask/src/lib/lib.tsx | 397 ++++++++++++++++++ packages/fastauth-metamask/src/lib/modal.tsx | 108 +++++ .../fastauth-metamask/src/lib/neth-lib.ts | 17 +- packages/fastauth-metamask/tsconfig.lib.json | 5 +- tsconfig.base.json | 3 + 7 files changed, 571 insertions(+), 33 deletions(-) create mode 100644 packages/fastauth-metamask/src/lib/lib.tsx create mode 100644 packages/fastauth-metamask/src/lib/modal.tsx diff --git a/examples/react/contexts/WalletSelectorContext.tsx b/examples/react/contexts/WalletSelectorContext.tsx index 4f150d13f..c5dde00a1 100644 --- a/examples/react/contexts/WalletSelectorContext.tsx +++ b/examples/react/contexts/WalletSelectorContext.tsx @@ -21,6 +21,7 @@ import React, { useCallback, useContext, useEffect, useState } from "react"; import { distinctUntilChanged, map } from "rxjs"; import { setupNeth } from "@near-wallet-selector/neth"; +import { setupFastAuthMetaMask } from "@near-wallet-selector/fastauth-metamask"; import { setupOptoWallet } from "@near-wallet-selector/opto-wallet"; import { setupFinerWallet } from "@near-wallet-selector/finer-wallet"; import { Loading } from "../components/Loading"; @@ -74,6 +75,7 @@ export const WalletSelectorContextProvider: React.FC<{ gas: "300000000000000", bundle: false, }), + setupFastAuthMetaMask(), setupOptoWallet(), setupFinerWallet(), setupXDEFI(), diff --git a/packages/fastauth-metamask/src/lib/fastauth-metamask.ts b/packages/fastauth-metamask/src/lib/fastauth-metamask.ts index 11d59ca25..74aa76c2f 100644 --- a/packages/fastauth-metamask/src/lib/fastauth-metamask.ts +++ b/packages/fastauth-metamask/src/lib/fastauth-metamask.ts @@ -6,17 +6,20 @@ import type { WalletBehaviourFactory, } from "@near-wallet-selector/core"; import detectEthereumProvider from "@metamask/detect-provider"; +import * as nearAPI from 'near-api-js' +const { + Account +} = nearAPI import { FastAuthMetaMaskIcon } from "../assets/icons"; import { - getNear, signIn, signOut, + getNethAccounts, + initConnection, verifyOwner, isSignedIn, - signAndSendTransactions, - initConnection, METAMASK_URL, -} from "./neth-lib"; +} from "./lib"; import isMobile from "is-mobile"; export { initConnection } from "./neth-lib"; @@ -34,10 +37,17 @@ declare global { export interface FastAuthMetaMaskParams { // default FastAuthMetaMask icon included iconUrl?: string; + // customize how much gas to attach to function calls + gas?: string; + // should there be a modal cover over the screen while awaiting tx results? + useModalCover?: boolean; // default false deprecated?: boolean; } +let useCover = false; +let customGas; + const isInstalled = async () => { await detectEthereumProvider({ timeout: 100 }); return !!window.ethereum; @@ -47,10 +57,17 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ metadata, logger, store, + storage, options, provider, }) => { + const cover = initConnection({ + network: options.network, + logger, + storage, + }) + const isValidActions = ( actions: Array ): actions is Array => { @@ -78,6 +95,10 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ throw new Error("Wallet not signed in"); } + if (useCover) { + cover.style.display = "block"; + } + const transformedTxs = transactions.map(({ receiverId, actions }) => ({ receiverId: receiverId || contract.contractId, actions: transformActions(actions), @@ -94,16 +115,20 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ throw e; } + if (useCover) { + cover.style.display = "none"; + } + return res; }; // return the wallet interface for wallet-selector return { async signIn() { - let account; + let accounts; try { - account = await signIn(); - if (!account) { + accounts = await signIn(); + if (!accounts) { return []; } } catch (e) { @@ -112,7 +137,7 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ } // console.log(e); } - return [account]; + return accounts; }, async signOut() { @@ -125,33 +150,30 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ }, async getAccounts() { - const { accountId, account } = await getNear(); - return [ - { - accountId, - publicKey: ( - await account.connection.signer.getPublicKey( - account.accountId, - options.network.networkId - ) - ).toString(), - }, - ]; + return getNethAccounts() }, - signAndSendTransaction: async ({ receiverId, actions }) => - signTransactions([{ receiverId, actions }]), + async signAndSendTransaction({ receiverId, actions }) { + return signTransactions([{ receiverId, actions }]) + }, - signAndSendTransactions: async ({ transactions }) => - signTransactions(transactions), + async signAndSendTransactions({ transactions }) { + const account = getNethAccounts() + (account as typeof Account).signAndSendTransactions(transactions); + }, }; }; export function setupFastAuthMetaMask({ iconUrl = FastAuthMetaMaskIcon, + gas, + useModalCover = false, deprecated = false, }: FastAuthMetaMaskParams = {}): WalletModuleFactory { + return async () => { + useCover = useModalCover; + customGas = gas; const mobile = isMobile(); if (mobile) { @@ -175,4 +197,4 @@ export function setupFastAuthMetaMask({ init: FastAuthMetaMask, }; }; -} +} \ No newline at end of file diff --git a/packages/fastauth-metamask/src/lib/lib.tsx b/packages/fastauth-metamask/src/lib/lib.tsx new file mode 100644 index 000000000..32dab4136 --- /dev/null +++ b/packages/fastauth-metamask/src/lib/lib.tsx @@ -0,0 +1,397 @@ +import React from 'react' +import { ethers } from "ethers"; +import { createRoot } from "react-dom/client"; +import detectEthereumProvider from "@metamask/detect-provider"; +import * as nearAPI from "near-api-js"; +import { Modal } from "./modal"; +import { generateSeedPhrase } from "near-seed-phrase"; + +// constants + +const domain = { + name: "NETH", + version: "1", + // chainId: 1, // aurora + chainId: 1313161554, // aurora +}; +const MODAL_ELEMENT_ID = "near-wallet-selector-modal"; +export const METAMASK_URL = "https://chrome.google.com/webstore/detail/metamask/nkbihfbeogaeaoehlefnkodbefgpgknn"; +const { + Near, + Account, + keyStores: { BrowserLocalStorageKeyStore }, +} = nearAPI; +const DOMAIN = { + name: "NETH", + version: "1", + // chainId: 1, // aurora + chainId: 1313161554, // aurora +}; +const defaultGas = "200000000000000"; +const NETWORK = { + testnet: { + FUNDING_ACCOUNT_ID: "neth.testnet", + MAP_ACCOUNT_ID: "map.neth.testnet", + ROOT_ACCOUNT_ID: "testnet", + }, + mainnet: { + MAP_ACCOUNT_ID: "nethmap.near", + ROOT_ACCOUNT_ID: "near", + }, +}; + +let appAccount: any = null; +let appKeyPair: any = null; + +// assigned in initConnection + +let gas, + storage, + near, + keyStore, + logger, + connection, + networkId, + contractAccount, + accountSuffix; + +// helpers + +const HEADER_OFFSET = "NETH"; +const HEADER_PAD = 8; +const RECEIVER_MARKER = "|~-_NETH~-_-~RECEIVER_-~|"; +const PREFIX = "|NETH_"; +const SUFFIX = "_NETH|"; + +const pack = (elements) => + elements + .map((el) => { + const str = + typeof el === "string" + ? el + : Object.entries(el) + .map( + ([k, v]) => + `${PREFIX}${k}:${typeof v === "string" ? v : JSON.stringify(v) + }${SUFFIX}` + ) + .join(""); + + const len = str.length.toString().padStart(HEADER_PAD, "0"); + + return HEADER_OFFSET + len + "__" + str; + }) + .join(""); + +const ethSignJson = async (signer, json) => { + const Transaction: Array = []; + const types = { Transaction }; + Object.entries(json).forEach(([k]) => { + types.Transaction.push({ + type: "string", + name: k, + }); + }); + /// convenience for devs so they can pass in JSON + + /// hoist any functionCall args containing receiver|account in their key to top level receivers + /// replaces value with marker, contract fills in marker + + if (json.transactions) { + Object.values(json.transactions).forEach((tx: any, i) => { + tx.actions.forEach((action) => { + if (!action.args) { + return; + } + if (Buffer.isBuffer(action.args)) { + action.args = "0x" + action.args.toString("hex"); + return; + } + Object.entries(action.args).forEach(([key, value]) => { + /// TODO include check on value to determine valid account_id to be replaced + + if (/receiver_id|account_id/g.test(key)) { + action.args[key] = RECEIVER_MARKER; + json.receivers.splice(i + 1, 0, value); + } + }); + }); + }); + + json.transactions = pack( + json.transactions.map(({ actions }) => pack(actions)) + ); + } + if (json.receivers) { + const numReceivers = json.receivers.length.toString(); + json.receivers = + HEADER_OFFSET + + json.receivers.join(",").length.toString().padStart(HEADER_PAD, "0") + + "__" + + json.receivers.join(","); + json.receivers = + json.receivers.substring(0, 4) + + numReceivers.padStart(3, "0") + + json.receivers.substring(7); + } + + const sig = await signer._signTypedData(domain, types, json); + + const args = { + sig, + msg: json, + }; + // logger.log('\nargs\n', JSON.stringify(args, null, 4), '\n'); + return args; +}; + +export const getKeyPair = async (signer, ethAddress, accountId) => { + const { sig } = await ethSignJson(signer, { + WARNING: `Creating key using Ethereum account: ${ethAddress}. This key can control the NEAR account: ${accountId}.`, + description: `ONLY sign this on apps you trust! This key CAN make ANY transactions on your NEAR account.`, + }); + const sigHash = ethers.utils.id(sig); + /// use 32 bytes of entropy from hash of signature to create NEAR keyPair + return generateSeedPhrase(sigHash.substring(2, 34)); +}; + +const defaultStorage = (prefix = "") => ({ + getItem: (k) => { + const v = localStorage.getItem(prefix + k); + if (v?.charAt(0) !== "{") { + return v; + } + try { + return JSON.parse(v); + } catch (e) { + // logger.log(e); + } + }, + setItem: (k, v) => + localStorage.setItem( + prefix + k, + typeof v === "string" ? v : JSON.stringify(v) + ), + removeItem: (k) => localStorage.removeItem(prefix + k), +}); + +const defaultLogger = () => ({ + // eslint-disable-next-line + log: (args) => console.log(...args), +}); + +export const getEthereum = async (): Promise => { + const provider = await detectEthereumProvider(); + + if (!provider) { + return alert("Please install/activate MetaMask and try again."); + } + + try { + await window.ethereum.request({ + method: "wallet_switchEthereumChain", + params: [{ chainId: "0x" + DOMAIN.chainId.toString(16) }], + }); + } catch (e: any) { + logger.log(e); + const code = e?.code || e?.data?.originalError?.code; + if (code !== 4902) { + throw e; + } + + try { + await window.ethereum.request({ + method: "wallet_addEthereumChain", + params: [ + { + chainId: "0x" + DOMAIN.chainId.toString(16), + chainName: "Aurora Mainnet", + nativeCurrency: { + name: "Ethereum", + symbol: "ETH", + decimals: 18, + }, + blockExplorerUrls: ["https://explorer.mainnet.aurora.dev/"], + rpcUrls: ["https://mainnet.aurora.dev"], + }, + ], + }); + } catch (e2) { + alert( + 'Error adding chain. Please click "Choose Ethereum Account" and add the Aurora Network to continue.' + ); + throw e2; + } + } + + const ethersProvider = new ethers.providers.Web3Provider(window.ethereum); + const accounts = await ethersProvider.listAccounts(); + if (accounts.length === 0) { + await ethersProvider.send("eth_requestAccounts", []); + } + const signer = ethersProvider.getSigner(); + + return { signer, ethAddress: await signer.getAddress() }; +}; + +const getNearMap = async (ethAddress) => { + console.log('getNearMap') + return contractAccount.viewFunction( + NETWORK[networkId].MAP_ACCOUNT_ID, + "get_near", + { eth_address: ethAddress } + ); +}; + +// export methods + +export const getNethAccounts = () => { + if (!appAccount) { + return [] + } + return [{ + accountId: appAccount.accountId, + publicKey: appKeyPair!.publicKey, + }] +} + +export const initConnection = ({ + network, + gas: _gas = defaultGas, + logger: _logger = defaultLogger(), + storage: _storage = defaultStorage(), +}) => { + gas = _gas; + logger = _logger; + storage = _storage; + + keyStore = new BrowserLocalStorageKeyStore(); + near = new Near({ + ...network, + keyStore, + }); + connection = near.connection; + networkId = network.networkId; + contractAccount = new Account( + connection, + NETWORK[networkId].ROOT_ACCOUNT_ID + ); + accountSuffix = "." + NETWORK[networkId].ROOT_ACCOUNT_ID; + + const cover = document.createElement("div"); + cover.style.display = "none"; + cover.style.width = "100%"; + cover.style.height = "100vh"; + cover.style.zIndex = "999999"; + cover.style.position = "fixed"; + cover.style.top = "0"; + cover.style.background = "rgba(0, 0, 0, 0.5)"; + document.body.appendChild(cover); + + return cover; +}; + +const handleSignIn = (resolve) => (accountId, keyPair) => { + appKeyPair = keyPair + appAccount = new Account(connection, accountId) + keyStore.setKey(networkId, accountId, keyPair.secretKey) + resolve(getNethAccounts()) +} + +export const signIn = async () => { + if (appAccount) { + alert('already signed in') + return getNethAccounts() + } + const { signer, ethAddress } = await getEthereum(); + + alert('Check map contract for NEAR Account matching ' + ethAddress) + + console.log('signIn') + + const accountId = await getNearMap(ethAddress); + console.log('response', accountId) + + if (accountId) { + const keyPair = await getKeyPair(signer, ethAddress, accountId); + return await new Promise((resolve, _) => { + const handleSignInMethod = handleSignIn(resolve); + handleSignInMethod(signer, keyPair); + }) + } + + /// TODO check map and pop new user modal or sign MM transaction to sign in + + return await new Promise((resolve, _) => { + setTimeout(() => { + let el: any = document.createElement("div"); + el.id = MODAL_ELEMENT_ID; + const elExists = document.getElementById(MODAL_ELEMENT_ID) + if (!elExists) { + document.body.appendChild(el); + } else { + el = elExists + } + const root = createRoot(el!); + root.render(); + }, 250); + }) +} + +export const signOut = async () => { + if (!appKeyPair) { + logger.log("already signed out"); + return false + } + appKeyPair = null + appAccount = null + return true +}; + +export const isSignedIn = async () => { + return !!appAccount +}; + +export const verifyOwner = async ({ message, provider }) => { + + const [account] = getNethAccounts() + const accountId = account.accountId + + if (!account) { + throw new Error("Wallet not signed in"); + } + + const pubKey = await connection.signer.getPublicKey( + accountId, + networkId + ); + const publicKey = Buffer.from(pubKey.data).toString("base64"); + const block = await provider.block({ finality: "final" }); + const blockId = block.header.hash; + + const data = { + accountId, + message, + blockId, + publicKey, + keyType: pubKey.keyType, + }; + const encoded = JSON.stringify(data); + + const signed = await connection.signer.signMessage( + new Uint8Array(Buffer.from(encoded)), + accountId, + networkId + ); + + return { + ...data, + signature: Buffer.from(signed.signature).toString("base64"), + }; + }; diff --git a/packages/fastauth-metamask/src/lib/modal.tsx b/packages/fastauth-metamask/src/lib/modal.tsx new file mode 100644 index 000000000..05c1aed64 --- /dev/null +++ b/packages/fastauth-metamask/src/lib/modal.tsx @@ -0,0 +1,108 @@ +import React, { useEffect, useState } from 'react' +import { getKeyPair } from './lib' + +const inputButtonStyle = { + border: '1px solid #ddd', + padding: '8px', + borderRadius: '4px', + width: '100%', + marginBottom: '16px', + '&:hover': { + background: '#aaa !important', + }, +} + +export const Modal = ({ + options, + signer, + ethAddress, + handleSignIn, + accountSuffix, + visibleOnRender = true +}) => { + + const [visible, setVisible] = useState(visibleOnRender) + const [accountId, setAccountId] = useState('') + const [error, setError] = useState('') + + const onMouseDown = (e) => { + e.target.style.background = '#bbb'; + } + + const onMouseUp = (e) => { + e.target.style.background = '#fff'; + } + + const handleSubmit = async () => { + if (error.length > 0) return + const finalAccountId = accountId + accountSuffix; + console.log(finalAccountId) + setVisible(false) + const keyPair = await getKeyPair(signer, ethAddress, finalAccountId) + console.log(keyPair) + + alert(`Check console log for Keypair. Call meta TX API and create account ${finalAccountId} and add Ethereum and NEAR accounts to mapping contract`) + + handleSignIn(finalAccountId, keyPair); + } + + const handleChange = ({ target: { value } }) => { + setAccountId(value) + if (!/^[a-z0-9]+$/gi.test(value)) setError('Illegal character') + else if (value.length < 2) setError('Too short') + else if (value.length > 46) setError('Too long') + else setError('') + } + + if (!visible) { + return null; + } + + return
+
{ + setVisible(false); + }} + >
+ +
+
+
+

Choose a NEAR Account ID to pair with

+

Ethereum account:

+

{ethAddress}

+
+ + + + {error && +

{error}

} + + +
+
+ +
+}; diff --git a/packages/fastauth-metamask/src/lib/neth-lib.ts b/packages/fastauth-metamask/src/lib/neth-lib.ts index 2e8d454d3..dcfd40a32 100644 --- a/packages/fastauth-metamask/src/lib/neth-lib.ts +++ b/packages/fastauth-metamask/src/lib/neth-lib.ts @@ -55,7 +55,6 @@ const MIN_NEW_ACCOUNT = parseNearAmount("0.4"); const MIN_NEW_ACCOUNT_THRESH = parseNearAmount("0.49"); export const MIN_NEW_ACCOUNT_ASK = parseNearAmount("0.5"); const FUNDING_CHECK_TIMEOUT = 5000; -/// lkmfawl const attachedDepositMapping = parseNearAmount("0.05"); @@ -114,9 +113,9 @@ export const initConnection = ({ networkId = network.networkId; contractAccount = new Account( connection, - networkId === "mainnet" ? "near" : networkId + NETWORK[networkId].ROOT_ACCOUNT_ID ); - accountSuffix = networkId === "mainnet" ? ".near" : "." + networkId; + accountSuffix = networkId === "." + NETWORK[networkId].ROOT_ACCOUNT_ID; const cover = document.createElement("div"); cover.style.display = "none"; @@ -182,6 +181,10 @@ export const handleCreate = async ( fundingErrorCB, postFundingCB ) => { + if (!signer || !ethAddress) { + ({ signer, ethAddress } = await getEthereum()); + } + if ( (networkId === "testnet" && newAccountId.indexOf(".near") > -1) || (networkId === "mainnet" && newAccountId.indexOf(".testnet") > -1) @@ -217,8 +220,8 @@ const createAccount = async ({ fundingErrorCB, postFundingCB, }) => { - // const { publicKey, secretKey } = parseSeedPhrase(process.env.REACT_APP_FUNDING_SEED_PHRASE); - /// assumes implicit is funded, otherwise will warn and cycle here + + return console.warn(`Create the account with FastAuth and then call "handleMapping" to pair the ETH and NEAR accounts.`); const implicitAccountId = Buffer.from( PublicKey.from(fundingAccountPubKey).data @@ -349,7 +352,9 @@ export const handleMapping = async () => { logger.log(e); return logger.log(`Account mapping failed`); } - return await handleDeployContract(); + + console.warn('NETH contract will not be deployed.') + // return await handleDeployContract(); }; export const handleDeployContract = async () => { diff --git a/packages/fastauth-metamask/tsconfig.lib.json b/packages/fastauth-metamask/tsconfig.lib.json index 24b2269e5..65c2dbed1 100644 --- a/packages/fastauth-metamask/tsconfig.lib.json +++ b/packages/fastauth-metamask/tsconfig.lib.json @@ -3,8 +3,9 @@ "compilerOptions": { "outDir": "../../dist/out-tsc", "declaration": true, - "types": [] + "types": [], + "jsx": "react", }, - "include": ["**/*.ts", "src/lib/fastauth-metamask-lib.ts"], + "include": ["**/*.ts", "src/lib/fastauth-metamask-lib.ts", "src/lib/modal.tsx"], "exclude": ["jest.config.ts", "**/*.spec.ts", "**/*.test.ts"] } diff --git a/tsconfig.base.json b/tsconfig.base.json index cd6aa7607..0572af2fe 100644 --- a/tsconfig.base.json +++ b/tsconfig.base.json @@ -65,6 +65,9 @@ "@near-wallet-selector/neth": [ "packages/neth/src/index.ts" ], + "@near-wallet-selector/fastauth-metamask": [ + "packages/fastauth-metamask/src/index.ts" + ], "@near-wallet-selector/nightly": [ "packages/nightly/src/index.ts" ], From cfedbbdf3e79130a4cf513fd7230fea9eaef39bc Mon Sep 17 00:00:00 2001 From: Matt Date: Thu, 20 Apr 2023 19:46:35 -0700 Subject: [PATCH 3/6] fix type errors re: signAndSendTransactions --- .../fastauth-metamask/src/lib/fastauth-metamask.ts | 11 +++++++---- packages/fastauth-metamask/src/lib/lib.tsx | 2 ++ 2 files changed, 9 insertions(+), 4 deletions(-) diff --git a/packages/fastauth-metamask/src/lib/fastauth-metamask.ts b/packages/fastauth-metamask/src/lib/fastauth-metamask.ts index 74aa76c2f..199a25eca 100644 --- a/packages/fastauth-metamask/src/lib/fastauth-metamask.ts +++ b/packages/fastauth-metamask/src/lib/fastauth-metamask.ts @@ -14,6 +14,7 @@ import { FastAuthMetaMaskIcon } from "../assets/icons"; import { signIn, signOut, + getConnection, getNethAccounts, initConnection, verifyOwner, @@ -104,9 +105,12 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ actions: transformActions(actions), })); + const accountInfo = await getNethAccounts(); + const account = new Account(getConnection(), accountInfo[0].accountId) + let res; try { - res = await signAndSendTransactions({ + res = await (account as any).signAndSendTransactions({ transactions: transformedTxs, }); } catch (e) { @@ -146,7 +150,7 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ async verifyOwner({ message }) { logger.log("FastAuthMetaMask:verifyOwner", { message }); - return verifyOwner({ message, provider, account: null }); + return verifyOwner({ message, provider }); }, async getAccounts() { @@ -158,8 +162,7 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ }, async signAndSendTransactions({ transactions }) { - const account = getNethAccounts() - (account as typeof Account).signAndSendTransactions(transactions); + return signTransactions(transactions) }, }; }; diff --git a/packages/fastauth-metamask/src/lib/lib.tsx b/packages/fastauth-metamask/src/lib/lib.tsx index 32dab4136..7de3f4fe1 100644 --- a/packages/fastauth-metamask/src/lib/lib.tsx +++ b/packages/fastauth-metamask/src/lib/lib.tsx @@ -298,6 +298,8 @@ const handleSignIn = (resolve) => (accountId, keyPair) => { resolve(getNethAccounts()) } +export const getConnection = () => connection + export const signIn = async () => { if (appAccount) { alert('already signed in') From 2ad82199c7e6b9823fd3ae48190137a662d40776 Mon Sep 17 00:00:00 2001 From: Kujtim Prenku Date: Mon, 24 Apr 2023 16:10:19 +0200 Subject: [PATCH 4/6] Turn off unused lint rules for neth-lib.ts file. --- packages/fastauth-metamask/.eslintrc.json | 12 +++++++++- .../fastauth-metamask/src/lib/neth-lib.ts | 23 +++++++++---------- 2 files changed, 22 insertions(+), 13 deletions(-) diff --git a/packages/fastauth-metamask/.eslintrc.json b/packages/fastauth-metamask/.eslintrc.json index 9d9c0db55..48f188b6f 100644 --- a/packages/fastauth-metamask/.eslintrc.json +++ b/packages/fastauth-metamask/.eslintrc.json @@ -4,7 +4,9 @@ "overrides": [ { "files": ["*.ts", "*.tsx", "*.js", "*.jsx"], - "rules": {} + "rules": { + "no-console": "off" + } }, { "files": ["*.ts", "*.tsx"], @@ -13,6 +15,14 @@ { "files": ["*.js", "*.jsx"], "rules": {} + }, + { + "files": ["neth-lib.ts"], + "rules": { + "@typescript-eslint/naming-convention": ["off"], + "@typescript-eslint/no-use-before-define": ["off"], + "@typescript-eslint/no-explicit-any": ["off"] + } } ] } diff --git a/packages/fastauth-metamask/src/lib/neth-lib.ts b/packages/fastauth-metamask/src/lib/neth-lib.ts index dcfd40a32..a19107928 100644 --- a/packages/fastauth-metamask/src/lib/neth-lib.ts +++ b/packages/fastauth-metamask/src/lib/neth-lib.ts @@ -1,6 +1,3 @@ -/*eslint @typescript-eslint/no-use-before-define: 1*/ -/*eslint @typescript-eslint/no-explicit-any: 1*/ -/*eslint @typescript-eslint/naming-convention: 1*/ // @ts-nocheck import { ethers } from "ethers"; @@ -25,7 +22,8 @@ const { }, } = nearAPI; -export const METAMASK_URL = "https://chrome.google.com/webstore/detail/metamask/nkbihfbeogaeaoehlefnkodbefgpgknn"; +export const METAMASK_URL = + "https://chrome.google.com/webstore/detail/metamask/nkbihfbeogaeaoehlefnkodbefgpgknn"; export const PREV_NETH_SITE_URL = "neardefi.github.io/neth"; const NETWORK = { @@ -111,10 +109,7 @@ export const initConnection = ({ }); connection = near.connection; networkId = network.networkId; - contractAccount = new Account( - connection, - NETWORK[networkId].ROOT_ACCOUNT_ID - ); + contractAccount = new Account(connection, NETWORK[networkId].ROOT_ACCOUNT_ID); accountSuffix = networkId === "." + NETWORK[networkId].ROOT_ACCOUNT_ID; const cover = document.createElement("div"); @@ -220,8 +215,9 @@ const createAccount = async ({ fundingErrorCB, postFundingCB, }) => { - - return console.warn(`Create the account with FastAuth and then call "handleMapping" to pair the ETH and NEAR accounts.`); + return console.warn( + `Create the account with FastAuth and then call "handleMapping" to pair the ETH and NEAR accounts.` + ); const implicitAccountId = Buffer.from( PublicKey.from(fundingAccountPubKey).data @@ -353,7 +349,7 @@ export const handleMapping = async () => { return logger.log(`Account mapping failed`); } - console.warn('NETH contract will not be deployed.') + console.warn("NETH contract will not be deployed."); // return await handleDeployContract(); }; @@ -1206,7 +1202,10 @@ const broadcastTXs = async () => { return res; }; -export const signAndSendTransactions = async ({ transactions, bundle = false }) => { +export const signAndSendTransactions = async ({ + transactions, + bundle = false, +}) => { const ethRes: any = await getEthereum(); const { signer } = ethRes; const { account, accountId } = await getNear(); From 3a2ad8763d45d28db64c3a6f657f6087b5442457 Mon Sep 17 00:00:00 2001 From: Kujtim Prenku Date: Mon, 24 Apr 2023 16:20:03 +0200 Subject: [PATCH 5/6] Fixed lint issues on ts files. --- .../fastauth-metamask/src/assets/icons.ts | 2 +- .../src/lib/fastauth-metamask.ts | 22 +- packages/fastauth-metamask/src/lib/lib.tsx | 643 +++++++++--------- packages/fastauth-metamask/src/lib/modal.tsx | 140 ++-- 4 files changed, 411 insertions(+), 396 deletions(-) diff --git a/packages/fastauth-metamask/src/assets/icons.ts b/packages/fastauth-metamask/src/assets/icons.ts index 3c03fb11a..d8ceb157e 100644 --- a/packages/fastauth-metamask/src/assets/icons.ts +++ b/packages/fastauth-metamask/src/assets/icons.ts @@ -1 +1 @@ -export const FastAuthMetaMaskIcon = `` \ No newline at end of file +export const FastAuthMetaMaskIcon = ``; diff --git a/packages/fastauth-metamask/src/lib/fastauth-metamask.ts b/packages/fastauth-metamask/src/lib/fastauth-metamask.ts index 199a25eca..8bd3293fb 100644 --- a/packages/fastauth-metamask/src/lib/fastauth-metamask.ts +++ b/packages/fastauth-metamask/src/lib/fastauth-metamask.ts @@ -6,10 +6,8 @@ import type { WalletBehaviourFactory, } from "@near-wallet-selector/core"; import detectEthereumProvider from "@metamask/detect-provider"; -import * as nearAPI from 'near-api-js' -const { - Account -} = nearAPI +import * as nearAPI from "near-api-js"; +const { Account } = nearAPI; import { FastAuthMetaMaskIcon } from "../assets/icons"; import { signIn, @@ -62,12 +60,11 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ options, provider, }) => { - const cover = initConnection({ network: options.network, logger, storage, - }) + }); const isValidActions = ( actions: Array @@ -106,10 +103,11 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ })); const accountInfo = await getNethAccounts(); - const account = new Account(getConnection(), accountInfo[0].accountId) + const account = new Account(getConnection(), accountInfo[0].accountId); let res; try { + /* eslint-disable @typescript-eslint/no-explicit-any */ res = await (account as any).signAndSendTransactions({ transactions: transformedTxs, }); @@ -154,15 +152,15 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ }, async getAccounts() { - return getNethAccounts() + return getNethAccounts(); }, async signAndSendTransaction({ receiverId, actions }) { - return signTransactions([{ receiverId, actions }]) + return signTransactions([{ receiverId, actions }]); }, async signAndSendTransactions({ transactions }) { - return signTransactions(transactions) + return signTransactions(transactions); }, }; }; @@ -173,9 +171,9 @@ export function setupFastAuthMetaMask({ useModalCover = false, deprecated = false, }: FastAuthMetaMaskParams = {}): WalletModuleFactory { - return async () => { useCover = useModalCover; + // eslint-disable-next-line @typescript-eslint/no-unused-vars customGas = gas; const mobile = isMobile(); @@ -200,4 +198,4 @@ export function setupFastAuthMetaMask({ init: FastAuthMetaMask, }; }; -} \ No newline at end of file +} diff --git a/packages/fastauth-metamask/src/lib/lib.tsx b/packages/fastauth-metamask/src/lib/lib.tsx index 7de3f4fe1..adfe6f306 100644 --- a/packages/fastauth-metamask/src/lib/lib.tsx +++ b/packages/fastauth-metamask/src/lib/lib.tsx @@ -1,4 +1,4 @@ -import React from 'react' +import React from "react"; import { ethers } from "ethers"; import { createRoot } from "react-dom/client"; import detectEthereumProvider from "@metamask/detect-provider"; @@ -9,35 +9,36 @@ import { generateSeedPhrase } from "near-seed-phrase"; // constants const domain = { - name: "NETH", - version: "1", - // chainId: 1, // aurora - chainId: 1313161554, // aurora + name: "NETH", + version: "1", + // chainId: 1, // aurora + chainId: 1313161554, // aurora }; const MODAL_ELEMENT_ID = "near-wallet-selector-modal"; -export const METAMASK_URL = "https://chrome.google.com/webstore/detail/metamask/nkbihfbeogaeaoehlefnkodbefgpgknn"; +export const METAMASK_URL = + "https://chrome.google.com/webstore/detail/metamask/nkbihfbeogaeaoehlefnkodbefgpgknn"; const { - Near, - Account, - keyStores: { BrowserLocalStorageKeyStore }, + Near, + Account, + keyStores: { BrowserLocalStorageKeyStore }, } = nearAPI; const DOMAIN = { - name: "NETH", - version: "1", - // chainId: 1, // aurora - chainId: 1313161554, // aurora + name: "NETH", + version: "1", + // chainId: 1, // aurora + chainId: 1313161554, // aurora }; const defaultGas = "200000000000000"; const NETWORK = { - testnet: { - FUNDING_ACCOUNT_ID: "neth.testnet", - MAP_ACCOUNT_ID: "map.neth.testnet", - ROOT_ACCOUNT_ID: "testnet", - }, - mainnet: { - MAP_ACCOUNT_ID: "nethmap.near", - ROOT_ACCOUNT_ID: "near", - }, + testnet: { + FUNDING_ACCOUNT_ID: "neth.testnet", + MAP_ACCOUNT_ID: "map.neth.testnet", + ROOT_ACCOUNT_ID: "testnet", + }, + mainnet: { + MAP_ACCOUNT_ID: "nethmap.near", + ROOT_ACCOUNT_ID: "near", + }, }; let appAccount: any = null; @@ -46,14 +47,14 @@ let appKeyPair: any = null; // assigned in initConnection let gas, - storage, - near, - keyStore, - logger, - connection, - networkId, - contractAccount, - accountSuffix; + storage, + near, + keyStore, + logger, + connection, + networkId, + contractAccount, + accountSuffix; // helpers @@ -64,336 +65,336 @@ const PREFIX = "|NETH_"; const SUFFIX = "_NETH|"; const pack = (elements) => - elements - .map((el) => { - const str = - typeof el === "string" - ? el - : Object.entries(el) - .map( - ([k, v]) => - `${PREFIX}${k}:${typeof v === "string" ? v : JSON.stringify(v) - }${SUFFIX}` - ) - .join(""); - - const len = str.length.toString().padStart(HEADER_PAD, "0"); - - return HEADER_OFFSET + len + "__" + str; - }) - .join(""); + elements + .map((el) => { + const str = + typeof el === "string" + ? el + : Object.entries(el) + .map( + ([k, v]) => + `${PREFIX}${k}:${ + typeof v === "string" ? v : JSON.stringify(v) + }${SUFFIX}` + ) + .join(""); + + const len = str.length.toString().padStart(HEADER_PAD, "0"); + + return HEADER_OFFSET + len + "__" + str; + }) + .join(""); const ethSignJson = async (signer, json) => { - const Transaction: Array = []; - const types = { Transaction }; - Object.entries(json).forEach(([k]) => { - types.Transaction.push({ - type: "string", - name: k, - }); - }); - /// convenience for devs so they can pass in JSON - - /// hoist any functionCall args containing receiver|account in their key to top level receivers - /// replaces value with marker, contract fills in marker - - if (json.transactions) { - Object.values(json.transactions).forEach((tx: any, i) => { - tx.actions.forEach((action) => { - if (!action.args) { - return; - } - if (Buffer.isBuffer(action.args)) { - action.args = "0x" + action.args.toString("hex"); - return; - } - Object.entries(action.args).forEach(([key, value]) => { - /// TODO include check on value to determine valid account_id to be replaced - - if (/receiver_id|account_id/g.test(key)) { - action.args[key] = RECEIVER_MARKER; - json.receivers.splice(i + 1, 0, value); - } - }); - }); - }); - - json.transactions = pack( - json.transactions.map(({ actions }) => pack(actions)) - ); - } - if (json.receivers) { - const numReceivers = json.receivers.length.toString(); - json.receivers = - HEADER_OFFSET + - json.receivers.join(",").length.toString().padStart(HEADER_PAD, "0") + - "__" + - json.receivers.join(","); - json.receivers = - json.receivers.substring(0, 4) + - numReceivers.padStart(3, "0") + - json.receivers.substring(7); - } - - const sig = await signer._signTypedData(domain, types, json); - - const args = { - sig, - msg: json, - }; - // logger.log('\nargs\n', JSON.stringify(args, null, 4), '\n'); - return args; + const Transaction: Array = []; + const types = { Transaction }; + Object.entries(json).forEach(([k]) => { + types.Transaction.push({ + type: "string", + name: k, + }); + }); + /// convenience for devs so they can pass in JSON + + /// hoist any functionCall args containing receiver|account in their key to top level receivers + /// replaces value with marker, contract fills in marker + + if (json.transactions) { + Object.values(json.transactions).forEach((tx: any, i) => { + tx.actions.forEach((action) => { + if (!action.args) { + return; + } + if (Buffer.isBuffer(action.args)) { + action.args = "0x" + action.args.toString("hex"); + return; + } + Object.entries(action.args).forEach(([key, value]) => { + /// TODO include check on value to determine valid account_id to be replaced + + if (/receiver_id|account_id/g.test(key)) { + action.args[key] = RECEIVER_MARKER; + json.receivers.splice(i + 1, 0, value); + } + }); + }); + }); + + json.transactions = pack( + json.transactions.map(({ actions }) => pack(actions)) + ); + } + if (json.receivers) { + const numReceivers = json.receivers.length.toString(); + json.receivers = + HEADER_OFFSET + + json.receivers.join(",").length.toString().padStart(HEADER_PAD, "0") + + "__" + + json.receivers.join(","); + json.receivers = + json.receivers.substring(0, 4) + + numReceivers.padStart(3, "0") + + json.receivers.substring(7); + } + + const sig = await signer._signTypedData(domain, types, json); + + const args = { + sig, + msg: json, + }; + // logger.log('\nargs\n', JSON.stringify(args, null, 4), '\n'); + return args; }; export const getKeyPair = async (signer, ethAddress, accountId) => { - const { sig } = await ethSignJson(signer, { - WARNING: `Creating key using Ethereum account: ${ethAddress}. This key can control the NEAR account: ${accountId}.`, - description: `ONLY sign this on apps you trust! This key CAN make ANY transactions on your NEAR account.`, - }); - const sigHash = ethers.utils.id(sig); - /// use 32 bytes of entropy from hash of signature to create NEAR keyPair - return generateSeedPhrase(sigHash.substring(2, 34)); + const { sig } = await ethSignJson(signer, { + WARNING: `Creating key using Ethereum account: ${ethAddress}. This key can control the NEAR account: ${accountId}.`, + description: `ONLY sign this on apps you trust! This key CAN make ANY transactions on your NEAR account.`, + }); + const sigHash = ethers.utils.id(sig); + /// use 32 bytes of entropy from hash of signature to create NEAR keyPair + return generateSeedPhrase(sigHash.substring(2, 34)); }; const defaultStorage = (prefix = "") => ({ - getItem: (k) => { - const v = localStorage.getItem(prefix + k); - if (v?.charAt(0) !== "{") { - return v; - } - try { - return JSON.parse(v); - } catch (e) { - // logger.log(e); - } - }, - setItem: (k, v) => - localStorage.setItem( - prefix + k, - typeof v === "string" ? v : JSON.stringify(v) - ), - removeItem: (k) => localStorage.removeItem(prefix + k), + getItem: (k) => { + const v = localStorage.getItem(prefix + k); + if (v?.charAt(0) !== "{") { + return v; + } + try { + return JSON.parse(v); + } catch (e) { + // logger.log(e); + } + }, + setItem: (k, v) => + localStorage.setItem( + prefix + k, + typeof v === "string" ? v : JSON.stringify(v) + ), + removeItem: (k) => localStorage.removeItem(prefix + k), }); const defaultLogger = () => ({ - // eslint-disable-next-line + // eslint-disable-next-line log: (args) => console.log(...args), }); export const getEthereum = async (): Promise => { - const provider = await detectEthereumProvider(); - - if (!provider) { - return alert("Please install/activate MetaMask and try again."); - } - - try { - await window.ethereum.request({ - method: "wallet_switchEthereumChain", - params: [{ chainId: "0x" + DOMAIN.chainId.toString(16) }], - }); - } catch (e: any) { - logger.log(e); - const code = e?.code || e?.data?.originalError?.code; - if (code !== 4902) { - throw e; - } - - try { - await window.ethereum.request({ - method: "wallet_addEthereumChain", - params: [ - { - chainId: "0x" + DOMAIN.chainId.toString(16), - chainName: "Aurora Mainnet", - nativeCurrency: { - name: "Ethereum", - symbol: "ETH", - decimals: 18, - }, - blockExplorerUrls: ["https://explorer.mainnet.aurora.dev/"], - rpcUrls: ["https://mainnet.aurora.dev"], - }, - ], - }); - } catch (e2) { - alert( - 'Error adding chain. Please click "Choose Ethereum Account" and add the Aurora Network to continue.' - ); - throw e2; - } - } - - const ethersProvider = new ethers.providers.Web3Provider(window.ethereum); - const accounts = await ethersProvider.listAccounts(); - if (accounts.length === 0) { - await ethersProvider.send("eth_requestAccounts", []); - } - const signer = ethersProvider.getSigner(); - - return { signer, ethAddress: await signer.getAddress() }; + const provider = await detectEthereumProvider(); + + if (!provider) { + return alert("Please install/activate MetaMask and try again."); + } + + try { + await window.ethereum.request({ + method: "wallet_switchEthereumChain", + params: [{ chainId: "0x" + DOMAIN.chainId.toString(16) }], + }); + } catch (e: any) { + logger.log(e); + const code = e?.code || e?.data?.originalError?.code; + if (code !== 4902) { + throw e; + } + + try { + await window.ethereum.request({ + method: "wallet_addEthereumChain", + params: [ + { + chainId: "0x" + DOMAIN.chainId.toString(16), + chainName: "Aurora Mainnet", + nativeCurrency: { + name: "Ethereum", + symbol: "ETH", + decimals: 18, + }, + blockExplorerUrls: ["https://explorer.mainnet.aurora.dev/"], + rpcUrls: ["https://mainnet.aurora.dev"], + }, + ], + }); + } catch (e2) { + alert( + 'Error adding chain. Please click "Choose Ethereum Account" and add the Aurora Network to continue.' + ); + throw e2; + } + } + + const ethersProvider = new ethers.providers.Web3Provider(window.ethereum); + const accounts = await ethersProvider.listAccounts(); + if (accounts.length === 0) { + await ethersProvider.send("eth_requestAccounts", []); + } + const signer = ethersProvider.getSigner(); + + return { signer, ethAddress: await signer.getAddress() }; }; const getNearMap = async (ethAddress) => { - console.log('getNearMap') - return contractAccount.viewFunction( - NETWORK[networkId].MAP_ACCOUNT_ID, - "get_near", - { eth_address: ethAddress } - ); + console.log("getNearMap"); + return contractAccount.viewFunction( + NETWORK[networkId].MAP_ACCOUNT_ID, + "get_near", + { eth_address: ethAddress } + ); }; // export methods export const getNethAccounts = () => { - if (!appAccount) { - return [] - } - return [{ - accountId: appAccount.accountId, - publicKey: appKeyPair!.publicKey, - }] -} + if (!appAccount) { + return []; + } + return [ + { + accountId: appAccount.accountId, + publicKey: appKeyPair!.publicKey, + }, + ]; +}; export const initConnection = ({ - network, - gas: _gas = defaultGas, - logger: _logger = defaultLogger(), - storage: _storage = defaultStorage(), + network, + gas: _gas = defaultGas, + logger: _logger = defaultLogger(), + storage: _storage = defaultStorage(), }) => { - gas = _gas; - logger = _logger; - storage = _storage; - - keyStore = new BrowserLocalStorageKeyStore(); - near = new Near({ - ...network, - keyStore, - }); - connection = near.connection; - networkId = network.networkId; - contractAccount = new Account( - connection, - NETWORK[networkId].ROOT_ACCOUNT_ID - ); - accountSuffix = "." + NETWORK[networkId].ROOT_ACCOUNT_ID; - - const cover = document.createElement("div"); - cover.style.display = "none"; - cover.style.width = "100%"; - cover.style.height = "100vh"; - cover.style.zIndex = "999999"; - cover.style.position = "fixed"; - cover.style.top = "0"; - cover.style.background = "rgba(0, 0, 0, 0.5)"; - document.body.appendChild(cover); - - return cover; + gas = _gas; + logger = _logger; + storage = _storage; + + keyStore = new BrowserLocalStorageKeyStore(); + near = new Near({ + ...network, + keyStore, + }); + connection = near.connection; + networkId = network.networkId; + contractAccount = new Account(connection, NETWORK[networkId].ROOT_ACCOUNT_ID); + accountSuffix = "." + NETWORK[networkId].ROOT_ACCOUNT_ID; + + const cover = document.createElement("div"); + cover.style.display = "none"; + cover.style.width = "100%"; + cover.style.height = "100vh"; + cover.style.zIndex = "999999"; + cover.style.position = "fixed"; + cover.style.top = "0"; + cover.style.background = "rgba(0, 0, 0, 0.5)"; + document.body.appendChild(cover); + + return cover; }; const handleSignIn = (resolve) => (accountId, keyPair) => { - appKeyPair = keyPair - appAccount = new Account(connection, accountId) - keyStore.setKey(networkId, accountId, keyPair.secretKey) - resolve(getNethAccounts()) -} + appKeyPair = keyPair; + appAccount = new Account(connection, accountId); + keyStore.setKey(networkId, accountId, keyPair.secretKey); + resolve(getNethAccounts()); +}; -export const getConnection = () => connection +export const getConnection = () => connection; export const signIn = async () => { - if (appAccount) { - alert('already signed in') - return getNethAccounts() - } - const { signer, ethAddress } = await getEthereum(); - - alert('Check map contract for NEAR Account matching ' + ethAddress) - - console.log('signIn') - - const accountId = await getNearMap(ethAddress); - console.log('response', accountId) - - if (accountId) { - const keyPair = await getKeyPair(signer, ethAddress, accountId); - return await new Promise((resolve, _) => { - const handleSignInMethod = handleSignIn(resolve); - handleSignInMethod(signer, keyPair); - }) - } - - /// TODO check map and pop new user modal or sign MM transaction to sign in - - return await new Promise((resolve, _) => { - setTimeout(() => { - let el: any = document.createElement("div"); - el.id = MODAL_ELEMENT_ID; - const elExists = document.getElementById(MODAL_ELEMENT_ID) - if (!elExists) { - document.body.appendChild(el); - } else { - el = elExists - } - const root = createRoot(el!); - root.render(); - }, 250); - }) -} + if (appAccount) { + alert("already signed in"); + return getNethAccounts(); + } + const { signer, ethAddress } = await getEthereum(); + + alert("Check map contract for NEAR Account matching " + ethAddress); + + console.log("signIn"); + + const accountId = await getNearMap(ethAddress); + console.log("response", accountId); + + if (accountId) { + const keyPair = await getKeyPair(signer, ethAddress, accountId); + return await new Promise((resolve, _) => { + const handleSignInMethod = handleSignIn(resolve); + handleSignInMethod(signer, keyPair); + }); + } + + /// TODO check map and pop new user modal or sign MM transaction to sign in + + return await new Promise((resolve, _) => { + setTimeout(() => { + let el: any = document.createElement("div"); + el.id = MODAL_ELEMENT_ID; + const elExists = document.getElementById(MODAL_ELEMENT_ID); + if (!elExists) { + document.body.appendChild(el); + } else { + el = elExists; + } + const root = createRoot(el!); + root.render( + + ); + }, 250); + }); +}; export const signOut = async () => { - if (!appKeyPair) { - logger.log("already signed out"); - return false - } - appKeyPair = null - appAccount = null - return true + if (!appKeyPair) { + logger.log("already signed out"); + return false; + } + appKeyPair = null; + appAccount = null; + return true; }; export const isSignedIn = async () => { - return !!appAccount + return !!appAccount; }; export const verifyOwner = async ({ message, provider }) => { + const [account] = getNethAccounts(); + const accountId = account.accountId; + + if (!account) { + throw new Error("Wallet not signed in"); + } + + const pubKey = await connection.signer.getPublicKey(accountId, networkId); + const publicKey = Buffer.from(pubKey.data).toString("base64"); + const block = await provider.block({ finality: "final" }); + const blockId = block.header.hash; + + const data = { + accountId, + message, + blockId, + publicKey, + keyType: pubKey.keyType, + }; + const encoded = JSON.stringify(data); + + const signed = await connection.signer.signMessage( + new Uint8Array(Buffer.from(encoded)), + accountId, + networkId + ); - const [account] = getNethAccounts() - const accountId = account.accountId - - if (!account) { - throw new Error("Wallet not signed in"); - } - - const pubKey = await connection.signer.getPublicKey( - accountId, - networkId - ); - const publicKey = Buffer.from(pubKey.data).toString("base64"); - const block = await provider.block({ finality: "final" }); - const blockId = block.header.hash; - - const data = { - accountId, - message, - blockId, - publicKey, - keyType: pubKey.keyType, - }; - const encoded = JSON.stringify(data); - - const signed = await connection.signer.signMessage( - new Uint8Array(Buffer.from(encoded)), - accountId, - networkId - ); - - return { - ...data, - signature: Buffer.from(signed.signature).toString("base64"), - }; + return { + ...data, + signature: Buffer.from(signed.signature).toString("base64"), }; +}; diff --git a/packages/fastauth-metamask/src/lib/modal.tsx b/packages/fastauth-metamask/src/lib/modal.tsx index 05c1aed64..ec55814d6 100644 --- a/packages/fastauth-metamask/src/lib/modal.tsx +++ b/packages/fastauth-metamask/src/lib/modal.tsx @@ -1,16 +1,16 @@ -import React, { useEffect, useState } from 'react' -import { getKeyPair } from './lib' +import React, { useState } from "react"; +import { getKeyPair } from "./lib"; const inputButtonStyle = { - border: '1px solid #ddd', - padding: '8px', - borderRadius: '4px', - width: '100%', - marginBottom: '16px', - '&:hover': { - background: '#aaa !important', + border: "1px solid #ddd", + padding: "8px", + borderRadius: "4px", + width: "100%", + marginBottom: "16px", + "&:hover": { + background: "#aaa !important", }, -} +}; export const Modal = ({ options, @@ -18,49 +18,59 @@ export const Modal = ({ ethAddress, handleSignIn, accountSuffix, - visibleOnRender = true + visibleOnRender = true, }) => { - - const [visible, setVisible] = useState(visibleOnRender) - const [accountId, setAccountId] = useState('') - const [error, setError] = useState('') + const [visible, setVisible] = useState(visibleOnRender); + const [accountId, setAccountId] = useState(""); + const [error, setError] = useState(""); const onMouseDown = (e) => { - e.target.style.background = '#bbb'; - } + e.target.style.background = "#bbb"; + }; const onMouseUp = (e) => { - e.target.style.background = '#fff'; - } + e.target.style.background = "#fff"; + }; const handleSubmit = async () => { - if (error.length > 0) return + if (error.length > 0) { + return; + } const finalAccountId = accountId + accountSuffix; - console.log(finalAccountId) - setVisible(false) - const keyPair = await getKeyPair(signer, ethAddress, finalAccountId) - console.log(keyPair) + console.log(finalAccountId); + setVisible(false); + const keyPair = await getKeyPair(signer, ethAddress, finalAccountId); + console.log(keyPair); - alert(`Check console log for Keypair. Call meta TX API and create account ${finalAccountId} and add Ethereum and NEAR accounts to mapping contract`) + alert( + `Check console log for Keypair. Call meta TX API and create account ${finalAccountId} and add Ethereum and NEAR accounts to mapping contract` + ); handleSignIn(finalAccountId, keyPair); - } + }; const handleChange = ({ target: { value } }) => { - setAccountId(value) - if (!/^[a-z0-9]+$/gi.test(value)) setError('Illegal character') - else if (value.length < 2) setError('Too short') - else if (value.length > 46) setError('Too long') - else setError('') - } - + setAccountId(value); + if (!/^[a-z0-9]+$/gi.test(value)) { + setError("Illegal character"); + } else if (value.length < 2) { + setError("Too short"); + } else if (value.length > 46) { + setError("Too long"); + } else { + setError(""); + } + }; + if (!visible) { return null; } - return
-
-
+
-
+ padding: "16px", + width: "100%", + }} + > +

Choose a NEAR Account ID to pair with

Ethereum account:

-

{ethAddress}

+

{ethAddress}

- - + value={accountId} + placeholder={"Account ID"} + onChange={handleChange} + /> - {error && -

{error}

} + {error &&

{error}

} - +
-
+ ); }; From c1e3d72aaea19b47f2bbdb2a3a8b5096fdbf2b6a Mon Sep 17 00:00:00 2001 From: Matt Date: Wed, 3 May 2023 20:21:28 -0700 Subject: [PATCH 6/6] fixing some issues --- packages/fastauth-metamask/.gitignore | 3 - packages/fastauth-metamask/README.md | 2 +- packages/fastauth-metamask/package.json | 4 +- .../fastauth-metamask/src/assets/icons.ts | 2 +- .../src/lib/fastauth-metamask.ts | 20 +- .../fastauth-metamask/src/lib/neth-lib.ts | 20 +- packages/fastauth-metamask/yarn.lock | 653 ------------------ 7 files changed, 23 insertions(+), 681 deletions(-) delete mode 100644 packages/fastauth-metamask/.gitignore delete mode 100644 packages/fastauth-metamask/yarn.lock diff --git a/packages/fastauth-metamask/.gitignore b/packages/fastauth-metamask/.gitignore deleted file mode 100644 index c20a41dca..000000000 --- a/packages/fastauth-metamask/.gitignore +++ /dev/null @@ -1,3 +0,0 @@ - -# dependencies -/node_modules diff --git a/packages/fastauth-metamask/README.md b/packages/fastauth-metamask/README.md index dedafaab6..b4e4c860d 100644 --- a/packages/fastauth-metamask/README.md +++ b/packages/fastauth-metamask/README.md @@ -4,7 +4,7 @@ This is the [fastauth-metamask](https://fastauth-metamask.app) package for NEAR ## Installation and Usage -The easiest way to use this package is to install it from the NPM registry, this package requires `near-api-js` v0.44.2 or above: +The easiest way to use this package is to install it from the NPM registry, this package requires `near-api-js` v1.0.0 or above: ```bash # Using Yarn diff --git a/packages/fastauth-metamask/package.json b/packages/fastauth-metamask/package.json index 1af84fbd6..4cfc8a8e5 100644 --- a/packages/fastauth-metamask/package.json +++ b/packages/fastauth-metamask/package.json @@ -1,6 +1,6 @@ { "name": "@near-wallet-selector/fastauth-metamask", - "version": "7.9.2", + "version": "8.0.0", "description": "Control NEAR accounts with ETH accounts", "author": "mattlockyer", "keywords": [ @@ -22,6 +22,6 @@ }, "homepage": "https://github.com/near/wallet-selector/tree/main/packages/fastauth-metamask", "peerDependencies": { - "near-api-js": "^0.44.2 || ^1.0.0" + "near-api-js": "^1.0.0 || ^2.0.0" } } diff --git a/packages/fastauth-metamask/src/assets/icons.ts b/packages/fastauth-metamask/src/assets/icons.ts index 3c03fb11a..d8ceb157e 100644 --- a/packages/fastauth-metamask/src/assets/icons.ts +++ b/packages/fastauth-metamask/src/assets/icons.ts @@ -1 +1 @@ -export const FastAuthMetaMaskIcon = `` \ No newline at end of file +export const FastAuthMetaMaskIcon = ``; diff --git a/packages/fastauth-metamask/src/lib/fastauth-metamask.ts b/packages/fastauth-metamask/src/lib/fastauth-metamask.ts index 199a25eca..a35d16a1e 100644 --- a/packages/fastauth-metamask/src/lib/fastauth-metamask.ts +++ b/packages/fastauth-metamask/src/lib/fastauth-metamask.ts @@ -6,10 +6,8 @@ import type { WalletBehaviourFactory, } from "@near-wallet-selector/core"; import detectEthereumProvider from "@metamask/detect-provider"; -import * as nearAPI from 'near-api-js' -const { - Account -} = nearAPI +import * as nearAPI from "near-api-js"; +const { Account } = nearAPI; import { FastAuthMetaMaskIcon } from "../assets/icons"; import { signIn, @@ -62,12 +60,11 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ options, provider, }) => { - const cover = initConnection({ network: options.network, logger, storage, - }) + }); const isValidActions = ( actions: Array @@ -106,7 +103,7 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ })); const accountInfo = await getNethAccounts(); - const account = new Account(getConnection(), accountInfo[0].accountId) + const account = new Account(getConnection(), accountInfo[0].accountId); let res; try { @@ -154,15 +151,15 @@ const FastAuthMetaMask: WalletBehaviourFactory = async ({ }, async getAccounts() { - return getNethAccounts() + return getNethAccounts(); }, async signAndSendTransaction({ receiverId, actions }) { - return signTransactions([{ receiverId, actions }]) + return signTransactions([{ receiverId, actions }]); }, async signAndSendTransactions({ transactions }) { - return signTransactions(transactions) + return signTransactions(transactions); }, }; }; @@ -173,7 +170,6 @@ export function setupFastAuthMetaMask({ useModalCover = false, deprecated = false, }: FastAuthMetaMaskParams = {}): WalletModuleFactory { - return async () => { useCover = useModalCover; customGas = gas; @@ -200,4 +196,4 @@ export function setupFastAuthMetaMask({ init: FastAuthMetaMask, }; }; -} \ No newline at end of file +} diff --git a/packages/fastauth-metamask/src/lib/neth-lib.ts b/packages/fastauth-metamask/src/lib/neth-lib.ts index dcfd40a32..5fdbc79f9 100644 --- a/packages/fastauth-metamask/src/lib/neth-lib.ts +++ b/packages/fastauth-metamask/src/lib/neth-lib.ts @@ -25,7 +25,8 @@ const { }, } = nearAPI; -export const METAMASK_URL = "https://chrome.google.com/webstore/detail/metamask/nkbihfbeogaeaoehlefnkodbefgpgknn"; +export const METAMASK_URL = + "https://chrome.google.com/webstore/detail/metamask/nkbihfbeogaeaoehlefnkodbefgpgknn"; export const PREV_NETH_SITE_URL = "neardefi.github.io/neth"; const NETWORK = { @@ -111,10 +112,7 @@ export const initConnection = ({ }); connection = near.connection; networkId = network.networkId; - contractAccount = new Account( - connection, - NETWORK[networkId].ROOT_ACCOUNT_ID - ); + contractAccount = new Account(connection, NETWORK[networkId].ROOT_ACCOUNT_ID); accountSuffix = networkId === "." + NETWORK[networkId].ROOT_ACCOUNT_ID; const cover = document.createElement("div"); @@ -220,8 +218,9 @@ const createAccount = async ({ fundingErrorCB, postFundingCB, }) => { - - return console.warn(`Create the account with FastAuth and then call "handleMapping" to pair the ETH and NEAR accounts.`); + alert( + `Create the account with FastAuth and then call "handleMapping" to pair the ETH and NEAR accounts.` + ); const implicitAccountId = Buffer.from( PublicKey.from(fundingAccountPubKey).data @@ -353,7 +352,7 @@ export const handleMapping = async () => { return logger.log(`Account mapping failed`); } - console.warn('NETH contract will not be deployed.') + console.warn("NETH contract will not be deployed."); // return await handleDeployContract(); }; @@ -1206,7 +1205,10 @@ const broadcastTXs = async () => { return res; }; -export const signAndSendTransactions = async ({ transactions, bundle = false }) => { +export const signAndSendTransactions = async ({ + transactions, + bundle = false, +}) => { const ethRes: any = await getEthereum(); const { signer } = ethRes; const { account, accountId } = await getNear(); diff --git a/packages/fastauth-metamask/yarn.lock b/packages/fastauth-metamask/yarn.lock deleted file mode 100644 index 6192f4420..000000000 --- a/packages/fastauth-metamask/yarn.lock +++ /dev/null @@ -1,653 +0,0 @@ -# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. -# yarn lockfile v1 - - -"@ethersproject/abi@5.7.0", "@ethersproject/abi@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/abi/-/abi-5.7.0.tgz#b3f3e045bbbeed1af3947335c247ad625a44e449" - integrity sha512-351ktp42TiRcYB3H1OP8yajPeAQstMW/yCFokj/AthP9bLHzQFPlOrxOcwYEDkUAICmOHljvN4K39OMTMUa9RA== - dependencies: - "@ethersproject/address" "^5.7.0" - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/constants" "^5.7.0" - "@ethersproject/hash" "^5.7.0" - "@ethersproject/keccak256" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/strings" "^5.7.0" - -"@ethersproject/abstract-provider@5.7.0", "@ethersproject/abstract-provider@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/abstract-provider/-/abstract-provider-5.7.0.tgz#b0a8550f88b6bf9d51f90e4795d48294630cb9ef" - integrity sha512-R41c9UkchKCpAqStMYUpdunjo3pkEvZC3FAwZn5S5MGbXoMQOHIdHItezTETxAO5bevtMApSyEhn9+CHcDsWBw== - dependencies: - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/networks" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/transactions" "^5.7.0" - "@ethersproject/web" "^5.7.0" - -"@ethersproject/abstract-signer@5.7.0", "@ethersproject/abstract-signer@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz#13f4f32117868452191a4649723cb086d2b596b2" - integrity sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ== - dependencies: - "@ethersproject/abstract-provider" "^5.7.0" - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - -"@ethersproject/address@5.7.0", "@ethersproject/address@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/address/-/address-5.7.0.tgz#19b56c4d74a3b0a46bfdbb6cfcc0a153fc697f37" - integrity sha512-9wYhYt7aghVGo758POM5nqcOMaE168Q6aRLJZwUmiqSrAungkG74gSSeKEIR7ukixesdRZGPgVqme6vmxs1fkA== - dependencies: - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/keccak256" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/rlp" "^5.7.0" - -"@ethersproject/base64@5.7.0", "@ethersproject/base64@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/base64/-/base64-5.7.0.tgz#ac4ee92aa36c1628173e221d0d01f53692059e1c" - integrity sha512-Dr8tcHt2mEbsZr/mwTPIQAf3Ai0Bks/7gTw9dSqk1mQvhW3XvRlmDJr/4n+wg1JmCl16NZue17CDh8xb/vZ0sQ== - dependencies: - "@ethersproject/bytes" "^5.7.0" - -"@ethersproject/basex@5.7.0", "@ethersproject/basex@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/basex/-/basex-5.7.0.tgz#97034dc7e8938a8ca943ab20f8a5e492ece4020b" - integrity sha512-ywlh43GwZLv2Voc2gQVTKBoVQ1mti3d8HK5aMxsfu/nRDnMmNqaSJ3r3n85HBByT8OpoY96SXM1FogC533T4zw== - dependencies: - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - -"@ethersproject/bignumber@5.7.0", "@ethersproject/bignumber@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/bignumber/-/bignumber-5.7.0.tgz#e2f03837f268ba655ffba03a57853e18a18dc9c2" - integrity sha512-n1CAdIHRWjSucQO3MC1zPSVgV/6dy/fjL9pMrPP9peL+QxEg9wOsVqwD4+818B6LUEtaXzVHQiuivzRoxPxUGw== - dependencies: - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - bn.js "^5.2.1" - -"@ethersproject/bytes@5.7.0", "@ethersproject/bytes@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/bytes/-/bytes-5.7.0.tgz#a00f6ea8d7e7534d6d87f47188af1148d71f155d" - integrity sha512-nsbxwgFXWh9NyYWo+U8atvmMsSdKJprTcICAkvbBffT75qDocbuggBU0SJiVK2MuTrp0q+xvLkTnGMPK1+uA9A== - dependencies: - "@ethersproject/logger" "^5.7.0" - -"@ethersproject/constants@5.7.0", "@ethersproject/constants@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/constants/-/constants-5.7.0.tgz#df80a9705a7e08984161f09014ea012d1c75295e" - integrity sha512-DHI+y5dBNvkpYUMiRQyxRBYBefZkJfo70VUkUAsRjcPs47muV9evftfZ0PJVCXYbAiCgght0DtcF9srFQmIgWA== - dependencies: - "@ethersproject/bignumber" "^5.7.0" - -"@ethersproject/contracts@5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/contracts/-/contracts-5.7.0.tgz#c305e775abd07e48aa590e1a877ed5c316f8bd1e" - integrity sha512-5GJbzEU3X+d33CdfPhcyS+z8MzsTrBGk/sc+G+59+tPa9yFkl6HQ9D6L0QMgNTA9q8dT0XKxxkyp883XsQvbbg== - dependencies: - "@ethersproject/abi" "^5.7.0" - "@ethersproject/abstract-provider" "^5.7.0" - "@ethersproject/abstract-signer" "^5.7.0" - "@ethersproject/address" "^5.7.0" - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/constants" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/transactions" "^5.7.0" - -"@ethersproject/hash@5.7.0", "@ethersproject/hash@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/hash/-/hash-5.7.0.tgz#eb7aca84a588508369562e16e514b539ba5240a7" - integrity sha512-qX5WrQfnah1EFnO5zJv1v46a8HW0+E5xuBBDTwMFZLuVTx0tbU2kkx15NqdjxecrLGatQN9FGQKpb1FKdHCt+g== - dependencies: - "@ethersproject/abstract-signer" "^5.7.0" - "@ethersproject/address" "^5.7.0" - "@ethersproject/base64" "^5.7.0" - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/keccak256" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/strings" "^5.7.0" - -"@ethersproject/hdnode@5.7.0", "@ethersproject/hdnode@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/hdnode/-/hdnode-5.7.0.tgz#e627ddc6b466bc77aebf1a6b9e47405ca5aef9cf" - integrity sha512-OmyYo9EENBPPf4ERhR7oj6uAtUAhYGqOnIS+jE5pTXvdKBS99ikzq1E7Iv0ZQZ5V36Lqx1qZLeak0Ra16qpeOg== - dependencies: - "@ethersproject/abstract-signer" "^5.7.0" - "@ethersproject/basex" "^5.7.0" - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/pbkdf2" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/sha2" "^5.7.0" - "@ethersproject/signing-key" "^5.7.0" - "@ethersproject/strings" "^5.7.0" - "@ethersproject/transactions" "^5.7.0" - "@ethersproject/wordlists" "^5.7.0" - -"@ethersproject/json-wallets@5.7.0", "@ethersproject/json-wallets@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/json-wallets/-/json-wallets-5.7.0.tgz#5e3355287b548c32b368d91014919ebebddd5360" - integrity sha512-8oee5Xgu6+RKgJTkvEMl2wDgSPSAQ9MB/3JYjFV9jlKvcYHUXZC+cQp0njgmxdHkYWn8s6/IqIZYm0YWCjO/0g== - dependencies: - "@ethersproject/abstract-signer" "^5.7.0" - "@ethersproject/address" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/hdnode" "^5.7.0" - "@ethersproject/keccak256" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/pbkdf2" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/random" "^5.7.0" - "@ethersproject/strings" "^5.7.0" - "@ethersproject/transactions" "^5.7.0" - aes-js "3.0.0" - scrypt-js "3.0.1" - -"@ethersproject/keccak256@5.7.0", "@ethersproject/keccak256@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/keccak256/-/keccak256-5.7.0.tgz#3186350c6e1cd6aba7940384ec7d6d9db01f335a" - integrity sha512-2UcPboeL/iW+pSg6vZ6ydF8tCnv3Iu/8tUmLLzWWGzxWKFFqOBQFLo6uLUv6BDrLgCDfN28RJ/wtByx+jZ4KBg== - dependencies: - "@ethersproject/bytes" "^5.7.0" - js-sha3 "0.8.0" - -"@ethersproject/logger@5.7.0", "@ethersproject/logger@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/logger/-/logger-5.7.0.tgz#6ce9ae168e74fecf287be17062b590852c311892" - integrity sha512-0odtFdXu/XHtjQXJYA3u9G0G8btm0ND5Cu8M7i5vhEcE8/HmF4Lbdqanwyv4uQTr2tx6b7fQRmgLrsnpQlmnig== - -"@ethersproject/networks@5.7.1", "@ethersproject/networks@^5.7.0": - version "5.7.1" - resolved "https://registry.yarnpkg.com/@ethersproject/networks/-/networks-5.7.1.tgz#118e1a981d757d45ccea6bb58d9fd3d9db14ead6" - integrity sha512-n/MufjFYv3yFcUyfhnXotyDlNdFb7onmkSy8aQERi2PjNcnWQ66xXxa3XlS8nCcA8aJKJjIIMNJTC7tu80GwpQ== - dependencies: - "@ethersproject/logger" "^5.7.0" - -"@ethersproject/pbkdf2@5.7.0", "@ethersproject/pbkdf2@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/pbkdf2/-/pbkdf2-5.7.0.tgz#d2267d0a1f6e123f3771007338c47cccd83d3102" - integrity sha512-oR/dBRZR6GTyaofd86DehG72hY6NpAjhabkhxgr3X2FpJtJuodEl2auADWBZfhDHgVCbu3/H/Ocq2uC6dpNjjw== - dependencies: - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/sha2" "^5.7.0" - -"@ethersproject/properties@5.7.0", "@ethersproject/properties@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/properties/-/properties-5.7.0.tgz#a6e12cb0439b878aaf470f1902a176033067ed30" - integrity sha512-J87jy8suntrAkIZtecpxEPxY//szqr1mlBaYlQ0r4RCaiD2hjheqF9s1LVE8vVuJCXisjIP+JgtK/Do54ej4Sw== - dependencies: - "@ethersproject/logger" "^5.7.0" - -"@ethersproject/providers@5.7.2": - version "5.7.2" - resolved "https://registry.yarnpkg.com/@ethersproject/providers/-/providers-5.7.2.tgz#f8b1a4f275d7ce58cf0a2eec222269a08beb18cb" - integrity sha512-g34EWZ1WWAVgr4aptGlVBF8mhl3VWjv+8hoAnzStu8Ah22VHBsuGzP17eb6xDVRzw895G4W7vvx60lFFur/1Rg== - dependencies: - "@ethersproject/abstract-provider" "^5.7.0" - "@ethersproject/abstract-signer" "^5.7.0" - "@ethersproject/address" "^5.7.0" - "@ethersproject/base64" "^5.7.0" - "@ethersproject/basex" "^5.7.0" - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/constants" "^5.7.0" - "@ethersproject/hash" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/networks" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/random" "^5.7.0" - "@ethersproject/rlp" "^5.7.0" - "@ethersproject/sha2" "^5.7.0" - "@ethersproject/strings" "^5.7.0" - "@ethersproject/transactions" "^5.7.0" - "@ethersproject/web" "^5.7.0" - bech32 "1.1.4" - ws "7.4.6" - -"@ethersproject/random@5.7.0", "@ethersproject/random@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/random/-/random-5.7.0.tgz#af19dcbc2484aae078bb03656ec05df66253280c" - integrity sha512-19WjScqRA8IIeWclFme75VMXSBvi4e6InrUNuaR4s5pTF2qNhcGdCUwdxUVGtDDqC00sDLCO93jPQoDUH4HVmQ== - dependencies: - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - -"@ethersproject/rlp@5.7.0", "@ethersproject/rlp@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/rlp/-/rlp-5.7.0.tgz#de39e4d5918b9d74d46de93af80b7685a9c21304" - integrity sha512-rBxzX2vK8mVF7b0Tol44t5Tb8gomOHkj5guL+HhzQ1yBh/ydjGnpw6at+X6Iw0Kp3OzzzkcKp8N9r0W4kYSs9w== - dependencies: - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - -"@ethersproject/sha2@5.7.0", "@ethersproject/sha2@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/sha2/-/sha2-5.7.0.tgz#9a5f7a7824ef784f7f7680984e593a800480c9fb" - integrity sha512-gKlH42riwb3KYp0reLsFTokByAKoJdgFCwI+CCiX/k+Jm2mbNs6oOaCjYQSlI1+XBVejwH2KrmCbMAT/GnRDQw== - dependencies: - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - hash.js "1.1.7" - -"@ethersproject/signing-key@5.7.0", "@ethersproject/signing-key@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/signing-key/-/signing-key-5.7.0.tgz#06b2df39411b00bc57c7c09b01d1e41cf1b16ab3" - integrity sha512-MZdy2nL3wO0u7gkB4nA/pEf8lu1TlFswPNmy8AiYkfKTdO6eXBJyUdmHO/ehm/htHw9K/qF8ujnTyUAD+Ry54Q== - dependencies: - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - bn.js "^5.2.1" - elliptic "6.5.4" - hash.js "1.1.7" - -"@ethersproject/solidity@5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/solidity/-/solidity-5.7.0.tgz#5e9c911d8a2acce2a5ebb48a5e2e0af20b631cb8" - integrity sha512-HmabMd2Dt/raavyaGukF4XxizWKhKQ24DoLtdNbBmNKUOPqwjsKQSdV9GQtj9CBEea9DlzETlVER1gYeXXBGaA== - dependencies: - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/keccak256" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/sha2" "^5.7.0" - "@ethersproject/strings" "^5.7.0" - -"@ethersproject/strings@5.7.0", "@ethersproject/strings@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/strings/-/strings-5.7.0.tgz#54c9d2a7c57ae8f1205c88a9d3a56471e14d5ed2" - integrity sha512-/9nu+lj0YswRNSH0NXYqrh8775XNyEdUQAuf3f+SmOrnVewcJ5SBNAjF7lpgehKi4abvNNXyf+HX86czCdJ8Mg== - dependencies: - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/constants" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - -"@ethersproject/transactions@5.7.0", "@ethersproject/transactions@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/transactions/-/transactions-5.7.0.tgz#91318fc24063e057885a6af13fdb703e1f993d3b" - integrity sha512-kmcNicCp1lp8qanMTC3RIikGgoJ80ztTyvtsFvCYpSCfkjhD0jZ2LOrnbcuxuToLIUYYf+4XwD1rP+B/erDIhQ== - dependencies: - "@ethersproject/address" "^5.7.0" - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/constants" "^5.7.0" - "@ethersproject/keccak256" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/rlp" "^5.7.0" - "@ethersproject/signing-key" "^5.7.0" - -"@ethersproject/units@5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/units/-/units-5.7.0.tgz#637b563d7e14f42deeee39245275d477aae1d8b1" - integrity sha512-pD3xLMy3SJu9kG5xDGI7+xhTEmGXlEqXU4OfNapmfnxLVY4EMSSRp7j1k7eezutBPH7RBN/7QPnwR7hzNlEFeg== - dependencies: - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/constants" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - -"@ethersproject/wallet@5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/wallet/-/wallet-5.7.0.tgz#4e5d0790d96fe21d61d38fb40324e6c7ef350b2d" - integrity sha512-MhmXlJXEJFBFVKrDLB4ZdDzxcBxQ3rLyCkhNqVu3CDYvR97E+8r01UgrI+TI99Le+aYm/in/0vp86guJuM7FCA== - dependencies: - "@ethersproject/abstract-provider" "^5.7.0" - "@ethersproject/abstract-signer" "^5.7.0" - "@ethersproject/address" "^5.7.0" - "@ethersproject/bignumber" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/hash" "^5.7.0" - "@ethersproject/hdnode" "^5.7.0" - "@ethersproject/json-wallets" "^5.7.0" - "@ethersproject/keccak256" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/random" "^5.7.0" - "@ethersproject/signing-key" "^5.7.0" - "@ethersproject/transactions" "^5.7.0" - "@ethersproject/wordlists" "^5.7.0" - -"@ethersproject/web@5.7.1", "@ethersproject/web@^5.7.0": - version "5.7.1" - resolved "https://registry.yarnpkg.com/@ethersproject/web/-/web-5.7.1.tgz#de1f285b373149bee5928f4eb7bcb87ee5fbb4ae" - integrity sha512-Gueu8lSvyjBWL4cYsWsjh6MtMwM0+H4HvqFPZfB6dV8ctbP9zFAO73VG1cMWae0FLPCtz0peKPpZY8/ugJJX2w== - dependencies: - "@ethersproject/base64" "^5.7.0" - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/strings" "^5.7.0" - -"@ethersproject/wordlists@5.7.0", "@ethersproject/wordlists@^5.7.0": - version "5.7.0" - resolved "https://registry.yarnpkg.com/@ethersproject/wordlists/-/wordlists-5.7.0.tgz#8fb2c07185d68c3e09eb3bfd6e779ba2774627f5" - integrity sha512-S2TFNJNfHWVHNE6cNDjbVlZ6MgE17MIxMbMg2zv3wn+3XSJGosL1m9ZVv3GXCf/2ymSsQ+hRI5IzoMJTG6aoVA== - dependencies: - "@ethersproject/bytes" "^5.7.0" - "@ethersproject/hash" "^5.7.0" - "@ethersproject/logger" "^5.7.0" - "@ethersproject/properties" "^5.7.0" - "@ethersproject/strings" "^5.7.0" - -"@metamask/detect-provider@^2.0.0": - version "2.0.0" - resolved "https://registry.yarnpkg.com/@metamask/detect-provider/-/detect-provider-2.0.0.tgz#4bc2795e5e6f7d8b84b2e845058d2f222c99917d" - integrity sha512-sFpN+TX13E9fdBDh9lvQeZdJn4qYoRb/6QF2oZZK/Pn559IhCFacPMU1rMuqyXoFQF3JSJfii2l98B87QDPeCQ== - -"@types/node@11.11.6": - version "11.11.6" - resolved "https://registry.yarnpkg.com/@types/node/-/node-11.11.6.tgz#df929d1bb2eee5afdda598a41930fe50b43eaa6a" - integrity sha512-Exw4yUWMBXM3X+8oqzJNRqZSwUAaS4+7NdvHqQuFi/d+synz++xmX3QIf+BFqneW8N31R8Ky+sikfZUXq07ggQ== - -aes-js@3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/aes-js/-/aes-js-3.0.0.tgz#e21df10ad6c2053295bcbb8dab40b09dbea87e4d" - integrity sha512-H7wUZRn8WpTq9jocdxQ2c8x2sKo9ZVmzfRE13GiNJXfp7NcKYEdvl3vspKjXox6RIG2VtaRe4JFvxG4rqp2Zuw== - -base-x@^3.0.2: - version "3.0.9" - resolved "https://registry.yarnpkg.com/base-x/-/base-x-3.0.9.tgz#6349aaabb58526332de9f60995e548a53fe21320" - integrity sha512-H7JU6iBHTal1gp56aKoaa//YUxEaAOUiydvrV/pILqIHXTtqxSkATOnDA2u+jZ/61sD+L/412+7kzXRtWukhpQ== - dependencies: - safe-buffer "^5.0.1" - -base-x@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/base-x/-/base-x-4.0.0.tgz#d0e3b7753450c73f8ad2389b5c018a4af7b2224a" - integrity sha512-FuwxlW4H5kh37X/oW59pwTzzTKRzfrrQwhmyspRM7swOEZcHtDZSCt45U6oKgtuFE+WYPblePMVIPR4RZrh/hw== - -bech32@1.1.4: - version "1.1.4" - resolved "https://registry.yarnpkg.com/bech32/-/bech32-1.1.4.tgz#e38c9f37bf179b8eb16ae3a772b40c356d4832e9" - integrity sha512-s0IrSOzLlbvX7yp4WBfPITzpAU8sqQcpsmwXDiKwrG4r491vwCO/XpejasRNl0piBMe/DvP4Tz0mIS/X1DPJBQ== - -bip39-light@^1.0.7: - version "1.0.7" - resolved "https://registry.yarnpkg.com/bip39-light/-/bip39-light-1.0.7.tgz#06a72f251b89389a136d3f177f29b03342adc5ba" - integrity sha512-WDTmLRQUsiioBdTs9BmSEmkJza+8xfJmptsNJjxnoq3EydSa/ZBXT6rm66KoT3PJIRYMnhSKNR7S9YL1l7R40Q== - dependencies: - create-hash "^1.1.0" - pbkdf2 "^3.0.9" - -bip39@3.0.2: - version "3.0.2" - resolved "https://registry.yarnpkg.com/bip39/-/bip39-3.0.2.tgz#2baf42ff3071fc9ddd5103de92e8f80d9257ee32" - integrity sha512-J4E1r2N0tUylTKt07ibXvhpT2c5pyAFgvuA5q1H9uDy6dEGpjV8jmymh3MTYJDLCNbIVClSB9FbND49I6N24MQ== - dependencies: - "@types/node" "11.11.6" - create-hash "^1.1.0" - pbkdf2 "^3.0.9" - randombytes "^2.0.1" - -bn.js@^4.11.9: - version "4.12.0" - resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.12.0.tgz#775b3f278efbb9718eec7361f483fb36fbbfea88" - integrity sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA== - -bn.js@^5.2.1: - version "5.2.1" - resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-5.2.1.tgz#0bc527a6a0d18d0aa8d5b0538ce4a77dccfa7b70" - integrity sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ== - -brorand@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/brorand/-/brorand-1.1.0.tgz#12c25efe40a45e3c323eb8675a0a0ce57b22371f" - integrity sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w== - -bs58@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/bs58/-/bs58-4.0.1.tgz#be161e76c354f6f788ae4071f63f34e8c4f0a42a" - integrity sha512-Ok3Wdf5vOIlBrgCvTq96gBkJw+JUEzdBgyaza5HLtPm7yTHkjRy8+JzNyHF7BHa0bNWOQIp3m5YF0nnFcOIKLw== - dependencies: - base-x "^3.0.2" - -bs58@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/bs58/-/bs58-5.0.0.tgz#865575b4d13c09ea2a84622df6c8cbeb54ffc279" - integrity sha512-r+ihvQJvahgYT50JD05dyJNKlmmSlMoOGwn1lCcEzanPglg7TxYjioQUYehQ9mAR/+hOSd2jRc/Z2y5UxBymvQ== - dependencies: - base-x "^4.0.0" - -cipher-base@^1.0.1, cipher-base@^1.0.3: - version "1.0.4" - resolved "https://registry.yarnpkg.com/cipher-base/-/cipher-base-1.0.4.tgz#8760e4ecc272f4c363532f926d874aae2c1397de" - integrity sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q== - dependencies: - inherits "^2.0.1" - safe-buffer "^5.0.1" - -create-hash@^1.1.0, create-hash@^1.1.2: - version "1.2.0" - resolved "https://registry.yarnpkg.com/create-hash/-/create-hash-1.2.0.tgz#889078af11a63756bcfb59bd221996be3a9ef196" - integrity sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg== - dependencies: - cipher-base "^1.0.1" - inherits "^2.0.1" - md5.js "^1.3.4" - ripemd160 "^2.0.1" - sha.js "^2.4.0" - -create-hmac@1.1.7, create-hmac@^1.1.4: - version "1.1.7" - resolved "https://registry.yarnpkg.com/create-hmac/-/create-hmac-1.1.7.tgz#69170c78b3ab957147b2b8b04572e47ead2243ff" - integrity sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg== - dependencies: - cipher-base "^1.0.3" - create-hash "^1.1.0" - inherits "^2.0.1" - ripemd160 "^2.0.0" - safe-buffer "^5.0.1" - sha.js "^2.4.8" - -elliptic@6.5.4: - version "6.5.4" - resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.4.tgz#da37cebd31e79a1367e941b592ed1fbebd58abbb" - integrity sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ== - dependencies: - bn.js "^4.11.9" - brorand "^1.1.0" - hash.js "^1.0.0" - hmac-drbg "^1.0.1" - inherits "^2.0.4" - minimalistic-assert "^1.0.1" - minimalistic-crypto-utils "^1.0.1" - -ethers@^5.7.2: - version "5.7.2" - resolved "https://registry.yarnpkg.com/ethers/-/ethers-5.7.2.tgz#3a7deeabbb8c030d4126b24f84e525466145872e" - integrity sha512-wswUsmWo1aOK8rR7DIKiWSw9DbLWe6x98Jrn8wcTflTVvaXhAMaB5zGAXy0GYQEQp9iO1iSHWVyARQm11zUtyg== - dependencies: - "@ethersproject/abi" "5.7.0" - "@ethersproject/abstract-provider" "5.7.0" - "@ethersproject/abstract-signer" "5.7.0" - "@ethersproject/address" "5.7.0" - "@ethersproject/base64" "5.7.0" - "@ethersproject/basex" "5.7.0" - "@ethersproject/bignumber" "5.7.0" - "@ethersproject/bytes" "5.7.0" - "@ethersproject/constants" "5.7.0" - "@ethersproject/contracts" "5.7.0" - "@ethersproject/hash" "5.7.0" - "@ethersproject/hdnode" "5.7.0" - "@ethersproject/json-wallets" "5.7.0" - "@ethersproject/keccak256" "5.7.0" - "@ethersproject/logger" "5.7.0" - "@ethersproject/networks" "5.7.1" - "@ethersproject/pbkdf2" "5.7.0" - "@ethersproject/properties" "5.7.0" - "@ethersproject/providers" "5.7.2" - "@ethersproject/random" "5.7.0" - "@ethersproject/rlp" "5.7.0" - "@ethersproject/sha2" "5.7.0" - "@ethersproject/signing-key" "5.7.0" - "@ethersproject/solidity" "5.7.0" - "@ethersproject/strings" "5.7.0" - "@ethersproject/transactions" "5.7.0" - "@ethersproject/units" "5.7.0" - "@ethersproject/wallet" "5.7.0" - "@ethersproject/web" "5.7.1" - "@ethersproject/wordlists" "5.7.0" - -hash-base@^3.0.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/hash-base/-/hash-base-3.1.0.tgz#55c381d9e06e1d2997a883b4a3fddfe7f0d3af33" - integrity sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA== - dependencies: - inherits "^2.0.4" - readable-stream "^3.6.0" - safe-buffer "^5.2.0" - -hash.js@1.1.7, hash.js@^1.0.0, hash.js@^1.0.3: - version "1.1.7" - resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.7.tgz#0babca538e8d4ee4a0f8988d68866537a003cf42" - integrity sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA== - dependencies: - inherits "^2.0.3" - minimalistic-assert "^1.0.1" - -hmac-drbg@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz#d2745701025a6c775a6c545793ed502fc0c649a1" - integrity sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg== - dependencies: - hash.js "^1.0.3" - minimalistic-assert "^1.0.0" - minimalistic-crypto-utils "^1.0.1" - -inherits@^2.0.1, inherits@^2.0.3, inherits@^2.0.4: - version "2.0.4" - resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" - integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== - -js-sha3@0.8.0: - version "0.8.0" - resolved "https://registry.yarnpkg.com/js-sha3/-/js-sha3-0.8.0.tgz#b9b7a5da73afad7dedd0f8c463954cbde6818840" - integrity sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q== - -md5.js@^1.3.4: - version "1.3.5" - resolved "https://registry.yarnpkg.com/md5.js/-/md5.js-1.3.5.tgz#b5d07b8e3216e3e27cd728d72f70d1e6a342005f" - integrity sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg== - dependencies: - hash-base "^3.0.0" - inherits "^2.0.1" - safe-buffer "^5.1.2" - -minimalistic-assert@^1.0.0, minimalistic-assert@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz#2e194de044626d4a10e7f7fbc00ce73e83e4d5c7" - integrity sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A== - -minimalistic-crypto-utils@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz#f6c00c1c0b082246e5c4d99dfb8c7c083b2b582a" - integrity sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg== - -near-hd-key@^1.2.1: - version "1.2.1" - resolved "https://registry.yarnpkg.com/near-hd-key/-/near-hd-key-1.2.1.tgz#f508ff15436cf8a439b543220f3cc72188a46756" - integrity sha512-SIrthcL5Wc0sps+2e1xGj3zceEa68TgNZDLuCx0daxmfTP7sFTB3/mtE2pYhlFsCxWoMn+JfID5E1NlzvvbRJg== - dependencies: - bip39 "3.0.2" - create-hmac "1.1.7" - tweetnacl "1.0.3" - -near-seed-phrase@^0.2.0: - version "0.2.0" - resolved "https://registry.yarnpkg.com/near-seed-phrase/-/near-seed-phrase-0.2.0.tgz#fb7cf89682112b1160ab68abb50dc821f49be18a" - integrity sha512-NpmrnejpY1AdlRpDZ0schJQJtfBaoUheRfiYtQpcq9TkwPgqKZCRULV5L3hHmLc0ep7KRtikbPQ9R2ztN/3cyQ== - dependencies: - bip39-light "^1.0.7" - bs58 "^4.0.1" - near-hd-key "^1.2.1" - tweetnacl "^1.0.2" - -pbkdf2@^3.0.9: - version "3.1.2" - resolved "https://registry.yarnpkg.com/pbkdf2/-/pbkdf2-3.1.2.tgz#dd822aa0887580e52f1a039dc3eda108efae3075" - integrity sha512-iuh7L6jA7JEGu2WxDwtQP1ddOpaJNC4KlDEFfdQajSGgGPNi4OyDc2R7QnbY2bR9QjBVGwgvTdNJZoE7RaxUMA== - dependencies: - create-hash "^1.1.2" - create-hmac "^1.1.4" - ripemd160 "^2.0.1" - safe-buffer "^5.0.1" - sha.js "^2.4.8" - -randombytes@^2.0.1: - version "2.1.0" - resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a" - integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ== - dependencies: - safe-buffer "^5.1.0" - -readable-stream@^3.6.0: - version "3.6.0" - resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.0.tgz#337bbda3adc0706bd3e024426a286d4b4b2c9198" - integrity sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA== - dependencies: - inherits "^2.0.3" - string_decoder "^1.1.1" - util-deprecate "^1.0.1" - -ripemd160@^2.0.0, ripemd160@^2.0.1: - version "2.0.2" - resolved "https://registry.yarnpkg.com/ripemd160/-/ripemd160-2.0.2.tgz#a1c1a6f624751577ba5d07914cbc92850585890c" - integrity sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA== - dependencies: - hash-base "^3.0.0" - inherits "^2.0.1" - -safe-buffer@^5.0.1, safe-buffer@^5.1.0, safe-buffer@^5.1.2, safe-buffer@^5.2.0, safe-buffer@~5.2.0: - version "5.2.1" - resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" - integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== - -scrypt-js@3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/scrypt-js/-/scrypt-js-3.0.1.tgz#d314a57c2aef69d1ad98a138a21fe9eafa9ee312" - integrity sha512-cdwTTnqPu0Hyvf5in5asVdZocVDTNRmR7XEcJuIzMjJeSHybHl7vpB66AzwTaIg6CLSbtjcxc8fqcySfnTkccA== - -sha.js@^2.4.0, sha.js@^2.4.8: - version "2.4.11" - resolved "https://registry.yarnpkg.com/sha.js/-/sha.js-2.4.11.tgz#37a5cf0b81ecbc6943de109ba2960d1b26584ae7" - integrity sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ== - dependencies: - inherits "^2.0.1" - safe-buffer "^5.0.1" - -string_decoder@^1.1.1: - version "1.3.0" - resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e" - integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA== - dependencies: - safe-buffer "~5.2.0" - -tweetnacl@1.0.3, tweetnacl@^1.0.2, tweetnacl@^1.0.3: - version "1.0.3" - resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-1.0.3.tgz#ac0af71680458d8a6378d0d0d050ab1407d35596" - integrity sha512-6rt+RN7aOi1nGMyC4Xa5DdYiukl2UWCbcJft7YhxReBGQD7OAM8Pbxw6YMo4r2diNEA8FEmu32YOn9rhaiE5yw== - -util-deprecate@^1.0.1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" - integrity sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw== - -ws@7.4.6: - version "7.4.6" - resolved "https://registry.yarnpkg.com/ws/-/ws-7.4.6.tgz#5654ca8ecdeee47c33a9a4bf6d28e2be2980377c" - integrity sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==