@cubist-labs/cubesigner-sdk-viem 0.4.183-0
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/README.md +51 -0
- package/dist/index.d.ts +55 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +129 -0
- package/package.json +38 -0
- package/src/index.ts +158 -0
package/README.md
ADDED
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
# CubeSigner Plugin for Viem
|
|
2
|
+
|
|
3
|
+
This package exposes a single `CubeSignerSource` class which implements
|
|
4
|
+
a custom source in Viem, backed by CubeSigner.
|
|
5
|
+
|
|
6
|
+
## Simple example usage
|
|
7
|
+
|
|
8
|
+
```typescript
|
|
9
|
+
import { type CustomSource, createWalletClient, http, publicActions } from "viem";
|
|
10
|
+
import { CubeSignerClient } from "@cubist-labs/cubesigner-sdk";
|
|
11
|
+
import { CubeSignerSource } from "@cubist-labs/cubesigner-sdk-viem";
|
|
12
|
+
import { sepolia } from "viem/chains";
|
|
13
|
+
import { toAccount } from "viem/accounts";
|
|
14
|
+
|
|
15
|
+
const client = await CubeSignerClient.create(...); // must have permissions to sign with `key`
|
|
16
|
+
const key = ...; // A CubeSigner key object or a string of an address in your org
|
|
17
|
+
const chain = sepolia; // use an object from "viem/chains"
|
|
18
|
+
|
|
19
|
+
const account = toAccount(new CubeSignerSource(key, client) as CustomSource);
|
|
20
|
+
|
|
21
|
+
// Create a WalletClient to perform actions
|
|
22
|
+
const walletClient = createWalletClient({
|
|
23
|
+
account,
|
|
24
|
+
chain,
|
|
25
|
+
transport: http(), // uses Viem's default RPC provider
|
|
26
|
+
}).extend(publicActions);
|
|
27
|
+
|
|
28
|
+
// Sign transaction as usual:
|
|
29
|
+
const tx = await walletClient.prepareTransactionRequest({
|
|
30
|
+
to: "0x96be1e4c198ecb1a55e769f653b1934950294f19",
|
|
31
|
+
value: 0n,
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
const signature = await walletClient.signTransaction(tx);
|
|
35
|
+
...
|
|
36
|
+
```
|
|
37
|
+
|
|
38
|
+
Check out our [Viem example](../../examples/viem/src/index.ts) or the unit tests
|
|
39
|
+
in the `test` folder for more examples.
|
|
40
|
+
|
|
41
|
+
The [@cubist-labs/cubesigner-sdk](https://www.npmjs.com/package/@cubist-labs/cubesigner-sdk) contains more details on how to create signer sessions.
|
|
42
|
+
|
|
43
|
+
## Supported Transactions
|
|
44
|
+
|
|
45
|
+
Please note that CubeSigner only supports legacy or EIP-1559 EVM
|
|
46
|
+
transactions with an explicitly defined `type` field. It's recommended to use
|
|
47
|
+
`prepareTransactionRequest` before signing to fill this field.
|
|
48
|
+
|
|
49
|
+
Transactions must take place on a specific chain---either use a client which
|
|
50
|
+
holds a specific chain, or ensure your transaction has the `chainId` field
|
|
51
|
+
defined.
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
import { type Address, type CustomSource } from "viem";
|
|
2
|
+
import { type CubeSignerClient, type EvmSignerOptions, type Key } from "@cubist-labs/cubesigner-sdk";
|
|
3
|
+
/**
|
|
4
|
+
* A class to wrap a CubeSigner key and client into a Viem {@link CustomSource}.
|
|
5
|
+
* Use Viem's `toAccount` to convert this to a Viem Account.
|
|
6
|
+
*/
|
|
7
|
+
export declare class CubeSignerSource implements CustomSource {
|
|
8
|
+
#private;
|
|
9
|
+
/** The address the wallet is associated with. */
|
|
10
|
+
readonly address: Address;
|
|
11
|
+
/**
|
|
12
|
+
* Construct a Viem {@link CustomSource} around a CubeSigner key and client.
|
|
13
|
+
* Use Viem's `toAccount` to convert this to a Viem Account.
|
|
14
|
+
*
|
|
15
|
+
* @param address The EVM address this wallet is associated with
|
|
16
|
+
* @param client The session used for signing. Must have necessary scopes.
|
|
17
|
+
* @param options MFA options for the client to respect
|
|
18
|
+
* @throws {Error} if the address is not a valid EVM address
|
|
19
|
+
*/
|
|
20
|
+
constructor(address: Key | string, client: CubeSignerClient, options?: EvmSignerOptions);
|
|
21
|
+
/**
|
|
22
|
+
* Signs arbitrary messages. This uses CubeSigner's EIP-191 signing endpoint.
|
|
23
|
+
* The key (for this session) must have the `"AllowRawBlobSigning"` or
|
|
24
|
+
* `"AllowEip191Signing"` policy attached.
|
|
25
|
+
*
|
|
26
|
+
* @param root Object around the message
|
|
27
|
+
* @param root.message The message to sign
|
|
28
|
+
* @returns The signature
|
|
29
|
+
*/
|
|
30
|
+
signMessage: CustomSource["signMessage"];
|
|
31
|
+
/**
|
|
32
|
+
* Signs EIP-712 typed data. This uses CubeSigner's EIP-712 signing endpoint.
|
|
33
|
+
* The key (for this session) must have the `"AllowRawBlobSigning"` or
|
|
34
|
+
* `"AllowEip712Signing"` policy attached.
|
|
35
|
+
* `chainId` must be specified within the `domain`.
|
|
36
|
+
*
|
|
37
|
+
* @param parameters Typed data
|
|
38
|
+
* @returns signature for the typed data
|
|
39
|
+
*/
|
|
40
|
+
signTypedData: CustomSource["signTypedData"];
|
|
41
|
+
/**
|
|
42
|
+
* It is recommended to use `prepareTransactionRequest` on your request
|
|
43
|
+
* before calling this function.
|
|
44
|
+
*
|
|
45
|
+
* Sign a transaction. This method will block if the key requires MFA approval.
|
|
46
|
+
* `type` and `chainId` must be defined. Only supports type "legacy" or "eip1559".
|
|
47
|
+
*
|
|
48
|
+
* @param transaction The transaction to sign
|
|
49
|
+
* @param options Contains an optional custom serializer
|
|
50
|
+
* @returns Signed transaction
|
|
51
|
+
* @throws {Error} if transaction.type isn't "legacy" or "eip1559", or if "chainId" isn't specified
|
|
52
|
+
*/
|
|
53
|
+
signTransaction: CustomSource["signTransaction"];
|
|
54
|
+
}
|
|
55
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,KAAK,OAAO,EACZ,KAAK,YAAY,EAUlB,MAAM,MAAM,CAAC;AACd,OAAO,EACL,KAAK,gBAAgB,EAIrB,KAAK,gBAAgB,EACrB,KAAK,GAAG,EACT,MAAM,6BAA6B,CAAC;AAErC;;;GAGG;AACH,qBAAa,gBAAiB,YAAW,YAAY;;IAInD,iDAAiD;IACjD,SAAgB,OAAO,EAAE,OAAO,CAAC;IAEjC;;;;;;;;OAQG;gBACS,OAAO,EAAE,GAAG,GAAG,MAAM,EAAE,MAAM,EAAE,gBAAgB,EAAE,OAAO,CAAC,EAAE,gBAAgB;IAYvF;;;;;;;;OAQG;IACI,WAAW,EAAE,YAAY,CAAC,aAAa,CAAC,CAa7C;IAEF;;;;;;;;OAQG;IACI,aAAa,EAAE,YAAY,CAAC,eAAe,CAAC,CAYjD;IAEF;;;;;;;;;;;OAWG;IACI,eAAe,EAAE,YAAY,CAAC,iBAAiB,CAAC,CAsBrD;CACH"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) {
|
|
2
|
+
if (kind === "m") throw new TypeError("Private method is not writable");
|
|
3
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
|
|
4
|
+
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
|
|
5
|
+
return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
|
|
6
|
+
};
|
|
7
|
+
var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
|
|
8
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
|
|
9
|
+
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");
|
|
10
|
+
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
|
|
11
|
+
};
|
|
12
|
+
var _CubeSignerSource_signer;
|
|
13
|
+
import { bytesToHex, formatTransactionRequest, getAddress, isHex, parseTransaction, serializeTransaction, stringToHex, } from "viem";
|
|
14
|
+
import { EvmSigner, } from "@cubist-labs/cubesigner-sdk";
|
|
15
|
+
/**
|
|
16
|
+
* A class to wrap a CubeSigner key and client into a Viem {@link CustomSource}.
|
|
17
|
+
* Use Viem's `toAccount` to convert this to a Viem Account.
|
|
18
|
+
*/
|
|
19
|
+
export class CubeSignerSource {
|
|
20
|
+
/**
|
|
21
|
+
* Construct a Viem {@link CustomSource} around a CubeSigner key and client.
|
|
22
|
+
* Use Viem's `toAccount` to convert this to a Viem Account.
|
|
23
|
+
*
|
|
24
|
+
* @param address The EVM address this wallet is associated with
|
|
25
|
+
* @param client The session used for signing. Must have necessary scopes.
|
|
26
|
+
* @param options MFA options for the client to respect
|
|
27
|
+
* @throws {Error} if the address is not a valid EVM address
|
|
28
|
+
*/
|
|
29
|
+
constructor(address, client, options) {
|
|
30
|
+
/** The internal CubeSigner signer used. */
|
|
31
|
+
_CubeSignerSource_signer.set(this, void 0);
|
|
32
|
+
/**
|
|
33
|
+
* Signs arbitrary messages. This uses CubeSigner's EIP-191 signing endpoint.
|
|
34
|
+
* The key (for this session) must have the `"AllowRawBlobSigning"` or
|
|
35
|
+
* `"AllowEip191Signing"` policy attached.
|
|
36
|
+
*
|
|
37
|
+
* @param root Object around the message
|
|
38
|
+
* @param root.message The message to sign
|
|
39
|
+
* @returns The signature
|
|
40
|
+
*/
|
|
41
|
+
this.signMessage = async ({ message }) => {
|
|
42
|
+
let hex;
|
|
43
|
+
if (typeof message === "string") {
|
|
44
|
+
hex = stringToHex(message);
|
|
45
|
+
}
|
|
46
|
+
else if (isHex(message.raw)) {
|
|
47
|
+
hex = message.raw;
|
|
48
|
+
}
|
|
49
|
+
else {
|
|
50
|
+
hex = bytesToHex(message.raw);
|
|
51
|
+
}
|
|
52
|
+
const signature = await __classPrivateFieldGet(this, _CubeSignerSource_signer, "f").signEip191({ data: hex });
|
|
53
|
+
return ensureHex(signature);
|
|
54
|
+
};
|
|
55
|
+
/**
|
|
56
|
+
* Signs EIP-712 typed data. This uses CubeSigner's EIP-712 signing endpoint.
|
|
57
|
+
* The key (for this session) must have the `"AllowRawBlobSigning"` or
|
|
58
|
+
* `"AllowEip712Signing"` policy attached.
|
|
59
|
+
* `chainId` must be specified within the `domain`.
|
|
60
|
+
*
|
|
61
|
+
* @param parameters Typed data
|
|
62
|
+
* @returns signature for the typed data
|
|
63
|
+
*/
|
|
64
|
+
this.signTypedData = async (parameters) => {
|
|
65
|
+
assert(parameters.domain, "`domain` must be defined");
|
|
66
|
+
const castedParameters = parameters;
|
|
67
|
+
assert(castedParameters.domain.chainId, "`domain.chainId` must be defined");
|
|
68
|
+
const signature = await __classPrivateFieldGet(this, _CubeSignerSource_signer, "f").signEip712({
|
|
69
|
+
chain_id: Number(castedParameters.domain.chainId),
|
|
70
|
+
typed_data: castedParameters,
|
|
71
|
+
});
|
|
72
|
+
return ensureHex(signature);
|
|
73
|
+
};
|
|
74
|
+
/**
|
|
75
|
+
* It is recommended to use `prepareTransactionRequest` on your request
|
|
76
|
+
* before calling this function.
|
|
77
|
+
*
|
|
78
|
+
* Sign a transaction. This method will block if the key requires MFA approval.
|
|
79
|
+
* `type` and `chainId` must be defined. Only supports type "legacy" or "eip1559".
|
|
80
|
+
*
|
|
81
|
+
* @param transaction The transaction to sign
|
|
82
|
+
* @param options Contains an optional custom serializer
|
|
83
|
+
* @returns Signed transaction
|
|
84
|
+
* @throws {Error} if transaction.type isn't "legacy" or "eip1559", or if "chainId" isn't specified
|
|
85
|
+
*/
|
|
86
|
+
this.signTransaction = async (transaction, options) => {
|
|
87
|
+
assert(transaction.type === "legacy" || transaction.type === "eip1559", `Unsupported transaction type '${transaction.type}', CubeSigner only supports type 'legacy' or 'eip1559'")}'`);
|
|
88
|
+
assert(transaction.chainId, "`chainId` must be defined");
|
|
89
|
+
const formatted = formatTransactionRequest(transaction);
|
|
90
|
+
const rlpSignedTransaction = await __classPrivateFieldGet(this, _CubeSignerSource_signer, "f").signTransaction({
|
|
91
|
+
chain_id: transaction.chainId,
|
|
92
|
+
tx: formatted,
|
|
93
|
+
});
|
|
94
|
+
// CubeSigner returns an RLP-encoded transaction. Since Viem allows users to pass a custom serializer, we will
|
|
95
|
+
// now unserialize and reserialize with Viem's serializer.
|
|
96
|
+
const { r, s, v, yParity, ...parsedTransaction } = parseTransaction(ensureHex(rlpSignedTransaction));
|
|
97
|
+
const serializer = options?.serializer ?? serializeTransaction;
|
|
98
|
+
return await serializer(parsedTransaction, { r, s, v, yParity });
|
|
99
|
+
};
|
|
100
|
+
__classPrivateFieldSet(this, _CubeSignerSource_signer, new EvmSigner(address, client, options), "f");
|
|
101
|
+
// NOTE: `getAddress` will checksum the address and throw if it's an invalid EVM address.
|
|
102
|
+
this.address = getAddress(__classPrivateFieldGet(this, _CubeSignerSource_signer, "f").address);
|
|
103
|
+
// Scope these functions to properly resolve `this`
|
|
104
|
+
this.signMessage = this.signMessage.bind(this);
|
|
105
|
+
this.signTransaction = this.signTransaction.bind(this);
|
|
106
|
+
this.signTypedData = this.signTypedData.bind(this);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
_CubeSignerSource_signer = new WeakMap();
|
|
110
|
+
/**
|
|
111
|
+
* @param input A hex string
|
|
112
|
+
* @returns the input, type narrowed to Hex
|
|
113
|
+
* @throws {Error} if input is not Hex
|
|
114
|
+
*/
|
|
115
|
+
function ensureHex(input) {
|
|
116
|
+
assert(isHex(input), `${input} is not hex`);
|
|
117
|
+
return input;
|
|
118
|
+
}
|
|
119
|
+
/**
|
|
120
|
+
* @param value A value that is expected to be truthy
|
|
121
|
+
* @param message The error message if the value is falsy
|
|
122
|
+
* @throws {Error} if value is not truthy
|
|
123
|
+
*/
|
|
124
|
+
function assert(value, message) {
|
|
125
|
+
if (!value) {
|
|
126
|
+
throw new Error(message);
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  type Address,\n  type CustomSource,\n  type Hex,\n  type Signature,\n  bytesToHex,\n  formatTransactionRequest,\n  getAddress,\n  isHex,\n  parseTransaction,\n  serializeTransaction,\n  stringToHex,\n} from \"viem\";\nimport {\n  type CubeSignerClient,\n  type Eip712SignRequest,\n  type EvmSignRequest,\n  EvmSigner,\n  type EvmSignerOptions,\n  type Key,\n} from \"@cubist-labs/cubesigner-sdk\";\n\n/**\n * A class to wrap a CubeSigner key and client into a Viem {@link CustomSource}.\n * Use Viem's `toAccount` to convert this to a Viem Account.\n */\nexport class CubeSignerSource implements CustomSource {\n  /** The internal CubeSigner signer used. */\n  readonly #signer: EvmSigner;\n\n  /** The address the wallet is associated with. */\n  public readonly address: Address;\n\n  /**\n   * Construct a Viem {@link CustomSource} around a CubeSigner key and client.\n   * Use Viem's `toAccount` to convert this to a Viem Account.\n   *\n   * @param address The EVM address this wallet is associated with\n   * @param client The session used for signing. Must have necessary scopes.\n   * @param options MFA options for the client to respect\n   * @throws {Error} if the address is not a valid EVM address\n   */\n  constructor(address: Key | string, client: CubeSignerClient, options?: EvmSignerOptions) {\n    this.#signer = new EvmSigner(address, client, options);\n\n    // NOTE: `getAddress` will checksum the address and throw if it's an invalid EVM address.\n    this.address = getAddress(this.#signer.address);\n\n    // Scope these functions to properly resolve `this`\n    this.signMessage = this.signMessage.bind(this);\n    this.signTransaction = this.signTransaction.bind(this);\n    this.signTypedData = this.signTypedData.bind(this);\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   *\n   * @param root Object around the message\n   * @param root.message The message to sign\n   * @returns The signature\n   */\n  public signMessage: CustomSource[\"signMessage\"] = async ({ message }) => {\n    let hex;\n    if (typeof message === \"string\") {\n      hex = stringToHex(message);\n    } else if (isHex(message.raw)) {\n      hex = message.raw;\n    } else {\n      hex = bytesToHex(message.raw);\n    }\n\n    const signature = await this.#signer.signEip191({ data: hex });\n\n    return ensureHex(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   * `chainId` must be specified within the `domain`.\n   *\n   * @param parameters Typed data\n   * @returns signature for the typed data\n   */\n  public signTypedData: CustomSource[\"signTypedData\"] = async (parameters) => {\n    assert(parameters.domain, \"`domain` must be defined\");\n\n    const castedParameters = parameters as Eip712SignRequest[\"typed_data\"];\n    assert(castedParameters.domain.chainId, \"`domain.chainId` must be defined\");\n\n    const signature = await this.#signer.signEip712({\n      chain_id: Number(castedParameters.domain.chainId),\n      typed_data: castedParameters,\n    });\n\n    return ensureHex(signature);\n  };\n\n  /**\n   * It is recommended to use `prepareTransactionRequest` on your request\n   * before calling this function.\n   *\n   * Sign a transaction. This method will block if the key requires MFA approval.\n   * `type` and `chainId` must be defined. Only supports type \"legacy\" or \"eip1559\".\n   *\n   * @param transaction The transaction to sign\n   * @param options Contains an optional custom serializer\n   * @returns Signed transaction\n   * @throws {Error} if transaction.type isn't \"legacy\" or \"eip1559\", or if \"chainId\" isn't specified\n   */\n  public signTransaction: CustomSource[\"signTransaction\"] = async (transaction, options) => {\n    assert(\n      transaction.type === \"legacy\" || transaction.type === \"eip1559\",\n      `Unsupported transaction type '${transaction.type}', CubeSigner only supports type 'legacy' or 'eip1559'\")}'`,\n    );\n\n    assert(transaction.chainId, \"`chainId` must be defined\");\n\n    const formatted = formatTransactionRequest(transaction);\n    const rlpSignedTransaction = await this.#signer.signTransaction({\n      chain_id: transaction.chainId,\n      tx: formatted as EvmSignRequest[\"tx\"],\n    });\n\n    // CubeSigner returns an RLP-encoded transaction. Since Viem allows users to pass a custom serializer, we will\n    // now unserialize and reserialize with Viem's serializer.\n    const { r, s, v, yParity, ...parsedTransaction } = parseTransaction(\n      ensureHex(rlpSignedTransaction),\n    );\n    const serializer = options?.serializer ?? serializeTransaction;\n\n    return await serializer(parsedTransaction, { r, s, v, yParity } as Signature);\n  };\n}\n\n/**\n * @param input A hex string\n * @returns the input, type narrowed to Hex\n * @throws {Error} if input is not Hex\n */\nfunction ensureHex(input: string): Hex {\n  assert(isHex(input), `${input} is not hex`);\n  return input;\n}\n\n/**\n * @param value A value that is expected to be truthy\n * @param message The error message if the value is falsy\n * @throws {Error} if value is not truthy\n */\nfunction assert(value: unknown, message: string): asserts value {\n  if (!value) {\n    throw new Error(message);\n  }\n}\n"]}
|
package/package.json
ADDED
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@cubist-labs/cubesigner-sdk-viem",
|
|
3
|
+
"repository": {
|
|
4
|
+
"type": "git",
|
|
5
|
+
"url": "git+https://github.com/cubist-labs/CubeSigner-TypeScript-SDK.git",
|
|
6
|
+
"directory": "packages/viem"
|
|
7
|
+
},
|
|
8
|
+
"version": "0.4.183-0",
|
|
9
|
+
"type": "module",
|
|
10
|
+
"description": "Viem CustomSource implementation",
|
|
11
|
+
"license": "MIT OR Apache-2.0",
|
|
12
|
+
"author": "Cubist, Inc.",
|
|
13
|
+
"main": "dist/index.js",
|
|
14
|
+
"types": "dist/index.d.ts",
|
|
15
|
+
"files": [
|
|
16
|
+
"tsconfig.json",
|
|
17
|
+
"src/**",
|
|
18
|
+
"dist/**",
|
|
19
|
+
"../../NOTICE",
|
|
20
|
+
"../../LICENSE-APACHE",
|
|
21
|
+
"../../LICENSE-MIT"
|
|
22
|
+
],
|
|
23
|
+
"scripts": {
|
|
24
|
+
"build": "tsc",
|
|
25
|
+
"prepack": "tsc",
|
|
26
|
+
"test": "jest --maxWorkers=1"
|
|
27
|
+
},
|
|
28
|
+
"directories": {
|
|
29
|
+
"test": "test"
|
|
30
|
+
},
|
|
31
|
+
"peerDependencies": {
|
|
32
|
+
"@cubist-labs/cubesigner-sdk": "^0.4.183-0",
|
|
33
|
+
"viem": "^2.38.2"
|
|
34
|
+
},
|
|
35
|
+
"devDependencies": {
|
|
36
|
+
"@cubist-labs/cubesigner-sdk-fs-storage": "^0.4.183-0"
|
|
37
|
+
}
|
|
38
|
+
}
|
package/src/index.ts
ADDED
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
import {
|
|
2
|
+
type Address,
|
|
3
|
+
type CustomSource,
|
|
4
|
+
type Hex,
|
|
5
|
+
type Signature,
|
|
6
|
+
bytesToHex,
|
|
7
|
+
formatTransactionRequest,
|
|
8
|
+
getAddress,
|
|
9
|
+
isHex,
|
|
10
|
+
parseTransaction,
|
|
11
|
+
serializeTransaction,
|
|
12
|
+
stringToHex,
|
|
13
|
+
} from "viem";
|
|
14
|
+
import {
|
|
15
|
+
type CubeSignerClient,
|
|
16
|
+
type Eip712SignRequest,
|
|
17
|
+
type EvmSignRequest,
|
|
18
|
+
EvmSigner,
|
|
19
|
+
type EvmSignerOptions,
|
|
20
|
+
type Key,
|
|
21
|
+
} from "@cubist-labs/cubesigner-sdk";
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* A class to wrap a CubeSigner key and client into a Viem {@link CustomSource}.
|
|
25
|
+
* Use Viem's `toAccount` to convert this to a Viem Account.
|
|
26
|
+
*/
|
|
27
|
+
export class CubeSignerSource implements CustomSource {
|
|
28
|
+
/** The internal CubeSigner signer used. */
|
|
29
|
+
readonly #signer: EvmSigner;
|
|
30
|
+
|
|
31
|
+
/** The address the wallet is associated with. */
|
|
32
|
+
public readonly address: Address;
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* Construct a Viem {@link CustomSource} around a CubeSigner key and client.
|
|
36
|
+
* Use Viem's `toAccount` to convert this to a Viem Account.
|
|
37
|
+
*
|
|
38
|
+
* @param address The EVM address this wallet is associated with
|
|
39
|
+
* @param client The session used for signing. Must have necessary scopes.
|
|
40
|
+
* @param options MFA options for the client to respect
|
|
41
|
+
* @throws {Error} if the address is not a valid EVM address
|
|
42
|
+
*/
|
|
43
|
+
constructor(address: Key | string, client: CubeSignerClient, options?: EvmSignerOptions) {
|
|
44
|
+
this.#signer = new EvmSigner(address, client, options);
|
|
45
|
+
|
|
46
|
+
// NOTE: `getAddress` will checksum the address and throw if it's an invalid EVM address.
|
|
47
|
+
this.address = getAddress(this.#signer.address);
|
|
48
|
+
|
|
49
|
+
// Scope these functions to properly resolve `this`
|
|
50
|
+
this.signMessage = this.signMessage.bind(this);
|
|
51
|
+
this.signTransaction = this.signTransaction.bind(this);
|
|
52
|
+
this.signTypedData = this.signTypedData.bind(this);
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Signs arbitrary messages. This uses CubeSigner's EIP-191 signing endpoint.
|
|
57
|
+
* The key (for this session) must have the `"AllowRawBlobSigning"` or
|
|
58
|
+
* `"AllowEip191Signing"` policy attached.
|
|
59
|
+
*
|
|
60
|
+
* @param root Object around the message
|
|
61
|
+
* @param root.message The message to sign
|
|
62
|
+
* @returns The signature
|
|
63
|
+
*/
|
|
64
|
+
public signMessage: CustomSource["signMessage"] = async ({ message }) => {
|
|
65
|
+
let hex;
|
|
66
|
+
if (typeof message === "string") {
|
|
67
|
+
hex = stringToHex(message);
|
|
68
|
+
} else if (isHex(message.raw)) {
|
|
69
|
+
hex = message.raw;
|
|
70
|
+
} else {
|
|
71
|
+
hex = bytesToHex(message.raw);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
const signature = await this.#signer.signEip191({ data: hex });
|
|
75
|
+
|
|
76
|
+
return ensureHex(signature);
|
|
77
|
+
};
|
|
78
|
+
|
|
79
|
+
/**
|
|
80
|
+
* Signs EIP-712 typed data. This uses CubeSigner's EIP-712 signing endpoint.
|
|
81
|
+
* The key (for this session) must have the `"AllowRawBlobSigning"` or
|
|
82
|
+
* `"AllowEip712Signing"` policy attached.
|
|
83
|
+
* `chainId` must be specified within the `domain`.
|
|
84
|
+
*
|
|
85
|
+
* @param parameters Typed data
|
|
86
|
+
* @returns signature for the typed data
|
|
87
|
+
*/
|
|
88
|
+
public signTypedData: CustomSource["signTypedData"] = async (parameters) => {
|
|
89
|
+
assert(parameters.domain, "`domain` must be defined");
|
|
90
|
+
|
|
91
|
+
const castedParameters = parameters as Eip712SignRequest["typed_data"];
|
|
92
|
+
assert(castedParameters.domain.chainId, "`domain.chainId` must be defined");
|
|
93
|
+
|
|
94
|
+
const signature = await this.#signer.signEip712({
|
|
95
|
+
chain_id: Number(castedParameters.domain.chainId),
|
|
96
|
+
typed_data: castedParameters,
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
return ensureHex(signature);
|
|
100
|
+
};
|
|
101
|
+
|
|
102
|
+
/**
|
|
103
|
+
* It is recommended to use `prepareTransactionRequest` on your request
|
|
104
|
+
* before calling this function.
|
|
105
|
+
*
|
|
106
|
+
* Sign a transaction. This method will block if the key requires MFA approval.
|
|
107
|
+
* `type` and `chainId` must be defined. Only supports type "legacy" or "eip1559".
|
|
108
|
+
*
|
|
109
|
+
* @param transaction The transaction to sign
|
|
110
|
+
* @param options Contains an optional custom serializer
|
|
111
|
+
* @returns Signed transaction
|
|
112
|
+
* @throws {Error} if transaction.type isn't "legacy" or "eip1559", or if "chainId" isn't specified
|
|
113
|
+
*/
|
|
114
|
+
public signTransaction: CustomSource["signTransaction"] = async (transaction, options) => {
|
|
115
|
+
assert(
|
|
116
|
+
transaction.type === "legacy" || transaction.type === "eip1559",
|
|
117
|
+
`Unsupported transaction type '${transaction.type}', CubeSigner only supports type 'legacy' or 'eip1559'")}'`,
|
|
118
|
+
);
|
|
119
|
+
|
|
120
|
+
assert(transaction.chainId, "`chainId` must be defined");
|
|
121
|
+
|
|
122
|
+
const formatted = formatTransactionRequest(transaction);
|
|
123
|
+
const rlpSignedTransaction = await this.#signer.signTransaction({
|
|
124
|
+
chain_id: transaction.chainId,
|
|
125
|
+
tx: formatted as EvmSignRequest["tx"],
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
// CubeSigner returns an RLP-encoded transaction. Since Viem allows users to pass a custom serializer, we will
|
|
129
|
+
// now unserialize and reserialize with Viem's serializer.
|
|
130
|
+
const { r, s, v, yParity, ...parsedTransaction } = parseTransaction(
|
|
131
|
+
ensureHex(rlpSignedTransaction),
|
|
132
|
+
);
|
|
133
|
+
const serializer = options?.serializer ?? serializeTransaction;
|
|
134
|
+
|
|
135
|
+
return await serializer(parsedTransaction, { r, s, v, yParity } as Signature);
|
|
136
|
+
};
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
/**
|
|
140
|
+
* @param input A hex string
|
|
141
|
+
* @returns the input, type narrowed to Hex
|
|
142
|
+
* @throws {Error} if input is not Hex
|
|
143
|
+
*/
|
|
144
|
+
function ensureHex(input: string): Hex {
|
|
145
|
+
assert(isHex(input), `${input} is not hex`);
|
|
146
|
+
return input;
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
/**
|
|
150
|
+
* @param value A value that is expected to be truthy
|
|
151
|
+
* @param message The error message if the value is falsy
|
|
152
|
+
* @throws {Error} if value is not truthy
|
|
153
|
+
*/
|
|
154
|
+
function assert(value: unknown, message: string): asserts value {
|
|
155
|
+
if (!value) {
|
|
156
|
+
throw new Error(message);
|
|
157
|
+
}
|
|
158
|
+
}
|