Skip to content

Commit

Permalink
Mock Adapter with Deterministic Private Key (#130)
Browse files Browse the repository at this point in the history
  • Loading branch information
bh2smith authored Oct 10, 2024
1 parent a8fc45e commit 0c11b97
Show file tree
Hide file tree
Showing 9 changed files with 199 additions and 22 deletions.
1 change: 1 addition & 0 deletions .github/workflows/e2e.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -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}}
21 changes: 18 additions & 3 deletions src/chains/ethereum.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,26 +16,26 @@ import {
FunctionCallTransaction,
TxPayload,
SignArgs,
MpcContract,
Network,
buildTxPayload,
populateTx,
toPayload,
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;
}) {
Expand Down Expand Up @@ -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<NearEthAdapter> {
// 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...),
Expand Down
6 changes: 4 additions & 2 deletions src/constants.ts
Original file line number Diff line number Diff line change
@@ -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;
Expand Down Expand Up @@ -29,7 +30,8 @@ export const CHAIN_INFO: Record<number, ChainInfo> = {
wrappedToken: "0x094616f0bdfb0b526bd735bf66eca0ad254ca81f",
},
100: {
currencyIcon: "",
currencyIcon:
"",
icon: "",
testnet: false,
wrappedToken: "0x6a023ccd1ff6f2045c3309768ead9e68f978f6e1",
Expand Down
14 changes: 13 additions & 1 deletion src/mpcContract.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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;

Expand Down Expand Up @@ -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<Address>;
getDeposit(): Promise<string>;
requestSignature(signArgs: SignArgs, gas?: bigint): Promise<Signature>;
encodeSignatureRequestTx(
signArgs: SignArgs,
gas?: bigint
): Promise<FunctionCallTransaction<{ request: SignArgs }>>;
}
4 changes: 2 additions & 2 deletions src/types.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { MpcContract } from "./mpcContract";
import { IMpcContract } from "./mpcContract";
import {
Address,
Hex,
Expand Down Expand Up @@ -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;
}

Expand Down
2 changes: 2 additions & 0 deletions src/utils/index.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
export * from "./request";
export * from "./signature";
export * from "./transaction";

export { mockAdapter } from "./mock-sign";
106 changes: 106 additions & 0 deletions src/utils/mock-sign.ts
Original file line number Diff line number Diff line change
@@ -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<Address> => {
return this.ethAccount.address;
};

getDeposit = async (): Promise<string> => {
return "1";
};

requestSignature = async (
signArgs: SignArgs,
_gas?: bigint
): Promise<Signature> => {
const hexSignature = await this.ethAccount.sign({
hash: fromPayload(signArgs.payload),
});
return hexToSignature(hexSignature);
};

async encodeSignatureRequestTx(
signArgs: SignArgs,
gas?: bigint
): Promise<FunctionCallTransaction<{ request: SignArgs }>> {
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<NearEthAdapter> {
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 });
}
Loading

0 comments on commit 0c11b97

Please sign in to comment.