From 0c11b97fd103ed5fa2f16deaa785997691fd4829 Mon Sep 17 00:00:00 2001 From: Benjamin Smith Date: Thu, 10 Oct 2024 11:58:37 +0200 Subject: [PATCH] Mock Adapter with Deterministic Private Key (#130) --- .github/workflows/e2e.yaml | 1 + src/chains/ethereum.ts | 21 +++++- src/constants.ts | 6 +- src/mpcContract.ts | 14 +++- src/types.ts | 4 +- src/utils/index.ts | 2 + src/utils/mock-sign.ts | 106 +++++++++++++++++++++++++++++ tests/e2e.test.ts | 45 ++++++++---- tests/unit/utils.mock-sign.test.ts | 22 ++++++ 9 files changed, 199 insertions(+), 22 deletions(-) create mode 100644 src/utils/mock-sign.ts create mode 100644 tests/unit/utils.mock-sign.test.ts diff --git a/.github/workflows/e2e.yaml b/.github/workflows/e2e.yaml index 22cacbb..92270a0 100644 --- a/.github/workflows/e2e.yaml +++ b/.github/workflows/e2e.yaml @@ -21,3 +21,4 @@ jobs: MPC_CONTRACT_ID: v1.signer-prod.testnet NEAR_ACCOUNT_ID: ${{secrets.NEAR_ACCOUNT_ID}} NEAR_ACCOUNT_PRIVATE_KEY: ${{secrets.NEAR_PK}} + ETH_PK: ${{secrets.ETH_PK}} diff --git a/src/chains/ethereum.ts b/src/chains/ethereum.ts index 84de6de..e1d969c 100644 --- a/src/chains/ethereum.ts +++ b/src/chains/ethereum.ts @@ -16,7 +16,6 @@ import { FunctionCallTransaction, TxPayload, SignArgs, - MpcContract, Network, buildTxPayload, populateTx, @@ -24,18 +23,19 @@ import { broadcastSignedTransaction, SignRequestData, NearEthTxData, + IMpcContract, } from ".."; import { Beta } from "../beta"; import { requestRouter } from "../utils/request"; export class NearEthAdapter { - readonly mpcContract: MpcContract; + readonly mpcContract: IMpcContract; readonly address: Address; readonly derivationPath: string; readonly beta: Beta; private constructor(config: { - mpcContract: MpcContract; + mpcContract: IMpcContract; derivationPath: string; sender: Address; }) { @@ -78,6 +78,21 @@ export class NearEthAdapter { }); } + /** + * Constructs an EVM instance with the provided configuration. + * @param {AdapterParams} args - The configuration object for the Adapter instance. + */ + static async mocked(args: AdapterParams): Promise { + // Sender is uniquely determined by the derivation path! + const mpcContract = args.mpcContract; + const derivationPath = args.derivationPath || "ethereum,1"; + return new NearEthAdapter({ + sender: await mpcContract.deriveEthAddress(derivationPath), + derivationPath, + mpcContract, + }); + } + /** * Takes a minimally declared Ethereum Transaction, * builds the full transaction payload (with gas estimates, prices etc...), diff --git a/src/constants.ts b/src/constants.ts index 1639615..8e77156 100644 --- a/src/constants.ts +++ b/src/constants.ts @@ -1,5 +1,6 @@ /// A short list of networks with known wrapped tokens. -const ETHER_ICON = ""; +const ETHER_ICON = + ""; interface ChainInfo { currencyIcon?: string; icon?: string; @@ -29,7 +30,8 @@ export const CHAIN_INFO: Record = { wrappedToken: "0x094616f0bdfb0b526bd735bf66eca0ad254ca81f", }, 100: { - currencyIcon: "", + currencyIcon: + "", icon: "", testnet: false, wrappedToken: "0x6a023ccd1ff6f2045c3309768ead9e68f978f6e1", diff --git a/src/mpcContract.ts b/src/mpcContract.ts index 7f020a5..b264c54 100644 --- a/src/mpcContract.ts +++ b/src/mpcContract.ts @@ -43,7 +43,7 @@ interface MpcContractInterface extends Contract { * High-level interface for the Near MPC-Recovery Contract * located in: https://github.com/near/mpc-recovery */ -export class MpcContract { +export class MpcContract implements IMpcContract { contract: MpcContractInterface; connectedAccount: Account; @@ -142,3 +142,15 @@ function gasOrDefault(gas?: bigint): string { // Default of 250 TGAS return (TGAS * 250n).toString(); } + +export interface IMpcContract { + connectedAccount: Account; + accountId(): string; + deriveEthAddress(derivationPath: string): Promise
; + getDeposit(): Promise; + requestSignature(signArgs: SignArgs, gas?: bigint): Promise; + encodeSignatureRequestTx( + signArgs: SignArgs, + gas?: bigint + ): Promise>; +} diff --git a/src/types.ts b/src/types.ts index dedcf6c..036eb2a 100644 --- a/src/types.ts +++ b/src/types.ts @@ -1,4 +1,4 @@ -import { MpcContract } from "./mpcContract"; +import { IMpcContract } from "./mpcContract"; import { Address, Hex, @@ -48,7 +48,7 @@ export interface BaseTx { * @property {string} [derivationPath] - Path used to generate ETH account from NEAR account (e.g., "ethereum,1"). */ export interface AdapterParams { - mpcContract: MpcContract; + mpcContract: IMpcContract; derivationPath?: string; } diff --git a/src/utils/index.ts b/src/utils/index.ts index e756f28..56df95a 100644 --- a/src/utils/index.ts +++ b/src/utils/index.ts @@ -1,3 +1,5 @@ export * from "./request"; export * from "./signature"; export * from "./transaction"; + +export { mockAdapter } from "./mock-sign"; diff --git a/src/utils/mock-sign.ts b/src/utils/mock-sign.ts new file mode 100644 index 0000000..0c80fb2 --- /dev/null +++ b/src/utils/mock-sign.ts @@ -0,0 +1,106 @@ +import { Address, Hex, Signature, toHex } from "viem"; +import { PrivateKeyAccount, privateKeyToAccount } from "viem/accounts"; +import { FunctionCallTransaction, SignArgs } from "../types"; +import { Account } from "near-api-js"; +import { IMpcContract, nearAccountFromAccountId, NearEthAdapter } from ".."; + +function fromPayload(payload: number[]): Hex { + if (payload.length !== 32) { + throw new Error(`Payload must have 32 bytes: ${payload}`); + } + // Convert number[] back to Uint8Array + return toHex(new Uint8Array(payload)); +} + +/** + * Converts a raw hexadecimal signature into a structured Signature object + * @param hexSignature The raw hexadecimal signature (e.g., '0x...') + * @returns A structured Signature object with fields r, s, v, and yParity + */ +function hexToSignature(hexSignature: Hex): Signature { + // Strip "0x" prefix if it exists + const cleanedHex = hexSignature.slice(2); + + // Ensure the signature is 65 bytes (130 hex characters) + if (cleanedHex.length !== 130) { + throw new Error( + `Invalid hex signature length: ${cleanedHex.length}. Expected 130 characters (65 bytes).` + ); + } + + // Extract the r, s, and v components from the hex signature + const v = BigInt(`0x${cleanedHex.slice(128, 130)}`); // Last byte (2 hex characters) + return { + r: `0x${cleanedHex.slice(0, 64)}`, // First 32 bytes (64 hex characters) + s: `0x${cleanedHex.slice(64, 128)}`, // Next 32 bytes (64 hex characters), + v, + // Determine yParity based on v (27 or 28 maps to 0 or 1) + yParity: v === 27n ? 0 : v === 28n ? 1 : undefined, + }; +} + +export class MockMpcContract implements IMpcContract { + connectedAccount: Account; + private ethAccount: PrivateKeyAccount; + + constructor(account: Account, privateKey?: Hex) { + this.connectedAccount = account; + this.ethAccount = privateKeyToAccount( + privateKey || + // Known key from deterministic ganache client + "0x4f3edf983ac636a65a842ce7c78d9aa706d3b113bce9c46f30d7d21715b23b1d" + ); + } + + accountId(): string { + return "mock-mpc.offline"; + } + + deriveEthAddress = async (_unused?: string): Promise
=> { + return this.ethAccount.address; + }; + + getDeposit = async (): Promise => { + return "1"; + }; + + requestSignature = async ( + signArgs: SignArgs, + _gas?: bigint + ): Promise => { + const hexSignature = await this.ethAccount.sign({ + hash: fromPayload(signArgs.payload), + }); + return hexToSignature(hexSignature); + }; + + async encodeSignatureRequestTx( + signArgs: SignArgs, + gas?: bigint + ): Promise> { + return { + signerId: this.connectedAccount.accountId, + receiverId: this.accountId(), + actions: [ + { + type: "FunctionCall", + params: { + methodName: "sign", + args: { request: signArgs }, + gas: gas ? gas.toString() : "1", + deposit: await this.getDeposit(), + }, + }, + ], + }; + } +} + +export async function mockAdapter(privateKey?: Hex): Promise { + const account = await nearAccountFromAccountId("mock-user.offline", { + networkId: "testnet", + nodeUrl: "https://rpc.testnet.near.org", + }); + const mpcContract = new MockMpcContract(account, privateKey); + return NearEthAdapter.fromConfig({ mpcContract }); +} diff --git a/tests/e2e.test.ts b/tests/e2e.test.ts index 2af65dd..114fd44 100644 --- a/tests/e2e.test.ts +++ b/tests/e2e.test.ts @@ -1,6 +1,9 @@ -import { SEPOLIA_CHAIN_ID, setupNearEthAdapter } from "../examples/setup"; -import { NearEthAdapter, Network } from "../src"; +import { SEPOLIA_CHAIN_ID } from "../examples/setup"; +import { mockAdapter, NearEthAdapter, Network } from "../src"; import { getBalance } from "viem/actions"; +import dotenv from "dotenv"; +import { recoverTypedDataAddress, recoverMessageAddress } from "viem"; +dotenv.config(); describe("End To End", () => { let adapter: NearEthAdapter; @@ -9,7 +12,10 @@ describe("End To End", () => { const chainId = SEPOLIA_CHAIN_ID; beforeAll(async () => { - adapter = await setupNearEthAdapter(); + // This is a real adapter. + // adapter = await setupNearEthAdapter(); + // This is a Mock Adapter! + adapter = await mockAdapter(process.env.ETH_PK! as `0x${string}`); }); afterAll(async () => { @@ -20,7 +26,7 @@ describe("End To End", () => { await expect(adapter.getBalance(chainId)).resolves.not.toThrow(); }); - it.only("signAndSendTransaction", async () => { + it.skip("signAndSendTransaction", async () => { await expect( adapter.signAndSendTransaction({ // Sending 1 WEI to self (so we never run out of funds) @@ -34,7 +40,7 @@ describe("End To End", () => { it.skip("signAndSendTransaction - Gnosis Chain", async () => { await expect( adapter.signAndSendTransaction({ - // Sending 1 WEI to self (so we never run out of funds) + // Sending 1 WEI to self (so we ~never run out of funds) to: adapter.address, value: ONE_WEI, // Gnosis Chain! @@ -58,7 +64,13 @@ describe("End To End", () => { }); it("signMessage", async () => { - await expect(adapter.signMessage("NearEth")).resolves.not.toThrow(); + const message = "NearEth"; + const signature = await adapter.signMessage(message); + const recoveredAddress = await recoverMessageAddress({ + message, + signature, + }); + expect(recoveredAddress).toBe(adapter.address); }); it("signTypedData", async () => { @@ -93,13 +105,18 @@ describe("End To End", () => { ], } as const; - await expect( - adapter.signTypedData({ - types, - primaryType: "Mail", - message, - domain, - }) - ).resolves.not.toThrow(); + const typedData = { + types, + primaryType: "Mail", + message, + domain, + } as const; + const signature = await adapter.signTypedData(typedData); + + const recoveredAddress = await recoverTypedDataAddress({ + ...typedData, + signature, + }); + expect(recoveredAddress).toBe(adapter.address); }); }); diff --git a/tests/unit/utils.mock-sign.test.ts b/tests/unit/utils.mock-sign.test.ts new file mode 100644 index 0000000..59868ba --- /dev/null +++ b/tests/unit/utils.mock-sign.test.ts @@ -0,0 +1,22 @@ +import { recoverMessageAddress } from "viem"; +import { mockAdapter } from "../../src/utils/mock-sign"; + +describe("Mock Signing", () => { + it("MockAdapter", async () => { + const adapter = await mockAdapter(); + expect(adapter.address).toBe("0x90F8bf6A479f320ead074411a4B0e7944Ea8c9C1"); + + const message = "Hello Joe!"; + const signature = await adapter.signMessage(message); + + expect(signature).toBe( + "0xcadb9d3ade67e815c11646eea6cd52abb7f860af612cd914a2c64c01908af870246926e8d5774d0b63efc46cd9f77cc650a7ad923df69a5151805422ab1d625a1c" + ); + // Recover Address: + const recoveredAddress = await recoverMessageAddress({ + message, + signature, + }); + expect(recoveredAddress).toBe(adapter.address); + }); +});