@metamask/smart-accounts-kit 0.1.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/CHANGELOG.md +17 -0
- package/LICENSE.APACHE2 +201 -0
- package/LICENSE.MIT0 +16 -0
- package/README.md +77 -0
- package/dist/actions/index.cjs +39 -0
- package/dist/actions/index.cjs.map +1 -0
- package/dist/actions/index.d.cts +8 -0
- package/dist/actions/index.d.ts +8 -0
- package/dist/actions/index.mjs +39 -0
- package/dist/actions/index.mjs.map +1 -0
- package/dist/chunk-2EIX7ZYT.cjs +1325 -0
- package/dist/chunk-2EIX7ZYT.cjs.map +1 -0
- package/dist/chunk-662KDAAG.cjs +831 -0
- package/dist/chunk-662KDAAG.cjs.map +1 -0
- package/dist/chunk-6RN5RBOR.mjs +1325 -0
- package/dist/chunk-6RN5RBOR.mjs.map +1 -0
- package/dist/chunk-7MRTROLV.mjs +274 -0
- package/dist/chunk-7MRTROLV.mjs.map +1 -0
- package/dist/chunk-BFDBNFIP.cjs +336 -0
- package/dist/chunk-BFDBNFIP.cjs.map +1 -0
- package/dist/chunk-CPLIK3VF.mjs +77 -0
- package/dist/chunk-CPLIK3VF.mjs.map +1 -0
- package/dist/chunk-D4ILRL6Z.cjs +299 -0
- package/dist/chunk-D4ILRL6Z.cjs.map +1 -0
- package/dist/chunk-HD4O6IVC.cjs +274 -0
- package/dist/chunk-HD4O6IVC.cjs.map +1 -0
- package/dist/chunk-HLCOMYYU.cjs +1472 -0
- package/dist/chunk-HLCOMYYU.cjs.map +1 -0
- package/dist/chunk-KUEXI4ME.mjs +299 -0
- package/dist/chunk-KUEXI4ME.mjs.map +1 -0
- package/dist/chunk-MVKT5CLQ.mjs +336 -0
- package/dist/chunk-MVKT5CLQ.mjs.map +1 -0
- package/dist/chunk-RB4SOJM3.mjs +729 -0
- package/dist/chunk-RB4SOJM3.mjs.map +1 -0
- package/dist/chunk-T6PSFUOZ.cjs +77 -0
- package/dist/chunk-T6PSFUOZ.cjs.map +1 -0
- package/dist/chunk-TEI2NRPU.mjs +831 -0
- package/dist/chunk-TEI2NRPU.mjs.map +1 -0
- package/dist/chunk-UUDQWENY.mjs +1472 -0
- package/dist/chunk-UUDQWENY.mjs.map +1 -0
- package/dist/chunk-WA2O2K3K.cjs +729 -0
- package/dist/chunk-WA2O2K3K.cjs.map +1 -0
- package/dist/contracts/index.cjs +57 -0
- package/dist/contracts/index.cjs.map +1 -0
- package/dist/contracts/index.d.cts +8 -0
- package/dist/contracts/index.d.ts +8 -0
- package/dist/contracts/index.mjs +57 -0
- package/dist/contracts/index.mjs.map +1 -0
- package/dist/delegation-7PtFix8Y.d.ts +376 -0
- package/dist/delegation-d52Owevd.d.cts +376 -0
- package/dist/experimental/index.cjs +143 -0
- package/dist/experimental/index.cjs.map +1 -0
- package/dist/experimental/index.d.cts +29 -0
- package/dist/experimental/index.d.ts +29 -0
- package/dist/experimental/index.mjs +143 -0
- package/dist/experimental/index.mjs.map +1 -0
- package/dist/index-0-B0YlEP.d.cts +21935 -0
- package/dist/index-18Alar1P.d.ts +1174 -0
- package/dist/index-CLkk7zr4.d.cts +1174 -0
- package/dist/index-D1WbhKCc.d.ts +21935 -0
- package/dist/index.cjs +424 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +56 -0
- package/dist/index.d.ts +56 -0
- package/dist/index.mjs +424 -0
- package/dist/index.mjs.map +1 -0
- package/dist/smartAccountsEnvironment-BOhrxEnt.d.ts +12 -0
- package/dist/smartAccountsEnvironment-SVknZ_3f.d.cts +12 -0
- package/dist/types-Bwksz_U6.d.cts +136 -0
- package/dist/types-Bwksz_U6.d.ts +136 -0
- package/dist/userOp-CFv4wNkl.d.cts +33 -0
- package/dist/userOp-CFv4wNkl.d.ts +33 -0
- package/dist/utils/index.cjs +60 -0
- package/dist/utils/index.cjs.map +1 -0
- package/dist/utils/index.d.cts +47 -0
- package/dist/utils/index.d.ts +47 -0
- package/dist/utils/index.mjs +60 -0
- package/dist/utils/index.mjs.map +1 -0
- package/package.json +151 -0
|
@@ -0,0 +1,143 @@
|
|
|
1
|
+
import {
|
|
2
|
+
getDelegationHashOffchain
|
|
3
|
+
} from "../chunk-UUDQWENY.mjs";
|
|
4
|
+
|
|
5
|
+
// src/experimental/delegationStorage.ts
|
|
6
|
+
import { toHex } from "viem";
|
|
7
|
+
var DelegationStorageClient = class {
|
|
8
|
+
#apiVersionPrefix = "api/v0";
|
|
9
|
+
#config;
|
|
10
|
+
#fetcher;
|
|
11
|
+
#apiUrl;
|
|
12
|
+
constructor(config) {
|
|
13
|
+
const { apiUrl } = config.environment;
|
|
14
|
+
if (apiUrl.endsWith(this.#apiVersionPrefix)) {
|
|
15
|
+
this.#apiUrl = apiUrl;
|
|
16
|
+
} else {
|
|
17
|
+
const separator = apiUrl.endsWith("/") ? "" : "/";
|
|
18
|
+
this.#apiUrl = `${apiUrl}${separator}${this.#apiVersionPrefix}`;
|
|
19
|
+
}
|
|
20
|
+
this.#fetcher = this.#initializeFetcher(config);
|
|
21
|
+
this.#config = config;
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Initializes the fetch function for HTTP requests.
|
|
25
|
+
*
|
|
26
|
+
* - Uses `config.fetcher` if provided.
|
|
27
|
+
* - Falls back to global `fetch` if available.
|
|
28
|
+
* - Throws an error if no fetch function is available.
|
|
29
|
+
*
|
|
30
|
+
* @param config - Configuration object that may include a custom fetch function.
|
|
31
|
+
* @returns The fetch function to be used for HTTP requests.
|
|
32
|
+
* @throws Error if no fetch function is available in the environment.
|
|
33
|
+
*/
|
|
34
|
+
#initializeFetcher(config) {
|
|
35
|
+
if (config.fetcher) {
|
|
36
|
+
return config.fetcher;
|
|
37
|
+
} else if (typeof globalThis?.fetch === "function") {
|
|
38
|
+
return globalThis.fetch.bind(globalThis);
|
|
39
|
+
}
|
|
40
|
+
throw new Error(
|
|
41
|
+
"Fetch API is not available in this environment. Please provide a fetch function in the config."
|
|
42
|
+
);
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Fetches the delegation chain from the Delegation Storage Service, ending with
|
|
46
|
+
* the specified leaf delegation.
|
|
47
|
+
*
|
|
48
|
+
* @param leafDelegationOrDelegationHash - The leaf delegation, or the hash
|
|
49
|
+
* of the leaf delegation.
|
|
50
|
+
* @returns A promise that resolves to the delegation chain - empty array if the delegation
|
|
51
|
+
* is not found.
|
|
52
|
+
*/
|
|
53
|
+
async getDelegationChain(leafDelegationOrDelegationHash) {
|
|
54
|
+
const leafDelegationHash = typeof leafDelegationOrDelegationHash === "string" ? leafDelegationOrDelegationHash : getDelegationHashOffchain(leafDelegationOrDelegationHash);
|
|
55
|
+
const response = await this.#fetcher(
|
|
56
|
+
`${this.#apiUrl}/delegation/chain/${leafDelegationHash}`,
|
|
57
|
+
{
|
|
58
|
+
method: "GET",
|
|
59
|
+
headers: {
|
|
60
|
+
Authorization: `Bearer ${this.#config.apiKey}`,
|
|
61
|
+
"x-api-key-id": this.#config.apiKeyId
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
);
|
|
65
|
+
const responseData = await response.json();
|
|
66
|
+
if ("error" in responseData) {
|
|
67
|
+
throw new Error(
|
|
68
|
+
`Failed to fetch delegation chain: ${responseData.error}`
|
|
69
|
+
);
|
|
70
|
+
}
|
|
71
|
+
return responseData;
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Fetches the delegations from the Delegation Storage Service, either `Received`
|
|
75
|
+
* by, or `Given` by, (or both: `All`) the specified deleGatorAddress. Defaults
|
|
76
|
+
* to `Received`.
|
|
77
|
+
*
|
|
78
|
+
* @param deleGatorAddress - The deleGatorAddress to retrieve the delegations for.
|
|
79
|
+
* @param filterMode - The DelegationStoreFilter mode - defaults to Received.
|
|
80
|
+
* @returns A promise that resolves to the list of delegations received by the deleGatorAddress,
|
|
81
|
+
* empty array if the delegations are not found.
|
|
82
|
+
*/
|
|
83
|
+
async fetchDelegations(deleGatorAddress, filterMode = "RECEIVED" /* Received */) {
|
|
84
|
+
const response = await this.#fetcher(
|
|
85
|
+
`${this.#apiUrl}/delegation/accounts/${deleGatorAddress}?filter=${filterMode}`,
|
|
86
|
+
{
|
|
87
|
+
method: "GET",
|
|
88
|
+
headers: {
|
|
89
|
+
Authorization: `Bearer ${this.#config.apiKey}`,
|
|
90
|
+
"x-api-key-id": this.#config.apiKeyId
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
);
|
|
94
|
+
const responseData = await response.json();
|
|
95
|
+
if ("error" in responseData) {
|
|
96
|
+
throw new Error(`Failed to fetch delegations: ${responseData.error}`);
|
|
97
|
+
}
|
|
98
|
+
return responseData;
|
|
99
|
+
}
|
|
100
|
+
/**
|
|
101
|
+
* Stores the specified delegation in the Delegation Storage Service.
|
|
102
|
+
*
|
|
103
|
+
* @param delegation - The delegation to store.
|
|
104
|
+
* @returns A promise that resolves to the delegation hash indicating successful storage.
|
|
105
|
+
*/
|
|
106
|
+
async storeDelegation(delegation) {
|
|
107
|
+
if (!delegation.signature || delegation.signature === "0x") {
|
|
108
|
+
throw new Error("Delegation must be signed to be stored");
|
|
109
|
+
}
|
|
110
|
+
const delegationHash = getDelegationHashOffchain(delegation);
|
|
111
|
+
const body = JSON.stringify(
|
|
112
|
+
{
|
|
113
|
+
...delegation,
|
|
114
|
+
metadata: []
|
|
115
|
+
},
|
|
116
|
+
(_, value) => typeof value === "bigint" || typeof value === "number" ? toHex(value) : value,
|
|
117
|
+
2
|
|
118
|
+
);
|
|
119
|
+
const response = await this.#fetcher(`${this.#apiUrl}/delegation/store`, {
|
|
120
|
+
method: "POST",
|
|
121
|
+
headers: {
|
|
122
|
+
Authorization: `Bearer ${this.#config.apiKey}`,
|
|
123
|
+
"x-api-key-id": this.#config.apiKeyId,
|
|
124
|
+
"Content-Type": "application/json"
|
|
125
|
+
},
|
|
126
|
+
body
|
|
127
|
+
});
|
|
128
|
+
const responseData = await response.json();
|
|
129
|
+
if ("error" in responseData) {
|
|
130
|
+
throw new Error(responseData.error);
|
|
131
|
+
}
|
|
132
|
+
if (responseData.delegationHash !== delegationHash) {
|
|
133
|
+
throw Error(
|
|
134
|
+
"Failed to store the Delegation, the hash returned from the MM delegation storage API does not match the hash of the delegation"
|
|
135
|
+
);
|
|
136
|
+
}
|
|
137
|
+
return responseData.delegationHash;
|
|
138
|
+
}
|
|
139
|
+
};
|
|
140
|
+
export {
|
|
141
|
+
DelegationStorageClient
|
|
142
|
+
};
|
|
143
|
+
//# sourceMappingURL=index.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/experimental/delegationStorage.ts"],"sourcesContent":["import { type Hex, toHex } from 'viem';\n\nimport { getDelegationHashOffchain } from '../delegation';\nimport type { Delegation } from '../types';\n\ntype ErrorResponse = {\n error: string;\n data?: any;\n};\n\nexport type APIStoreDelegationResponse = {\n delegationHash: Hex;\n};\n\n/**\n * Represents the allowed filters when querying the data store for delegations.\n */\nexport enum DelegationStoreFilter {\n Given = 'GIVEN',\n Received = 'RECEIVED',\n All = 'ALL',\n}\n\n/**\n * Public Delegation Storage Service environments. To be used in the\n * DeleGationStorageService config.\n */\nexport const DelegationStorageEnvironment: {\n [K in 'dev' | 'prod']: Environment;\n} = {\n dev: { apiUrl: 'https://passkeys.dev-api.cx.metamask.io' },\n prod: { apiUrl: 'https://passkeys.api.cx.metamask.io' },\n};\n\nexport type Environment = {\n apiUrl: string;\n};\n\nexport type DelegationStorageConfig = {\n apiKey: string;\n apiKeyId: string;\n environment: Environment;\n fetcher?: typeof fetch;\n};\n\nexport class DelegationStorageClient {\n #apiVersionPrefix = 'api/v0';\n\n #config: DelegationStorageConfig;\n\n #fetcher: typeof fetch;\n\n #apiUrl: string;\n\n constructor(config: DelegationStorageConfig) {\n const { apiUrl } = config.environment;\n\n if (apiUrl.endsWith(this.#apiVersionPrefix)) {\n this.#apiUrl = apiUrl;\n } else {\n const separator = apiUrl.endsWith('/') ? '' : '/';\n this.#apiUrl = `${apiUrl}${separator}${this.#apiVersionPrefix}`;\n }\n this.#fetcher = this.#initializeFetcher(config);\n this.#config = config;\n }\n\n /**\n * Initializes the fetch function for HTTP requests.\n *\n * - Uses `config.fetcher` if provided.\n * - Falls back to global `fetch` if available.\n * - Throws an error if no fetch function is available.\n *\n * @param config - Configuration object that may include a custom fetch function.\n * @returns The fetch function to be used for HTTP requests.\n * @throws Error if no fetch function is available in the environment.\n */\n #initializeFetcher(config: DelegationStorageConfig): typeof fetch {\n if (config.fetcher) {\n return config.fetcher;\n } else if (typeof globalThis?.fetch === 'function') {\n return globalThis.fetch.bind(globalThis);\n }\n throw new Error(\n 'Fetch API is not available in this environment. Please provide a fetch function in the config.',\n );\n }\n\n /**\n * Fetches the delegation chain from the Delegation Storage Service, ending with\n * the specified leaf delegation.\n *\n * @param leafDelegationOrDelegationHash - The leaf delegation, or the hash\n * of the leaf delegation.\n * @returns A promise that resolves to the delegation chain - empty array if the delegation\n * is not found.\n */\n async getDelegationChain(\n leafDelegationOrDelegationHash: Hex | Delegation,\n ): Promise<Delegation[]> {\n const leafDelegationHash =\n typeof leafDelegationOrDelegationHash === 'string'\n ? leafDelegationOrDelegationHash\n : getDelegationHashOffchain(leafDelegationOrDelegationHash);\n\n const response = await this.#fetcher(\n `${this.#apiUrl}/delegation/chain/${leafDelegationHash}`,\n {\n method: 'GET',\n headers: {\n Authorization: `Bearer ${this.#config.apiKey}`,\n 'x-api-key-id': this.#config.apiKeyId,\n },\n },\n );\n\n const responseData: Delegation[] | ErrorResponse = await response.json();\n\n if ('error' in responseData) {\n throw new Error(\n `Failed to fetch delegation chain: ${responseData.error}`,\n );\n }\n\n return responseData;\n }\n\n /**\n * Fetches the delegations from the Delegation Storage Service, either `Received`\n * by, or `Given` by, (or both: `All`) the specified deleGatorAddress. Defaults\n * to `Received`.\n *\n * @param deleGatorAddress - The deleGatorAddress to retrieve the delegations for.\n * @param filterMode - The DelegationStoreFilter mode - defaults to Received.\n * @returns A promise that resolves to the list of delegations received by the deleGatorAddress,\n * empty array if the delegations are not found.\n */\n async fetchDelegations(\n deleGatorAddress: Hex,\n filterMode = DelegationStoreFilter.Received,\n ): Promise<Delegation[]> {\n const response = await this.#fetcher(\n `${this.#apiUrl}/delegation/accounts/${deleGatorAddress}?filter=${filterMode}`,\n {\n method: 'GET',\n headers: {\n Authorization: `Bearer ${this.#config.apiKey}`,\n 'x-api-key-id': this.#config.apiKeyId,\n },\n },\n );\n\n const responseData: Delegation[] | ErrorResponse = await response.json();\n\n if ('error' in responseData) {\n throw new Error(`Failed to fetch delegations: ${responseData.error}`);\n }\n\n return responseData;\n }\n\n /**\n * Stores the specified delegation in the Delegation Storage Service.\n *\n * @param delegation - The delegation to store.\n * @returns A promise that resolves to the delegation hash indicating successful storage.\n */\n async storeDelegation(delegation: Delegation): Promise<Hex> {\n if (!delegation.signature || delegation.signature === '0x') {\n throw new Error('Delegation must be signed to be stored');\n }\n\n const delegationHash = getDelegationHashOffchain(delegation);\n\n const body = JSON.stringify(\n {\n ...delegation,\n metadata: [],\n },\n (_, value: any) =>\n typeof value === 'bigint' || typeof value === 'number'\n ? toHex(value)\n : value,\n 2,\n );\n\n const response = await this.#fetcher(`${this.#apiUrl}/delegation/store`, {\n method: 'POST',\n headers: {\n Authorization: `Bearer ${this.#config.apiKey}`,\n 'x-api-key-id': this.#config.apiKeyId,\n 'Content-Type': 'application/json',\n },\n body,\n });\n\n const responseData: APIStoreDelegationResponse | ErrorResponse =\n await response.json();\n\n if ('error' in responseData) {\n throw new Error(responseData.error);\n }\n\n if (responseData.delegationHash !== delegationHash) {\n throw Error(\n 'Failed to store the Delegation, the hash returned from the MM delegation storage API does not match the hash of the delegation',\n );\n }\n\n return responseData.delegationHash;\n }\n}\n"],"mappings":"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","names":[]}
|