From d7821587980532824319d5d6882b90311a1576fc Mon Sep 17 00:00:00 2001 From: rabbyweb3team001 Date: Tue, 20 Aug 2024 16:11:54 +0800 Subject: [PATCH] add rabby wallet --- packages/rabby-wallet/.babelrc | 10 + packages/rabby-wallet/.eslintrc.json | 18 ++ packages/rabby-wallet/README.md | 70 ++++++ packages/rabby-wallet/assets/rabby-wallet.png | Bin 0 -> 21395 bytes packages/rabby-wallet/jest.config.ts | 16 ++ packages/rabby-wallet/package.json | 24 ++ packages/rabby-wallet/project.json | 56 +++++ packages/rabby-wallet/src/index.ts | 2 + packages/rabby-wallet/src/lib/icon.ts | 1 + .../src/lib/injected-rabby-wallet.ts | 73 ++++++ .../rabby-wallet/src/lib/rabby-wallet.spec.ts | 131 ++++++++++ packages/rabby-wallet/src/lib/rabby-wallet.ts | 236 ++++++++++++++++++ packages/rabby-wallet/tsconfig.json | 20 ++ packages/rabby-wallet/tsconfig.lib.json | 10 + packages/rabby-wallet/tsconfig.spec.json | 9 + 15 files changed, 676 insertions(+) create mode 100644 packages/rabby-wallet/.babelrc create mode 100644 packages/rabby-wallet/.eslintrc.json create mode 100644 packages/rabby-wallet/README.md create mode 100644 packages/rabby-wallet/assets/rabby-wallet.png create mode 100644 packages/rabby-wallet/jest.config.ts create mode 100644 packages/rabby-wallet/package.json create mode 100644 packages/rabby-wallet/project.json create mode 100644 packages/rabby-wallet/src/index.ts create mode 100644 packages/rabby-wallet/src/lib/icon.ts create mode 100644 packages/rabby-wallet/src/lib/injected-rabby-wallet.ts create mode 100644 packages/rabby-wallet/src/lib/rabby-wallet.spec.ts create mode 100644 packages/rabby-wallet/src/lib/rabby-wallet.ts create mode 100644 packages/rabby-wallet/tsconfig.json create mode 100644 packages/rabby-wallet/tsconfig.lib.json create mode 100644 packages/rabby-wallet/tsconfig.spec.json diff --git a/packages/rabby-wallet/.babelrc b/packages/rabby-wallet/.babelrc new file mode 100644 index 000000000..b63f0528f --- /dev/null +++ b/packages/rabby-wallet/.babelrc @@ -0,0 +1,10 @@ +{ + "presets": [ + [ + "@nrwl/js/babel", + { + "useBuiltIns": "usage" + } + ] + ] +} diff --git a/packages/rabby-wallet/.eslintrc.json b/packages/rabby-wallet/.eslintrc.json new file mode 100644 index 000000000..9d9c0db55 --- /dev/null +++ b/packages/rabby-wallet/.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/rabby-wallet/README.md b/packages/rabby-wallet/README.md new file mode 100644 index 000000000..c6d2d006a --- /dev/null +++ b/packages/rabby-wallet/README.md @@ -0,0 +1,70 @@ +# @near-wallet-selector/rabby-wallet + +This is the [Rabby Wallet](https://chromewebstore.google.com/detail/rabby-wallet/acmacodkjbdgmoleebolmdjonilkdbch) package for NEAR Wallet Selector. + +## Install Rabby Wallet + +- [Chrome](https://chromewebstore.google.com/detail/rabby-wallet/acmacodkjbdgmoleebolmdjonilkdbch) + + +## Installation and Usage + +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 +yarn add near-api-js + +# Using NPM. +npm install near-api-js +``` +```bash +# Using Yarn +yarn add @near-wallet-selector/rabby-wallet + +# Using NPM. +npm install @near-wallet-selector/rabby-wallet +``` + +Then use it in your dApp: + +```ts +import { setupWalletSelector } from "@near-wallet-selector/core"; +import { setupRabbyWallet } from "@near-wallet-selector/rabby-wallet"; + +// Rabby Wallet for Wallet Selector can be setup without any params or it can take few optional params, see options below. +const rabbyWallet = setupRabbyWallet({ + iconUrl: "https://" // optional +}); + +const selector = await setupWalletSelector({ + network: "mainnet", // Rabby Wallet only supports NEAR Mainnet at the moment. + modules: [rabbyWallet], +}); +``` + +## Options + +- `iconUrl`: (`string?`): Image URL for the icon shown in the modal. This can also be a relative path or base64 encoded image. Defaults is `./assets/rabby-wallet.png`. +- `deprecated`: (`boolean?`): Deprecated is optional. Default is `false`. + +## Assets + +Assets such as icons can be found in the `/assets` directory of the package. Below is an example using Webpack: + +```ts +import { setupRabbyWallet } from "@near-wallet-selector/rabby-wallet"; +import rabbyIconUrl from "@near-wallet-selector/rabby-wallet/assets/rabby-wallet.png"; + +const rabbyWallet = setupRabbyWallet({ + iconUrl: rabbyIconUrl +}); +``` + +## Tip + +Rabby Wallet only supports NEAR Mainnet at the moment. + +## License + +This repository is distributed under the terms of both the MIT license and the Apache License (Version 2.0). diff --git a/packages/rabby-wallet/assets/rabby-wallet.png b/packages/rabby-wallet/assets/rabby-wallet.png new file mode 100644 index 0000000000000000000000000000000000000000..1a8514fb85ae4fe4b1ad5c062208011d05337bd0 GIT binary patch literal 21395 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/tsconfig.spec.json", + }, + ], + }, + moduleFileExtensions: ["ts", "tsx", "js", "jsx"], + coverageDirectory: "../../coverage/packages/rabby-wallet", +}; diff --git a/packages/rabby-wallet/package.json b/packages/rabby-wallet/package.json new file mode 100644 index 000000000..218f160f6 --- /dev/null +++ b/packages/rabby-wallet/package.json @@ -0,0 +1,24 @@ +{ + "name": "@near-wallet-selector/rabby-wallet", + "version": "8.9.12", + "description": "rabby Wallet package for NEAR Wallet Selector.", + "keywords": [ + "near", + "blockchain", + "wallets", + "dapps", + "near-protocol", + "near-blockchain", + "wallet selector", + "injected wallet", + "Rabby wallet" + ], + "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/rabby-wallet" +} diff --git a/packages/rabby-wallet/project.json b/packages/rabby-wallet/project.json new file mode 100644 index 000000000..d63161f70 --- /dev/null +++ b/packages/rabby-wallet/project.json @@ -0,0 +1,56 @@ +{ + "name": "rabby-wallet", + "$schema": "../../node_modules/nx/schemas/project-schema.json", + "sourceRoot": "packages/rabby-wallet/src", + "projectType": "library", + "targets": { + "build": { + "executor": "@nrwl/rollup:rollup", + "outputs": ["{options.outputPath}"], + "options": { + "outputPath": "dist/packages/rabby-wallet", + "tsConfig": "packages/rabby-wallet/tsconfig.lib.json", + "project": "packages/rabby-wallet/package.json", + "entryFile": "packages/rabby-wallet/src/index.ts", + "buildableProjectDepsInPackageJsonType": "dependencies", + "compiler": "babel", + "format": ["esm", "cjs"], + "assets": [ + { + "glob": "packages/rabby-wallet/README.md", + "input": ".", + "output": "." + }, + { + "glob": "packages/rabby-wallet/assets/*", + "input": ".", + "output": "assets" + } + ] + } + }, + "lint": { + "executor": "@nx/linter:eslint", + "outputs": ["{options.outputFile}"], + "options": { + "lintFilePatterns": ["packages/rabby-wallet/**/*.ts"] + } + }, + "test": { + "executor": "@nx/jest:jest", + "outputs": ["{workspaceRoot}/coverage/packages/rabby-wallet"], + "options": { + "jestConfig": "packages/rabby-wallet/jest.config.ts", + "passWithNoTests": true + } + }, + "deploy": { + "executor": "ngx-deploy-npm:deploy", + "options": { + "access": "public" + }, + "dependsOn": ["^deploy"] + } + }, + "tags": ["injected-wallet"] +} diff --git a/packages/rabby-wallet/src/index.ts b/packages/rabby-wallet/src/index.ts new file mode 100644 index 000000000..da9a935b4 --- /dev/null +++ b/packages/rabby-wallet/src/index.ts @@ -0,0 +1,2 @@ +export { setupRabbyWallet } from "./lib/rabby-wallet"; +export type { RabbyWalletParams } from "./lib/rabby-wallet"; diff --git a/packages/rabby-wallet/src/lib/icon.ts b/packages/rabby-wallet/src/lib/icon.ts new file mode 100644 index 000000000..5f42ac156 --- /dev/null +++ b/packages/rabby-wallet/src/lib/icon.ts @@ -0,0 +1 @@ +export default ``; diff --git a/packages/rabby-wallet/src/lib/injected-rabby-wallet.ts b/packages/rabby-wallet/src/lib/injected-rabby-wallet.ts new file mode 100644 index 000000000..35b7270d2 --- /dev/null +++ b/packages/rabby-wallet/src/lib/injected-rabby-wallet.ts @@ -0,0 +1,73 @@ +export interface AccessKey { + publicKey: string; + secretKey: string; +} + +interface RequestSignInResponse { + accessKey: AccessKey; + accountId: string; + error: string | { type: string }; +} + +export type SignOutResponse = true | { error: string | { type: string } }; + +interface RequestSignInParams { + contractId?: string; + methodNames?: Array; +} + +interface Action { + methodName: string; + args: object; + gas: string; + deposit: string; +} + +interface SignAndSendTransactionParams { + receiverId: string; + actions: Array; +} + +export interface SignAndSendTransactionResponse { + method: string; + txHash: string; + code: number; +} + +interface Transaction { + receiverId: string; + actions: Array; +} + +interface RequestSignTransactionsParams { + transactions: Array; +} + +interface RequestSignTransactionsResponse { + txs: Array<{ + signedTx: string; + }>; +} + +export interface RabbyEvents { + signIn: () => void; + signOut: () => void; + accountChanged: (changedAccountId: string) => void; +} + +export interface InjectedRabby { + getAccountId: () => string | null; + requestSignIn: ( + params: RequestSignInParams + ) => Promise; + signOut: () => Promise; + isSignedIn: () => boolean; + on: ( + event: Event, + callback: RabbyEvents[Event] + ) => void; + signTransaction: (params: SignAndSendTransactionParams) => Promise; + requestSignTransactions: ( + params: RequestSignTransactionsParams + ) => Promise; +} diff --git a/packages/rabby-wallet/src/lib/rabby-wallet.spec.ts b/packages/rabby-wallet/src/lib/rabby-wallet.spec.ts new file mode 100644 index 000000000..03736efb1 --- /dev/null +++ b/packages/rabby-wallet/src/lib/rabby-wallet.spec.ts @@ -0,0 +1,131 @@ +/* eslint-disable @nx/enforce-module-boundaries */ +import { mock } from "jest-mock-extended"; +import type { ProviderService } from "packages/core/src/lib/services"; +import type { FinalExecutionOutcome } from "near-api-js/lib/providers"; + +import { setupRabbyWallet } from "./rabby-wallet"; +import { mockWallet } from "../../../core/src/lib/testUtils"; +import type { AccessKey, SignOutResponse } from "./injected-rabby-wallet"; +import type { InjectedWallet } from "../../../core/src/lib/wallet"; + +const accountId = "test-account.testnet"; +const transactions = [ + { + signerId: accountId, + receiverId: "test.testnet", + actions: [], + }, +]; + +jest.mock("near-api-js/lib/transaction", () => { + return { + ...jest.requireActual("near-api-js/lib/transaction"), + SignedTransaction: { + ...jest.requireActual("near-api-js/lib/transaction").SignedTransaction, + decode: jest.fn().mockReturnValue({}), + }, + }; +}); + +const mockRabby = () => { + const rabbyWallet = { + near: { + getAccountId: jest.fn().mockReturnValue(""), + requestSignIn: jest.fn(async () => { + window.rabby.near!.getAccountId = jest.fn().mockReturnValue(accountId); + + return { + accountId, + accessKey: mock(), + error: "", + }; + }), + signOut: jest.fn().mockReturnValue(mock()), + isSignedIn: jest.fn().mockReturnValue(true), + on: jest.fn(), + signTransaction: jest.fn().mockResolvedValue("signedTx"), + requestSignTransactions: jest.fn().mockResolvedValue({ + txs: [{ signedTx: "signedTx" }], + }), + }, + }; + + window.rabby = rabbyWallet; + + return window.rabby.near; +}; + +const createRabbyWallet = async () => { + const injectedRabby = mockRabby(); + const provider = mock(); + provider.sendTransaction.mockResolvedValue(mock()); + const { wallet } = await mockWallet(setupRabbyWallet(), { + provider, + }); + + return { + wallet, + injectedRabby: injectedRabby, + }; +}; + +afterEach(() => { + jest.resetModules(); +}); + +describe("signIn", () => { + it("sign into rabby wallet", async () => { + const { wallet, injectedRabby } = await createRabbyWallet(); + const accounts = await wallet.signIn({ contractId: "test.testnet" }); + expect(injectedRabby!.requestSignIn).toHaveBeenCalled(); + expect(accounts[0]?.accountId).toBe(accountId); + }); +}); + +describe("signOut", () => { + it("sign out of rabby wallet", async () => { + const { wallet, injectedRabby } = await createRabbyWallet(); + await wallet.signIn({ contractId: "test.testnet" }); + await wallet.signOut(); + + expect(injectedRabby!.signOut).toHaveBeenCalled(); + }); +}); + +describe("getAccounts", () => { + it("returns accounts", async () => { + const { wallet, injectedRabby } = await createRabbyWallet(); + await wallet.signIn({ contractId: "test.testnet" }); + const result = await wallet.getAccounts(); + + expect(injectedRabby!.getAccountId).toHaveBeenCalled(); + expect(result).toEqual([{ accountId, publicKey: undefined }]); + }); +}); + +describe("signTransaction", () => { + it("sign transaction in rabby wallet", async () => { + const { wallet, injectedRabby } = await createRabbyWallet(); + await wallet.signIn({ contractId: "test.testnet" }); + await wallet.signAndSendTransaction({ + signerId: accountId, + receiverId: "test.testnet", + actions: [], + }); + + expect(injectedRabby!.signTransaction).toHaveBeenCalled(); + }); +}); + +describe("signAndSendTransactions", () => { + it("sign transactions in rabby wallet", async () => { + const { wallet, injectedRabby } = await createRabbyWallet(); + await wallet.signIn({ contractId: "test.testnet" }); + const result = await wallet.signAndSendTransactions({ + transactions, + }); + + expect(injectedRabby!.requestSignTransactions).toHaveBeenCalled(); + expect(result.length).toEqual(transactions.length); + }); +}); diff --git a/packages/rabby-wallet/src/lib/rabby-wallet.ts b/packages/rabby-wallet/src/lib/rabby-wallet.ts new file mode 100644 index 000000000..acbec1b37 --- /dev/null +++ b/packages/rabby-wallet/src/lib/rabby-wallet.ts @@ -0,0 +1,236 @@ +import { isMobile } from "is-mobile"; +import { SignedTransaction } from "near-api-js/lib/transaction"; +import type { + Action, + FunctionCallAction, + InjectedWallet, + Optional, + Transaction, + WalletBehaviourFactory, + WalletModuleFactory, +} from "@near-wallet-selector/core"; +import { waitFor } from "@near-wallet-selector/core"; + +import icon from "./icon"; +import type { InjectedRabby } from "./injected-rabby-wallet"; + +const downloadUrl = + "https://chromewebstore.google.com/detail/rabby-wallet/acmacodkjbdgmoleebolmdjonilkdbch"; + +declare global { + interface Window { + rabby: { + near: InjectedRabby | undefined; + }; + } +} + +export interface RabbyWalletParams { + iconUrl?: string; + deprecated?: boolean; +} + +interface RabbyState { + wallet: InjectedRabby; +} + +const isInstalled = () => { + return waitFor(() => !!window.rabby?.near).catch(() => false); +}; + +const setupRabbyState = (): RabbyState => { + const wallet = window.rabby?.near!; + return { + wallet, + }; +}; + +const RabbyWallet: WalletBehaviourFactory = async ({ + metadata, + store, + emitter, + logger, + id, + provider, +}) => { + const _state = setupRabbyState(); + + const signOut = async () => { + if (!_state.wallet.isSignedIn()) { + return; + } + + await _state.wallet.signOut(); + }; + + const setupEvents = () => { + _state.wallet.on("accountChanged", async (newAccountId) => { + logger.log("onAccountChange", newAccountId); + emitter.emit("signedOut", null); + }); + }; + + 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 transformTransactions = ( + transactions: Array> + ) => { + return transactions.map((transaction) => { + return { + receiverId: transaction.receiverId, + actions: transformActions(transaction.actions), + }; + }); + }; + + if (_state.wallet && _state.wallet.isSignedIn()) { + setupEvents(); + } + + const getAccounts = async () => { + const { selectedWalletId } = store.getState(); + if (selectedWalletId === id) { + await waitFor(() => !!_state.wallet?.isSignedIn(), { + timeout: 1000, + }).catch(); + } + + const accountId = _state.wallet.getAccountId(); + + if (!accountId) { + return []; + } + + return [{ accountId }]; + }; + + const getSignedTransaction = (signedTx: string) => { + const buf = Buffer.from(signedTx, "base64"); + return SignedTransaction.decode(buf); + }; + + return { + async signIn({ contractId, methodNames }) { + try { + const { accessKey, accountId } = await _state.wallet.requestSignIn({ + contractId, + methodNames, + }); + setupEvents(); + + const publicKey = accessKey?.publicKey; + + return [ + { + accountId, + publicKey: publicKey ? publicKey.toString() : undefined, + }, + ]; + } catch (_) { + await signOut(); + throw new Error("Failed to sign in"); + } + }, + + signOut, + + getAccounts, + + async verifyOwner() { + throw new Error(`Method not supported by ${metadata.name}`); + }, + + async signMessage() { + throw new Error(`Method not supported by ${metadata.name}`); + }, + + async signAndSendTransaction({ signerId, receiverId, actions }) { + logger.log("signAndSendTransaction", { signerId, receiverId, actions }); + + const { contract } = store.getState(); + + if (!_state.wallet.isSignedIn() || !contract) { + throw new Error("Wallet not signed in"); + } + + try { + const signedTx = await _state.wallet.signTransaction({ + receiverId: receiverId || contract.contractId, + actions: transformActions(actions), + }); + const signedTransaction = getSignedTransaction(signedTx); + return provider.sendTransaction(signedTransaction); + } catch (error) { + throw new Error("sign Error"); + } + }, + + async signAndSendTransactions({ transactions }) { + logger.log("signAndSendTransactions", { transactions }); + + if (!_state.wallet.isSignedIn()) { + throw new Error("Wallet not signed in"); + } + + try { + const resp = await _state.wallet.requestSignTransactions({ + transactions: transformTransactions(transactions), + }); + + const { txs } = resp; + const results = []; + for (let i = 0; i < txs.length; i++) { + const signedTransaction = getSignedTransaction(txs[i].signedTx); + results.push(await provider.sendTransaction(signedTransaction)); + } + return results; + } catch (error) { + throw new Error("sign Error"); + } + }, + }; +}; + +export function setupRabbyWallet({ + iconUrl = icon, + deprecated = false, +}: RabbyWalletParams = {}): WalletModuleFactory { + return async () => { + const mobile = isMobile(); + if (mobile) { + return null; + } + + const installed = await isInstalled(); + + return { + id: "rabby-wallet", + type: "injected", + metadata: { + name: "Rabby Wallet", + description: "Browser extension wallet", + iconUrl, + downloadUrl, + deprecated, + available: installed, + }, + init: RabbyWallet, + }; + }; +} diff --git a/packages/rabby-wallet/tsconfig.json b/packages/rabby-wallet/tsconfig.json new file mode 100644 index 000000000..3aaed7519 --- /dev/null +++ b/packages/rabby-wallet/tsconfig.json @@ -0,0 +1,20 @@ +{ + "extends": "../../tsconfig.base.json", + "compilerOptions": { + "forceConsistentCasingInFileNames": true, + "strict": true, + "noImplicitReturns": true, + "noFallthroughCasesInSwitch": true, + "resolveJsonModule": true + }, + "files": [], + "include": [], + "references": [ + { + "path": "./tsconfig.lib.json" + }, + { + "path": "./tsconfig.spec.json" + } + ] +} diff --git a/packages/rabby-wallet/tsconfig.lib.json b/packages/rabby-wallet/tsconfig.lib.json new file mode 100644 index 000000000..e85ef50f6 --- /dev/null +++ b/packages/rabby-wallet/tsconfig.lib.json @@ -0,0 +1,10 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "../../dist/out-tsc", + "declaration": true, + "types": [] + }, + "include": ["**/*.ts"], + "exclude": ["jest.config.ts", "**/*.spec.ts", "**/*.test.ts"] +} diff --git a/packages/rabby-wallet/tsconfig.spec.json b/packages/rabby-wallet/tsconfig.spec.json new file mode 100644 index 000000000..b506d384e --- /dev/null +++ b/packages/rabby-wallet/tsconfig.spec.json @@ -0,0 +1,9 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "../../dist/out-tsc", + "module": "commonjs", + "types": ["jest", "node"] + }, + "include": ["jest.config.ts", "**/*.spec.ts"] +}