@cubist-labs/cubesigner-sdk-ethers-v6 0.3.13 → 0.3.23

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.d.ts CHANGED
@@ -1,19 +1,9 @@
1
1
  import { TypedDataDomain, TypedDataField, ethers } from "ethers";
2
- import { KeyInfo, EvmSignRequest, MfaRequestInfo, SignerSession } from "@cubist-labs/cubesigner-sdk";
2
+ import { EvmSignerOptions, KeyInfo, EvmSignRequest, MfaRequestInfo, SignerSession } from "@cubist-labs/cubesigner-sdk";
3
3
  /** Options for the signer */
4
- export interface SignerOptions {
4
+ export interface SignerOptions extends EvmSignerOptions {
5
5
  /** Optional provider to use */
6
6
  provider?: null | ethers.Provider;
7
- /**
8
- * The function to call when MFA information is retrieved. If this callback
9
- * throws, no transaction is broadcast.
10
- */
11
- onMfaPoll?: (arg0: MfaRequestInfo) => void;
12
- /**
13
- * The amount of time (in milliseconds) to wait between checks for MFA
14
- * updates. Default is 1000ms
15
- */
16
- mfaPollIntervalMs?: number;
17
7
  }
18
8
  /**
19
9
  * A ethers.js Signer using CubeSigner
@@ -27,7 +17,7 @@ export declare class Signer extends ethers.AbstractSigner {
27
17
  * @param {SignerOptions} options The options to use for the Signer instance
28
18
  */
29
19
  constructor(address: KeyInfo | string, signerSession: SignerSession, options?: SignerOptions);
30
- /** Resolves to the signer (potentially _NOT_ checksummed) address. */
20
+ /** Resolves to the checksummed signer address. */
31
21
  getAddress(): Promise<string>;
32
22
  /**
33
23
  * Returns the signer connected to %%provider%%.
@@ -67,8 +57,6 @@ export declare class Signer extends ethers.AbstractSigner {
67
57
  * @return {Promise<string>} The signature.
68
58
  */
69
59
  signTypedData(domain: TypedDataDomain, types: Record<string, Array<TypedDataField>>, value: Record<string, any>): Promise<string>;
70
- /** @return {KeyInfo} The key corresponding to this address */
71
- private key;
72
60
  /**
73
61
  * Initialize the signing a message using MFA approvals. This method populates
74
62
  * missing fields. If the signing does not require MFA, this method throws.
package/dist/index.js CHANGED
@@ -10,7 +10,7 @@ var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (
10
10
  if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
11
11
  return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
12
12
  };
13
- var _Signer_instances, _Signer_address, _Signer_key, _Signer_signerSession, _Signer_onMfaPoll, _Signer_mfaPollIntervalMs, _Signer_handleMfa;
13
+ var _Signer_signer;
14
14
  Object.defineProperty(exports, "__esModule", { value: true });
15
15
  exports.Signer = void 0;
16
16
  const ethers_1 = require("ethers");
@@ -27,34 +27,13 @@ class Signer extends ethers_1.ethers.AbstractSigner {
27
27
  */
28
28
  constructor(address, signerSession, options) {
29
29
  super(options?.provider);
30
- _Signer_instances.add(this);
31
- /** The address of the account */
32
- _Signer_address.set(this, void 0);
33
- /** The key to use for signing */
34
- _Signer_key.set(this, void 0);
35
- /** The underlying session */
36
- _Signer_signerSession.set(this, void 0);
37
- /**
38
- * The function to call when MFA information is retrieved. If this callback
39
- * throws, no transaction is broadcast.
40
- */
41
- _Signer_onMfaPoll.set(this, void 0);
42
- /** The amount of time to wait between checks for MFA updates */
43
- _Signer_mfaPollIntervalMs.set(this, void 0);
44
- if (typeof address === "string") {
45
- __classPrivateFieldSet(this, _Signer_address, address, "f");
46
- }
47
- else {
48
- __classPrivateFieldSet(this, _Signer_address, address.materialId, "f");
49
- __classPrivateFieldSet(this, _Signer_key, address, "f");
50
- }
51
- __classPrivateFieldSet(this, _Signer_signerSession, signerSession, "f");
52
- __classPrivateFieldSet(this, _Signer_onMfaPoll, options?.onMfaPoll ?? (( /* _mfaInfo: MfaRequestInfo */) => { }), "f"); // eslint-disable-line @typescript-eslint/no-empty-function
53
- __classPrivateFieldSet(this, _Signer_mfaPollIntervalMs, options?.mfaPollIntervalMs ?? 1000, "f");
30
+ /** The CubeSigner-backed ethers signer */
31
+ _Signer_signer.set(this, void 0);
32
+ __classPrivateFieldSet(this, _Signer_signer, new cubesigner_sdk_1.EvmSigner(address, signerSession, options), "f");
54
33
  }
55
- /** Resolves to the signer (potentially _NOT_ checksummed) address. */
34
+ /** Resolves to the checksummed signer address. */
56
35
  async getAddress() {
57
- return __classPrivateFieldGet(this, _Signer_address, "f");
36
+ return ethers_1.ethers.getAddress(__classPrivateFieldGet(this, _Signer_signer, "f").address);
58
37
  }
59
38
  /**
60
39
  * Returns the signer connected to %%provider%%.
@@ -62,7 +41,10 @@ class Signer extends ethers_1.ethers.AbstractSigner {
62
41
  * @return {Signer} The signer connected to signer.
63
42
  */
64
43
  connect(provider) {
65
- return new Signer(__classPrivateFieldGet(this, _Signer_address, "f"), __classPrivateFieldGet(this, _Signer_signerSession, "f"), { provider });
44
+ return new Signer(__classPrivateFieldGet(this, _Signer_signer, "f").address, __classPrivateFieldGet(this, _Signer_signer, "f").signerSession, {
45
+ ...__classPrivateFieldGet(this, _Signer_signer, "f").options,
46
+ provider,
47
+ });
66
48
  }
67
49
  /**
68
50
  * Construct a signing request from a transaction. This populates the transaction
@@ -101,9 +83,7 @@ class Signer extends ethers_1.ethers.AbstractSigner {
101
83
  */
102
84
  async signTransaction(tx) {
103
85
  const req = await this.evmSignRequestFromTx(tx);
104
- const res = await __classPrivateFieldGet(this, _Signer_signerSession, "f").signEvm(__classPrivateFieldGet(this, _Signer_address, "f"), req);
105
- const data = await __classPrivateFieldGet(this, _Signer_instances, "m", _Signer_handleMfa).call(this, res);
106
- return data.rlp_signed_tx;
86
+ return await __classPrivateFieldGet(this, _Signer_signer, "f").signTransaction(req);
107
87
  }
108
88
  /**
109
89
  * Signs arbitrary messages. This uses CubeSigner's EIP-191 signing endpoint.
@@ -113,12 +93,9 @@ class Signer extends ethers_1.ethers.AbstractSigner {
113
93
  * @return {Promise<string>} The signature.
114
94
  */
115
95
  async signMessage(message) {
116
- const key = await this.key();
117
- const res = await __classPrivateFieldGet(this, _Signer_signerSession, "f").signEip191(key.material_id, {
96
+ return await __classPrivateFieldGet(this, _Signer_signer, "f").signEip191({
118
97
  data: (0, cubesigner_sdk_1.encodeToHex)(message),
119
98
  });
120
- const data = await __classPrivateFieldGet(this, _Signer_instances, "m", _Signer_handleMfa).call(this, res);
121
- return data.signature;
122
99
  }
123
100
  /**
124
101
  * Signs EIP-712 typed data. This uses CubeSigner's EIP-712 signing endpoint.
@@ -130,7 +107,6 @@ class Signer extends ethers_1.ethers.AbstractSigner {
130
107
  * @return {Promise<string>} The signature.
131
108
  */
132
109
  async signTypedData(domain, types, value) {
133
- const key = await this.key();
134
110
  let chainId = domain.chainId;
135
111
  if (chainId === undefined) {
136
112
  // get chain id from provider
@@ -140,7 +116,7 @@ class Signer extends ethers_1.ethers.AbstractSigner {
140
116
  throw new Error("Cannot determine chainId");
141
117
  }
142
118
  }
143
- const res = await __classPrivateFieldGet(this, _Signer_signerSession, "f").signEip712(key.material_id, {
119
+ return await __classPrivateFieldGet(this, _Signer_signer, "f").signEip712({
144
120
  chain_id: Number(chainId),
145
121
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
146
122
  typed_data: {
@@ -150,19 +126,6 @@ class Signer extends ethers_1.ethers.AbstractSigner {
150
126
  message: value,
151
127
  },
152
128
  });
153
- const data = await __classPrivateFieldGet(this, _Signer_instances, "m", _Signer_handleMfa).call(this, res);
154
- return data.signature;
155
- }
156
- /** @return {KeyInfo} The key corresponding to this address */
157
- async key() {
158
- if (__classPrivateFieldGet(this, _Signer_key, "f") === undefined) {
159
- const key = (await __classPrivateFieldGet(this, _Signer_signerSession, "f").keys()).find((k) => k.material_id === __classPrivateFieldGet(this, _Signer_address, "f"));
160
- if (key === undefined) {
161
- throw new Error(`Cannot access key '${__classPrivateFieldGet(this, _Signer_address, "f")}'`);
162
- }
163
- __classPrivateFieldSet(this, _Signer_key, key, "f");
164
- }
165
- return __classPrivateFieldGet(this, _Signer_key, "f");
166
129
  }
167
130
  /**
168
131
  * Initialize the signing a message using MFA approvals. This method populates
@@ -173,7 +136,7 @@ class Signer extends ethers_1.ethers.AbstractSigner {
173
136
  async sendTransactionMfaInit(tx) {
174
137
  const popTx = await this.populateTransaction(tx);
175
138
  const req = await this.evmSignRequestFromTx(popTx);
176
- const res = await __classPrivateFieldGet(this, _Signer_signerSession, "f").signEvm(__classPrivateFieldGet(this, _Signer_address, "f"), req);
139
+ const res = await __classPrivateFieldGet(this, _Signer_signer, "f").signEvm(req);
177
140
  return res.mfaId();
178
141
  }
179
142
  /**
@@ -184,41 +147,10 @@ class Signer extends ethers_1.ethers.AbstractSigner {
184
147
  * @return {ethers.TransactionResponse} The result of submitting the transaction
185
148
  */
186
149
  async sendTransactionMfaApproved(mfaInfo) {
187
- if (!mfaInfo.request.path.includes("/eth1/sign/")) {
188
- throw new Error(`Expected EVM transaction signing request, got ${mfaInfo.request.path}`);
189
- }
190
- if (!mfaInfo.request.path.includes(__classPrivateFieldGet(this, _Signer_address, "f"))) {
191
- throw new Error(`Expected signing request for ${__classPrivateFieldGet(this, _Signer_address, "f")} but got ${mfaInfo.request.path}`);
192
- }
193
- const signedTx = await __classPrivateFieldGet(this, _Signer_signerSession, "f").signEvm(__classPrivateFieldGet(this, _Signer_address, "f"), mfaInfo.request.body, {
194
- mfaId: mfaInfo.id,
195
- mfaOrgId: __classPrivateFieldGet(this, _Signer_signerSession, "f").orgId,
196
- mfaConf: mfaInfo.receipt.confirmation,
197
- });
198
- return await this.provider.broadcastTransaction(signedTx.data().rlp_signed_tx);
150
+ const rlpSigned = await __classPrivateFieldGet(this, _Signer_signer, "f").signTransactionMfaApproved(mfaInfo);
151
+ return await this.provider.broadcastTransaction(rlpSigned);
199
152
  }
200
153
  }
201
154
  exports.Signer = Signer;
202
- _Signer_address = new WeakMap(), _Signer_key = new WeakMap(), _Signer_signerSession = new WeakMap(), _Signer_onMfaPoll = new WeakMap(), _Signer_mfaPollIntervalMs = new WeakMap(), _Signer_instances = new WeakSet(), _Signer_handleMfa =
203
- /**
204
- * If the sign request requires MFA, this method waits for approvals
205
- * @param {CubeSignerResponse<U>} res The response of a sign request
206
- * @return {Promise<U>} The sign data after MFA approvals
207
- */
208
- async function _Signer_handleMfa(res) {
209
- while (res.requiresMfa()) {
210
- await new Promise((resolve) => setTimeout(resolve, __classPrivateFieldGet(this, _Signer_mfaPollIntervalMs, "f")));
211
- const mfaId = res.mfaId();
212
- const mfaInfo = await __classPrivateFieldGet(this, _Signer_signerSession, "f").getMfaInfo(mfaId);
213
- __classPrivateFieldGet(this, _Signer_onMfaPoll, "f").call(this, mfaInfo);
214
- if (mfaInfo.receipt) {
215
- res = await res.signWithMfaApproval({
216
- mfaId,
217
- mfaOrgId: __classPrivateFieldGet(this, _Signer_signerSession, "f").orgId,
218
- mfaConf: mfaInfo.receipt.confirmation,
219
- });
220
- }
221
- }
222
- return res.data();
223
- };
224
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  JsonRpcApiProvider,\n  TypedDataDomain,\n  TypedDataField,\n  TypedDataEncoder,\n  ethers,\n  toBeHex,\n} from \"ethers\";\nimport {\n  CubeSignerResponse,\n  KeyInfo,\n  Eip191SignRequest,\n  Eip712SignRequest,\n  EvmSignRequest,\n  MfaRequestInfo,\n  SignerSession,\n  encodeToHex,\n} from \"@cubist-labs/cubesigner-sdk\";\n\n/** Options for the signer */\nexport interface SignerOptions {\n  /** Optional provider to use */\n  provider?: null | ethers.Provider;\n  /**\n   * The function to call when MFA information is retrieved. If this callback\n   * throws, no transaction is broadcast.\n   */\n  onMfaPoll?: (arg0: MfaRequestInfo) => void;\n  /**\n   * The amount of time (in milliseconds) to wait between checks for MFA\n   * updates. Default is 1000ms\n   */\n  mfaPollIntervalMs?: number;\n}\n\n/**\n * A ethers.js Signer using CubeSigner\n */\nexport class Signer extends ethers.AbstractSigner {\n  /** The address of the account */\n  readonly #address: string;\n\n  /** The key to use for signing */\n  #key?: KeyInfo;\n\n  /** The underlying session */\n  readonly #signerSession: SignerSession;\n\n  /**\n   * The function to call when MFA information is retrieved. If this callback\n   * throws, no transaction is broadcast.\n   */\n  readonly #onMfaPoll: (arg0: MfaRequestInfo) => void;\n\n  /** The amount of time to wait between checks for MFA updates */\n  readonly #mfaPollIntervalMs: number;\n\n  /**\n   * Create new Signer instance\n   * @param {KeyInfo | string} address The key or the eth address of the account to use.\n   * @param {SignerSession} signerSession The underlying Signer session.\n   * @param {SignerOptions} options The options to use for the Signer instance\n   */\n  constructor(address: KeyInfo | string, signerSession: SignerSession, options?: SignerOptions) {\n    super(options?.provider);\n    if (typeof address === \"string\") {\n      this.#address = address;\n    } else {\n      this.#address = address.materialId;\n      this.#key = address as KeyInfo;\n    }\n    this.#signerSession = signerSession;\n    this.#onMfaPoll = options?.onMfaPoll ?? ((/* _mfaInfo: MfaRequestInfo */) => {}); // eslint-disable-line @typescript-eslint/no-empty-function\n    this.#mfaPollIntervalMs = options?.mfaPollIntervalMs ?? 1000;\n  }\n\n  /** Resolves to the signer (potentially _NOT_ checksummed) address. */\n  async getAddress(): Promise<string> {\n    return this.#address;\n  }\n\n  /**\n   *  Returns the signer connected to %%provider%%.\n   *  @param {null | ethers.Provider} provider The optional provider instance to use.\n   *  @return {Signer} The signer connected to signer.\n   */\n  connect(provider: null | ethers.Provider): Signer {\n    return new Signer(this.#address, this.#signerSession, { provider });\n  }\n\n  /**\n   * Construct a signing request from a transaction. This populates the transaction\n   * type to `0x02` (EIP-1559) unless set.\n   *\n   * @param {ethers.TransactionRequest} tx The transaction\n   * @return {EvmSignRequest} The EVM sign request to be sent to CubeSigner\n   */\n  async evmSignRequestFromTx(tx: ethers.TransactionRequest): Promise<EvmSignRequest> {\n    // get the chain id from the network or tx\n    let chainId = tx.chainId;\n    if (chainId === undefined) {\n      const network = await this.provider?.getNetwork();\n      chainId = network?.chainId?.toString();\n      if (chainId === undefined) {\n        throw new Error(\"Cannot determine chainId\");\n      }\n    }\n\n    // Convert the transaction into a JSON-RPC transaction\n    const rpcTx =\n      this.provider instanceof JsonRpcApiProvider\n        ? this.provider.getRpcTransaction(tx)\n        : // We can just call the getRpcTransaction with a\n          // null receiver since it doesn't actually use it\n          // (and really should be declared static).\n          JsonRpcApiProvider.prototype.getRpcTransaction.call(null, tx);\n    rpcTx.type = toBeHex(tx.type ?? 0x02, 1); // we expect 0x0[0-2]\n\n    return <EvmSignRequest>{\n      chain_id: Number(chainId),\n      tx: rpcTx,\n    };\n  }\n\n  /**\n   * Sign a transaction. This method will block if the key requires MFA approval.\n   * @param {ethers.TransactionRequest} tx The transaction to sign.\n   * @return {Promise<string>} Hex-encoded RLP encoding of the transaction and its signature.\n   */\n  async signTransaction(tx: ethers.TransactionRequest): Promise<string> {\n    const req = await this.evmSignRequestFromTx(tx);\n    const res = await this.#signerSession.signEvm(this.#address, req);\n    const data = await this.#handleMfa(res);\n    return data.rlp_signed_tx;\n  }\n\n  /**\n   * Signs arbitrary messages. This uses CubeSigner's EIP-191 signing endpoint.\n   * The key (for this session) must have the `\"AllowRawBlobSigning\"` or\n   * `\"AllowEip191Signing\"` policy attached.\n   * @param {string | Uint8Array} message The message to sign.\n   * @return {Promise<string>} The signature.\n   */\n  async signMessage(message: string | Uint8Array): Promise<string> {\n    const key = await this.key();\n    const res = await this.#signerSession.signEip191(key.material_id, <Eip191SignRequest>{\n      data: encodeToHex(message),\n    });\n    const data = await this.#handleMfa(res);\n    return data.signature;\n  }\n\n  /**\n   * Signs EIP-712 typed data. This uses CubeSigner's EIP-712 signing endpoint.\n   * The key (for this session) must have the `\"AllowRawBlobSigning\"` or\n   * `\"AllowEip712Signing\"` policy attached.\n   * @param {TypedDataDomain} domain The domain of the typed data.\n   * @param {Record<string, Array<TypedDataField>>} types The types of the typed data.\n   * @param {Record<string, any>} value The value of the typed data.\n   * @return {Promise<string>} The signature.\n   */\n  async signTypedData(\n    domain: TypedDataDomain,\n    types: Record<string, Array<TypedDataField>>,\n    value: Record<string, any>, // eslint-disable-line @typescript-eslint/no-explicit-any\n  ): Promise<string> {\n    const key = await this.key();\n    let chainId = domain.chainId;\n    if (chainId === undefined) {\n      // get chain id from provider\n      const network = await this.provider?.getNetwork();\n      chainId = network?.chainId;\n      if (chainId === undefined) {\n        throw new Error(\"Cannot determine chainId\");\n      }\n    }\n    const res = await this.#signerSession.signEip712(key.material_id, <Eip712SignRequest>{\n      chain_id: Number(chainId),\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      typed_data: <any>{\n        domain,\n        types,\n        primaryType: TypedDataEncoder.getPrimaryType(types),\n        message: value,\n      },\n    });\n    const data = await this.#handleMfa(res);\n    return data.signature;\n  }\n\n  /** @return {KeyInfo} The key corresponding to this address */\n  private async key(): Promise<KeyInfo> {\n    if (this.#key === undefined) {\n      const key = (await this.#signerSession.keys()).find((k) => k.material_id === this.#address);\n      if (key === undefined) {\n        throw new Error(`Cannot access key '${this.#address}'`);\n      }\n      this.#key = key;\n    }\n    return this.#key;\n  }\n\n  /**\n   * Initialize the signing a message using MFA approvals. This method populates\n   * missing fields. If the signing does not require MFA, this method throws.\n   * @param {ethers.TransactionRequest} tx The transaction to send.\n   * @return {string} The MFA id associated with the signing request.\n   */\n  async sendTransactionMfaInit(tx: ethers.TransactionRequest): Promise<string> {\n    const popTx = await this.populateTransaction(tx);\n    const req = await this.evmSignRequestFromTx(popTx);\n    const res = await this.#signerSession.signEvm(this.#address, req);\n    return res.mfaId();\n  }\n\n  /**\n   * Send a transaction from an approved MFA request. The MFA request contains\n   * information about the approved signing request, which this method will\n   * execute.\n   * @param {MfaRequestInfo} mfaInfo The approved MFA request.\n   * @return {ethers.TransactionResponse} The result of submitting the transaction\n   */\n  async sendTransactionMfaApproved(mfaInfo: MfaRequestInfo): Promise<ethers.TransactionResponse> {\n    if (!mfaInfo.request.path.includes(\"/eth1/sign/\")) {\n      throw new Error(`Expected EVM transaction signing request, got ${mfaInfo.request.path}`);\n    }\n    if (!mfaInfo.request.path.includes(this.#address)) {\n      throw new Error(\n        `Expected signing request for ${this.#address} but got ${mfaInfo.request.path}`,\n      );\n    }\n\n    const signedTx = await this.#signerSession.signEvm(\n      this.#address,\n      mfaInfo.request.body as EvmSignRequest,\n      {\n        mfaId: mfaInfo.id,\n        mfaOrgId: this.#signerSession.orgId,\n        mfaConf: mfaInfo.receipt!.confirmation,\n      },\n    );\n    return await this.provider!.broadcastTransaction(signedTx.data().rlp_signed_tx);\n  }\n\n  /**\n   * If the sign request requires MFA, this method waits for approvals\n   * @param {CubeSignerResponse<U>} res The response of a sign request\n   * @return {Promise<U>} The sign data after MFA approvals\n   */\n  async #handleMfa<U>(res: CubeSignerResponse<U>): Promise<U> {\n    while (res.requiresMfa()) {\n      await new Promise((resolve) => setTimeout(resolve, this.#mfaPollIntervalMs));\n\n      const mfaId = res.mfaId();\n      const mfaInfo = await this.#signerSession.getMfaInfo(mfaId);\n      this.#onMfaPoll(mfaInfo);\n      if (mfaInfo.receipt) {\n        res = await res.signWithMfaApproval({\n          mfaId,\n          mfaOrgId: this.#signerSession.orgId,\n          mfaConf: mfaInfo.receipt.confirmation,\n        });\n      }\n    }\n    return res.data();\n  }\n}\n"]}
155
+ _Signer_signer = new WeakMap();
156
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  JsonRpcApiProvider,\n  TypedDataDomain,\n  TypedDataField,\n  TypedDataEncoder,\n  ethers,\n  toBeHex,\n} from \"ethers\";\nimport {\n  EvmSigner,\n  EvmSignerOptions,\n  KeyInfo,\n  Eip191SignRequest,\n  Eip712SignRequest,\n  EvmSignRequest,\n  MfaRequestInfo,\n  SignerSession,\n  encodeToHex,\n} from \"@cubist-labs/cubesigner-sdk\";\n\n/** Options for the signer */\nexport interface SignerOptions extends EvmSignerOptions {\n  /** Optional provider to use */\n  provider?: null | ethers.Provider;\n}\n\n/**\n * A ethers.js Signer using CubeSigner\n */\nexport class Signer extends ethers.AbstractSigner {\n  /** The CubeSigner-backed ethers signer */\n  readonly #signer: EvmSigner;\n\n  /**\n   * Create new Signer instance\n   * @param {KeyInfo | string} address The key or the eth address of the account to use.\n   * @param {SignerSession} signerSession The underlying Signer session.\n   * @param {SignerOptions} options The options to use for the Signer instance\n   */\n  constructor(address: KeyInfo | string, signerSession: SignerSession, options?: SignerOptions) {\n    super(options?.provider);\n    this.#signer = new EvmSigner(address, signerSession, options);\n  }\n\n  /** Resolves to the checksummed signer address. */\n  async getAddress(): Promise<string> {\n    return ethers.getAddress(this.#signer.address);\n  }\n\n  /**\n   *  Returns the signer connected to %%provider%%.\n   *  @param {null | ethers.Provider} provider The optional provider instance to use.\n   *  @return {Signer} The signer connected to signer.\n   */\n  connect(provider: null | ethers.Provider): Signer {\n    return new Signer(this.#signer.address, this.#signer.signerSession, {\n      ...this.#signer.options,\n      provider,\n    });\n  }\n\n  /**\n   * Construct a signing request from a transaction. This populates the transaction\n   * type to `0x02` (EIP-1559) unless set.\n   *\n   * @param {ethers.TransactionRequest} tx The transaction\n   * @return {EvmSignRequest} The EVM sign request to be sent to CubeSigner\n   */\n  async evmSignRequestFromTx(tx: ethers.TransactionRequest): Promise<EvmSignRequest> {\n    // get the chain id from the network or tx\n    let chainId = tx.chainId;\n    if (chainId === undefined) {\n      const network = await this.provider?.getNetwork();\n      chainId = network?.chainId?.toString();\n      if (chainId === undefined) {\n        throw new Error(\"Cannot determine chainId\");\n      }\n    }\n\n    // Convert the transaction into a JSON-RPC transaction\n    const rpcTx =\n      this.provider instanceof JsonRpcApiProvider\n        ? this.provider.getRpcTransaction(tx)\n        : // We can just call the getRpcTransaction with a\n          // null receiver since it doesn't actually use it\n          // (and really should be declared static).\n          JsonRpcApiProvider.prototype.getRpcTransaction.call(null, tx);\n    rpcTx.type = toBeHex(tx.type ?? 0x02, 1); // we expect 0x0[0-2]\n\n    return <EvmSignRequest>{\n      chain_id: Number(chainId),\n      tx: rpcTx,\n    };\n  }\n\n  /**\n   * Sign a transaction. This method will block if the key requires MFA approval.\n   * @param {ethers.TransactionRequest} tx The transaction to sign.\n   * @return {Promise<string>} Hex-encoded RLP encoding of the transaction and its signature.\n   */\n  async signTransaction(tx: ethers.TransactionRequest): Promise<string> {\n    const req = await this.evmSignRequestFromTx(tx);\n    return await this.#signer.signTransaction(req);\n  }\n\n  /**\n   * Signs arbitrary messages. This uses CubeSigner's EIP-191 signing endpoint.\n   * The key (for this session) must have the `\"AllowRawBlobSigning\"` or\n   * `\"AllowEip191Signing\"` policy attached.\n   * @param {string | Uint8Array} message The message to sign.\n   * @return {Promise<string>} The signature.\n   */\n  async signMessage(message: string | Uint8Array): Promise<string> {\n    return await this.#signer.signEip191(<Eip191SignRequest>{\n      data: encodeToHex(message),\n    });\n  }\n\n  /**\n   * Signs EIP-712 typed data. This uses CubeSigner's EIP-712 signing endpoint.\n   * The key (for this session) must have the `\"AllowRawBlobSigning\"` or\n   * `\"AllowEip712Signing\"` policy attached.\n   * @param {TypedDataDomain} domain The domain of the typed data.\n   * @param {Record<string, Array<TypedDataField>>} types The types of the typed data.\n   * @param {Record<string, any>} value The value of the typed data.\n   * @return {Promise<string>} The signature.\n   */\n  async signTypedData(\n    domain: TypedDataDomain,\n    types: Record<string, Array<TypedDataField>>,\n    value: Record<string, any>, // eslint-disable-line @typescript-eslint/no-explicit-any\n  ): Promise<string> {\n    let chainId = domain.chainId;\n    if (chainId === undefined) {\n      // get chain id from provider\n      const network = await this.provider?.getNetwork();\n      chainId = network?.chainId;\n      if (chainId === undefined) {\n        throw new Error(\"Cannot determine chainId\");\n      }\n    }\n    return await this.#signer.signEip712(<Eip712SignRequest>{\n      chain_id: Number(chainId),\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      typed_data: <any>{\n        domain,\n        types,\n        primaryType: TypedDataEncoder.getPrimaryType(types),\n        message: value,\n      },\n    });\n  }\n\n  /**\n   * Initialize the signing a message using MFA approvals. This method populates\n   * missing fields. If the signing does not require MFA, this method throws.\n   * @param {ethers.TransactionRequest} tx The transaction to send.\n   * @return {string} The MFA id associated with the signing request.\n   */\n  async sendTransactionMfaInit(tx: ethers.TransactionRequest): Promise<string> {\n    const popTx = await this.populateTransaction(tx);\n    const req = await this.evmSignRequestFromTx(popTx);\n    const res = await this.#signer.signEvm(req);\n    return res.mfaId();\n  }\n\n  /**\n   * Send a transaction from an approved MFA request. The MFA request contains\n   * information about the approved signing request, which this method will\n   * execute.\n   * @param {MfaRequestInfo} mfaInfo The approved MFA request.\n   * @return {ethers.TransactionResponse} The result of submitting the transaction\n   */\n  async sendTransactionMfaApproved(mfaInfo: MfaRequestInfo): Promise<ethers.TransactionResponse> {\n    const rlpSigned = await this.#signer.signTransactionMfaApproved(mfaInfo);\n    return await this.provider!.broadcastTransaction(rlpSigned);\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@cubist-labs/cubesigner-sdk-ethers-v6",
3
- "version": "0.3.13",
3
+ "version": "0.3.23",
4
4
  "description": "Ethers.js v6 Signer implementation",
5
5
  "license": "MIT OR Apache-2.0",
6
6
  "author": "Cubist, Inc.",
@@ -20,7 +20,7 @@
20
20
  "test": "jest --maxWorkers=1"
21
21
  },
