@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 +3 -15
- package/dist/index.js +18 -86
- package/package.json +2 -2
- package/src/index.ts +18 -107
- package/tsconfig.json +3 -3
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
|
|
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
|
|
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
|
-
|
|
31
|
-
|
|
32
|
-
|
|
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
|
|
34
|
+
/** Resolves to the checksummed signer address. */
|
|
56
35
|
async getAddress() {
|
|
57
|
-
return __classPrivateFieldGet(this,
|
|
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,
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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,
|
|
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
|
-
|
|
188
|
-
|
|
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
|
-
|
|
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":";;;;;;;;;;;;;;;AAAA,mCAOgB;AAChB,gEAUqC;AAQrC;;GAEG;AACH,MAAa,MAAO,SAAQ,eAAM,CAAC,cAAc;IAI/C;;;;;OAKG;IACH,YAAY,OAAyB,EAAE,aAA4B,EAAE,OAAuB;QAC1F,KAAK,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;QAV3B,0CAA0C;QACjC,iCAAmB;QAU1B,uBAAA,IAAI,kBAAW,IAAI,0BAAS,CAAC,OAAO,EAAE,aAAa,EAAE,OAAO,CAAC,MAAA,CAAC;IAChE,CAAC;IAED,kDAAkD;IAClD,KAAK,CAAC,UAAU;QACd,OAAO,eAAM,CAAC,UAAU,CAAC,uBAAA,IAAI,sBAAQ,CAAC,OAAO,CAAC,CAAC;IACjD,CAAC;IAED;;;;OAIG;IACH,OAAO,CAAC,QAAgC;QACtC,OAAO,IAAI,MAAM,CAAC,uBAAA,IAAI,sBAAQ,CAAC,OAAO,EAAE,uBAAA,IAAI,sBAAQ,CAAC,aAAa,EAAE;YAClE,GAAG,uBAAA,IAAI,sBAAQ,CAAC,OAAO;YACvB,QAAQ;SACT,CAAC,CAAC;IACL,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,oBAAoB,CAAC,EAA6B;QACtD,0CAA0C;QAC1C,IAAI,OAAO,GAAG,EAAE,CAAC,OAAO,CAAC;QACzB,IAAI,OAAO,KAAK,SAAS,EAAE,CAAC;YAC1B,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,QAAQ,EAAE,UAAU,EAAE,CAAC;YAClD,OAAO,GAAG,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,CAAC;YACvC,IAAI,OAAO,KAAK,SAAS,EAAE,CAAC;gBAC1B,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;YAC9C,CAAC;QACH,CAAC;QAED,sDAAsD;QACtD,MAAM,KAAK,GACT,IAAI,CAAC,QAAQ,YAAY,2BAAkB;YACzC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,iBAAiB,CAAC,EAAE,CAAC;YACrC,CAAC,CAAC,gDAAgD;gBAChD,iDAAiD;gBACjD,0CAA0C;gBAC1C,2BAAkB,CAAC,SAAS,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QACpE,KAAK,CAAC,IAAI,GAAG,IAAA,gBAAO,EAAC,EAAE,CAAC,IAAI,IAAI,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,qBAAqB;QAE/D,OAAuB;YACrB,QAAQ,EAAE,MAAM,CAAC,OAAO,CAAC;YACzB,EAAE,EAAE,KAAK;SACV,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,eAAe,CAAC,EAA6B;QACjD,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,oBAAoB,CAAC,EAAE,CAAC,CAAC;QAChD,OAAO,MAAM,uBAAA,IAAI,sBAAQ,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC;IACjD,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,WAAW,CAAC,OAA4B;QAC5C,OAAO,MAAM,uBAAA,IAAI,sBAAQ,CAAC,UAAU,CAAoB;YACtD,IAAI,EAAE,IAAA,4BAAW,EAAC,OAAO,CAAC;SAC3B,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;;OAQG;IACH,KAAK,CAAC,aAAa,CACjB,MAAuB,EACvB,KAA4C,EAC5C,KAA0B;QAE1B,IAAI,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC;QAC7B,IAAI,OAAO,KAAK,SAAS,EAAE,CAAC;YAC1B,6BAA6B;YAC7B,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,QAAQ,EAAE,UAAU,EAAE,CAAC;YAClD,OAAO,GAAG,OAAO,EAAE,OAAO,CAAC;YAC3B,IAAI,OAAO,KAAK,SAAS,EAAE,CAAC;gBAC1B,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;YAC9C,CAAC;QACH,CAAC;QACD,OAAO,MAAM,uBAAA,IAAI,sBAAQ,CAAC,UAAU,CAAoB;YACtD,QAAQ,EAAE,MAAM,CAAC,OAAO,CAAC;YACzB,8DAA8D;YAC9D,UAAU,EAAO;gBACf,MAAM;gBACN,KAAK;gBACL,WAAW,EAAE,yBAAgB,CAAC,cAAc,CAAC,KAAK,CAAC;gBACnD,OAAO,EAAE,KAAK;aACf;SACF,CAAC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,sBAAsB,CAAC,EAA6B;QACxD,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,mBAAmB,CAAC,EAAE,CAAC,CAAC;QACjD,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,CAAC;QACnD,MAAM,GAAG,GAAG,MAAM,uBAAA,IAAI,sBAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAC5C,OAAO,GAAG,CAAC,KAAK,EAAE,CAAC;IACrB,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,0BAA0B,CAAC,OAAuB;QACtD,MAAM,SAAS,GAAG,MAAM,uBAAA,IAAI,sBAAQ,CAAC,0BAA0B,CAAC,OAAO,CAAC,CAAC;QACzE,OAAO,MAAM,IAAI,CAAC,QAAS,CAAC,oBAAoB,CAAC,SAAS,CAAC,CAAC;IAC9D,CAAC;CACF;AApJD,wBAoJC","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.
|
|
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.
|
|
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
|
-
|
|
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
|
|
41
|
-
readonly #
|
|
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
|
-
|
|
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
|
|
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, {
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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.#
|
|
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
|
-
|
|
225
|
-
|
|
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
|
}
|