22
22
  "peerDependencies": {
23
- "@cubist-labs/cubesigner-sdk": "^0.3.13",
23
+ "@cubist-labs/cubesigner-sdk": "^0.3.23",
24
24
  "ethers": "6.7.1"
25
25
  },
26
26
  "directories": {
package/src/index.ts CHANGED
@@ -7,7 +7,8 @@ import {
7
7
  toBeHex,
8
8
  } from "ethers";
9
9
  import {
10
- CubeSignerResponse,
10
+ EvmSigner,
11
+ EvmSignerOptions,
11
12
  KeyInfo,
12
13
  Eip191SignRequest,
13
14
  Eip712SignRequest,
@@ -18,42 +19,17 @@ import {
18
19
  } from "@cubist-labs/cubesigner-sdk";
19
20
 
20
21
  /** Options for the signer */
21
- export interface SignerOptions {
22
+ export interface SignerOptions extends EvmSignerOptions {
22
23
  /** Optional provider to use */
23
24
  provider?: null | ethers.Provider;
24
- /**
25
- * The function to call when MFA information is retrieved. If this callback
26
- * throws, no transaction is broadcast.
27
- */
28
- onMfaPoll?: (arg0: MfaRequestInfo) => void;
29
- /**
30
- * The amount of time (in milliseconds) to wait between checks for MFA
31
- * updates. Default is 1000ms
32
- */
33
- mfaPollIntervalMs?: number;
34
25
  }
35
26
 
36
27
  /**
37
28
  * A ethers.js Signer using CubeSigner
38
29
  */
39
30
  export class Signer extends ethers.AbstractSigner {
40
- /** The address of the account */
41
- readonly #address: string;
42
-
43
- /** The key to use for signing */
44
- #key?: KeyInfo;
45
-
46
- /** The underlying session */
47
- readonly #signerSession: SignerSession;
48
-
49
- /**
50
- * The function to call when MFA information is retrieved. If this callback
51
- * throws, no transaction is broadcast.
52
- */
53
- readonly #onMfaPoll: (arg0: MfaRequestInfo) => void;
54
-
55
- /** The amount of time to wait between checks for MFA updates */
56
- readonly #mfaPollIntervalMs: number;
31
+ /** The CubeSigner-backed ethers signer */
32
+ readonly #signer: EvmSigner;
57
33
 
58
34
  /**
59
35
  * Create new Signer instance
@@ -63,20 +39,12 @@ export class Signer extends ethers.AbstractSigner {
63
39
  */
64
40
  constructor(address: KeyInfo | string, signerSession: SignerSession, options?: SignerOptions) {
65
41
  super(options?.provider);
66
- if (typeof address === "string") {
67
- this.#address = address;
68
- } else {
69
- this.#address = address.materialId;
70
- this.#key = address as KeyInfo;
71
- }
72
- this.#signerSession = signerSession;
73
- this.#onMfaPoll = options?.onMfaPoll ?? ((/* _mfaInfo: MfaRequestInfo */) => {}); // eslint-disable-line @typescript-eslint/no-empty-function
74
- this.#mfaPollIntervalMs = options?.mfaPollIntervalMs ?? 1000;
42
+ this.#signer = new EvmSigner(address, signerSession, options);
75
43
  }
76
44
 
77
- /** Resolves to the signer (potentially _NOT_ checksummed) address. */
45
+ /** Resolves to the checksummed signer address. */
78
46
  async getAddress(): Promise<string> {
79
- return this.#address;
47
+ return ethers.getAddress(this.#signer.address);
80
48
  }
81
49
 
82
50
  /**
@@ -85,7 +53,10 @@ export class Signer extends ethers.AbstractSigner {
85
53
  * @return {Signer} The signer connected to signer.
86
54
  */
87
55
  connect(provider: null | ethers.Provider): Signer {
88
- return new Signer(this.#address, this.#signerSession, { provider });
56
+ return new Signer(this.#signer.address, this.#signer.signerSession, {
57
+ ...this.#signer.options,
58
+ provider,
59
+ });
89
60
  }
90
61
 
91
62
  /**
@@ -129,9 +100,7 @@ export class Signer extends ethers.AbstractSigner {
129
100
  */
130
101
  async signTransaction(tx: ethers.TransactionRequest): Promise<string> {
131
102
  const req = await this.evmSignRequestFromTx(tx);
132
- const res = await this.#signerSession.signEvm(this.#address, req);
133
- const data = await this.#handleMfa(res);
134
- return data.rlp_signed_tx;
103
+ return await this.#signer.signTransaction(req);
135
104
  }
136
105
 
137
106
  /**
@@ -142,12 +111,9 @@ export class Signer extends ethers.AbstractSigner {
142
111
  * @return {Promise<string>} The signature.
143
112
  */
144
113
  async signMessage(message: string | Uint8Array): Promise<string> {
145
- const key = await this.key();
146
- const res = await this.#signerSession.signEip191(key.material_id, <Eip191SignRequest>{
114
+ return await this.#signer.signEip191(<Eip191SignRequest>{
147
115
  data: encodeToHex(message),
148
116
  });
149
- const data = await this.#handleMfa(res);
150
- return data.signature;
151
117
  }
152
118
 
153
119
  /**
@@ -164,7 +130,6 @@ export class Signer extends ethers.AbstractSigner {
164
130
  types: Record<string, Array<TypedDataField>>,
165
131
  value: Record<string, any>, // eslint-disable-line @typescript-eslint/no-explicit-any
166
132
  ): Promise<string> {
167
- const key = await this.key();
168
133
  let chainId = domain.chainId;
169
134
  if (chainId === undefined) {
170
135
  // get chain id from provider
@@ -174,7 +139,7 @@ export class Signer extends ethers.AbstractSigner {
174
139
  throw new Error("Cannot determine chainId");
175
140
  }
176
141
  }
177
- const res = await this.#signerSession.signEip712(key.material_id, <Eip712SignRequest>{
142
+ return await this.#signer.signEip712(<Eip712SignRequest>{
178
143
  chain_id: Number(chainId),
179
144
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
180
145
  typed_data: <any>{
@@ -184,20 +149,6 @@ export class Signer extends ethers.AbstractSigner {
184
149
  message: value,
185
150
  },
186
151
  });
187
- const data = await this.#handleMfa(res);
188
- return data.signature;
189
- }
190
-
191
- /** @return {KeyInfo} The key corresponding to this address */
192
- private async key(): Promise<KeyInfo> {
193
- if (this.#key === undefined) {
194
- const key = (await this.#signerSession.keys()).find((k) => k.material_id === this.#address);
195
- if (key === undefined) {
196
- throw new Error(`Cannot access key '${this.#address}'`);
197
- }
198
- this.#key = key;
199
- }
200
- return this.#key;
201
152
  }
202
153
 
203
154
  /**
@@ -209,7 +160,7 @@ export class Signer extends ethers.AbstractSigner {
209
160
  async sendTransactionMfaInit(tx: ethers.TransactionRequest): Promise<string> {
210
161
  const popTx = await this.populateTransaction(tx);
211
162
  const req = await this.evmSignRequestFromTx(popTx);
212
- const res = await this.#signerSession.signEvm(this.#address, req);
163
+ const res = await this.#signer.signEvm(req);
213
164
  return res.mfaId();
214
165
  }
215
166
 
@@ -221,47 +172,7 @@ export class Signer extends ethers.AbstractSigner {
221
172
  * @return {ethers.TransactionResponse} The result of submitting the transaction
222
173
  */
223
174
  async sendTransactionMfaApproved(mfaInfo: MfaRequestInfo): Promise<ethers.TransactionResponse> {
224
- if (!mfaInfo.request.path.includes("/eth1/sign/")) {
225
- throw new Error(`Expected EVM transaction signing request, got ${mfaInfo.request.path}`);
226
- }
227
- if (!mfaInfo.request.path.includes(this.#address)) {
228
- throw new Error(
229
- `Expected signing request for ${this.#address} but got ${mfaInfo.request.path}`,
230
- );
231
- }
232
-
233
- const signedTx = await this.#signerSession.signEvm(
234
- this.#address,
235
- mfaInfo.request.body as EvmSignRequest,
236
- {
237
- mfaId: mfaInfo.id,
238
- mfaOrgId: this.#signerSession.orgId,
239
- mfaConf: mfaInfo.receipt!.confirmation,
240
- },
241
- );
242
- return await this.provider!.broadcastTransaction(signedTx.data().rlp_signed_tx);
243
- }
244
-
245
- /**
246
- * If the sign request requires MFA, this method waits for approvals
247
- * @param {CubeSignerResponse<U>} res The response of a sign request
248
- * @return {Promise<U>} The sign data after MFA approvals
249
- */
250
- async #handleMfa<U>(res: CubeSignerResponse<U>): Promise<U> {
251
- while (res.requiresMfa()) {
252
- await new Promise((resolve) => setTimeout(resolve, this.#mfaPollIntervalMs));
253
-
254
- const mfaId = res.mfaId();
255
- const mfaInfo = await this.#signerSession.getMfaInfo(mfaId);
256
- this.#onMfaPoll(mfaInfo);
257
- if (mfaInfo.receipt) {
258
- res = await res.signWithMfaApproval({
259
- mfaId,
260
- mfaOrgId: this.#signerSession.orgId,
261
- mfaConf: mfaInfo.receipt.confirmation,
262
- });
263
- }
264
- }
265
- return res.data();
175
+ const rlpSigned = await this.#signer.signTransactionMfaApproved(mfaInfo);
176
+ return await this.provider!.broadcastTransaction(rlpSigned);
266
177
  }
267
178
  }
package/tsconfig.json CHANGED
@@ -3,12 +3,12 @@
3
3
  "compilerOptions": {
4
4
  "module": "Node16",
5
5
  "moduleResolution": "node16",
6
- "outDir": "./dist",
6
+ "outDir": "./dist"
7
7
  },
8
8
  "typedocOptions": {
9
9
  "out": "./docs",
10
- "entryPoints": ["src/index.ts"],
10
+ "entryPoints": ["src/index.ts"]
11
11
  },
12
12
  "exclude": ["node_modules", "dist"],
13
- "include": ["src/**/*.ts"],
13
+ "include": ["src/**/*.ts"]
14
14
  }