@t402/stacks 2.4.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.
Files changed (40) hide show
  1. package/README.md +178 -0
  2. package/dist/exact-direct/client/index.cjs +167 -0
  3. package/dist/exact-direct/client/index.cjs.map +1 -0
  4. package/dist/exact-direct/client/index.d.cts +39 -0
  5. package/dist/exact-direct/client/index.d.ts +39 -0
  6. package/dist/exact-direct/client/index.mjs +139 -0
  7. package/dist/exact-direct/client/index.mjs.map +1 -0
  8. package/dist/exact-direct/facilitator/index.cjs +395 -0
  9. package/dist/exact-direct/facilitator/index.cjs.map +1 -0
  10. package/dist/exact-direct/facilitator/index.d.cts +55 -0
  11. package/dist/exact-direct/facilitator/index.d.ts +55 -0
  12. package/dist/exact-direct/facilitator/index.mjs +367 -0
  13. package/dist/exact-direct/facilitator/index.mjs.map +1 -0
  14. package/dist/exact-direct/server/index.cjs +247 -0
  15. package/dist/exact-direct/server/index.cjs.map +1 -0
  16. package/dist/exact-direct/server/index.d.cts +109 -0
  17. package/dist/exact-direct/server/index.d.ts +109 -0
  18. package/dist/exact-direct/server/index.mjs +218 -0
  19. package/dist/exact-direct/server/index.mjs.map +1 -0
  20. package/dist/index.cjs +261 -0
  21. package/dist/index.cjs.map +1 -0
  22. package/dist/index.d.cts +126 -0
  23. package/dist/index.d.ts +126 -0
  24. package/dist/index.mjs +212 -0
  25. package/dist/index.mjs.map +1 -0
  26. package/dist/types-Bxzo3eQ1.d.cts +172 -0
  27. package/dist/types-Bxzo3eQ1.d.ts +172 -0
  28. package/package.json +102 -0
  29. package/src/constants.ts +66 -0
  30. package/src/exact-direct/client/index.ts +5 -0
  31. package/src/exact-direct/client/scheme.ts +115 -0
  32. package/src/exact-direct/facilitator/index.ts +4 -0
  33. package/src/exact-direct/facilitator/scheme.ts +308 -0
  34. package/src/exact-direct/server/index.ts +9 -0
  35. package/src/exact-direct/server/register.ts +57 -0
  36. package/src/exact-direct/server/scheme.ts +216 -0
  37. package/src/index.ts +78 -0
  38. package/src/tokens.ts +96 -0
  39. package/src/types.ts +184 -0
  40. package/src/utils.ts +198 -0
@@ -0,0 +1,367 @@
1
+ // src/constants.ts
2
+ var STACKS_CAIP2_NAMESPACE = "stacks";
3
+ var STACKS_MAINNET_CAIP2 = "stacks:1";
4
+ var STACKS_TESTNET_CAIP2 = "stacks:2147483648";
5
+ var SCHEME_EXACT_DIRECT = "exact-direct";
6
+ var DEFAULT_MAINNET_API = "https://api.mainnet.hiro.so";
7
+ var DEFAULT_TESTNET_API = "https://api.testnet.hiro.so";
8
+ var STACKS_NETWORKS = {
9
+ [STACKS_MAINNET_CAIP2]: {
10
+ name: "Stacks Mainnet",
11
+ caip2: STACKS_MAINNET_CAIP2,
12
+ apiUrl: DEFAULT_MAINNET_API,
13
+ chainId: 1,
14
+ addressPrefix: "SP",
15
+ isTestnet: false
16
+ },
17
+ [STACKS_TESTNET_CAIP2]: {
18
+ name: "Stacks Testnet",
19
+ caip2: STACKS_TESTNET_CAIP2,
20
+ apiUrl: DEFAULT_TESTNET_API,
21
+ chainId: 2147483648,
22
+ addressPrefix: "ST",
23
+ isTestnet: true
24
+ }
25
+ };
26
+ function getNetworkConfig(network) {
27
+ return STACKS_NETWORKS[network];
28
+ }
29
+
30
+ // src/tokens.ts
31
+ var SUSDC_MAINNET = {
32
+ contractAddress: "SP3Y2ZSH8P7D50B0VBTSX11S7XSG24M1VB9YFQA4K.token-susdc",
33
+ symbol: "sUSDC",
34
+ name: "Stacks USDC",
35
+ decimals: 6,
36
+ issuer: "Stacks"
37
+ };
38
+ var SUSDC_TESTNET = {
39
+ contractAddress: "ST1PQHQKV0RJXZFY1DGX8MNSNYVE3VGZJSRTPGZGM.token-susdc",
40
+ symbol: "sUSDC",
41
+ name: "Test Stacks USDC",
42
+ decimals: 6
43
+ };
44
+ var TOKEN_REGISTRY = {
45
+ [STACKS_MAINNET_CAIP2]: {
46
+ sUSDC: SUSDC_MAINNET
47
+ },
48
+ [STACKS_TESTNET_CAIP2]: {
49
+ sUSDC: SUSDC_TESTNET
50
+ }
51
+ };
52
+ var DEFAULT_TOKENS = {
53
+ [STACKS_MAINNET_CAIP2]: SUSDC_MAINNET,
54
+ [STACKS_TESTNET_CAIP2]: SUSDC_TESTNET
55
+ };
56
+ function getDefaultToken(network) {
57
+ return DEFAULT_TOKENS[network];
58
+ }
59
+
60
+ // src/utils.ts
61
+ function isValidPrincipal(address) {
62
+ if (!address || typeof address !== "string") {
63
+ return false;
64
+ }
65
+ const parts = address.split(".");
66
+ const principal = parts[0];
67
+ const principalRegex = /^(SP|ST)[0-9A-HJ-NP-Za-km-z]{33,41}$/;
68
+ if (!principalRegex.test(principal)) {
69
+ return false;
70
+ }
71
+ if (parts.length === 2) {
72
+ const contractName = parts[1];
73
+ const contractNameRegex = /^[a-zA-Z][a-zA-Z0-9\-_]{0,127}$/;
74
+ return contractNameRegex.test(contractName);
75
+ }
76
+ return parts.length === 1;
77
+ }
78
+ function isValidTxId(hash) {
79
+ if (!hash || typeof hash !== "string") {
80
+ return false;
81
+ }
82
+ return /^0x[a-fA-F0-9]{64}$/.test(hash);
83
+ }
84
+ function comparePrincipals(a, b) {
85
+ return a === b;
86
+ }
87
+ function extractTokenTransfer(result, contractAddress) {
88
+ if (result.txStatus !== "success") {
89
+ return null;
90
+ }
91
+ if (result.txType !== "contract_call") {
92
+ return null;
93
+ }
94
+ if (result.contractCall) {
95
+ const { contractId, functionName } = result.contractCall;
96
+ if (functionName !== "transfer") {
97
+ return null;
98
+ }
99
+ if (contractAddress && contractId !== contractAddress) {
100
+ return null;
101
+ }
102
+ const transferEvent = result.events.find(
103
+ (e) => e.eventType === "fungible_token_asset" && e.asset?.assetEventType === "transfer"
104
+ );
105
+ if (transferEvent?.asset) {
106
+ return {
107
+ contractAddress: contractId,
108
+ from: transferEvent.asset.sender,
109
+ to: transferEvent.asset.recipient,
110
+ amount: transferEvent.asset.amount,
111
+ success: true
112
+ };
113
+ }
114
+ if (result.contractCall.functionArgs.length >= 3) {
115
+ const amountArg = result.contractCall.functionArgs[0];
116
+ const senderArg = result.contractCall.functionArgs[1];
117
+ const recipientArg = result.contractCall.functionArgs[2];
118
+ const senderMatch = senderArg?.repr?.match(/^'?(S[PT][0-9A-HJ-NP-Za-km-z]+)/);
119
+ const recipientMatch = recipientArg?.repr?.match(/^'?(S[PT][0-9A-HJ-NP-Za-km-z]+)/);
120
+ const amountMatch = amountArg?.repr?.match(/^u(\d+)$/);
121
+ if (senderMatch && recipientMatch && amountMatch) {
122
+ return {
123
+ contractAddress: contractId,
124
+ from: senderMatch[1],
125
+ to: recipientMatch[1],
126
+ amount: amountMatch[1],
127
+ success: true
128
+ };
129
+ }
130
+ }
131
+ }
132
+ return null;
133
+ }
134
+ function extractTokenTransferFromPostConditions(result, contractAddress) {
135
+ if (result.txStatus !== "success") {
136
+ return null;
137
+ }
138
+ for (const pc of result.postConditions) {
139
+ if (pc.asset) {
140
+ const assetContractAddress = `${pc.asset.contractAddress}.${pc.asset.contractName}`;
141
+ if (contractAddress && assetContractAddress !== contractAddress) {
142
+ continue;
143
+ }
144
+ const transferEvent = result.events.find(
145
+ (e) => e.eventType === "fungible_token_asset" && e.asset?.assetEventType === "transfer" && e.asset?.sender === pc.principal.address
146
+ );
147
+ if (transferEvent?.asset) {
148
+ return {
149
+ contractAddress: assetContractAddress,
150
+ from: transferEvent.asset.sender,
151
+ to: transferEvent.asset.recipient,
152
+ amount: transferEvent.asset.amount,
153
+ success: true
154
+ };
155
+ }
156
+ }
157
+ }
158
+ return null;
159
+ }
160
+
161
+ // src/exact-direct/facilitator/scheme.ts
162
+ var DEFAULT_MAX_TRANSACTION_AGE = 3600;
163
+ var DEFAULT_CACHE_DURATION = 24 * 60 * 60 * 1e3;
164
+ var ExactDirectStacksFacilitator = class {
165
+ scheme = SCHEME_EXACT_DIRECT;
166
+ caipFamily = `${STACKS_CAIP2_NAMESPACE}:*`;
167
+ signer;
168
+ config;
169
+ usedTransactions = /* @__PURE__ */ new Map();
170
+ constructor(signer, config = {}) {
171
+ this.signer = signer;
172
+ this.config = {
173
+ maxTransactionAge: config.maxTransactionAge ?? DEFAULT_MAX_TRANSACTION_AGE,
174
+ usedTxCacheDuration: config.usedTxCacheDuration ?? DEFAULT_CACHE_DURATION
175
+ };
176
+ this.startCleanupInterval();
177
+ }
178
+ /**
179
+ * Get extra data for payment requirements
180
+ */
181
+ getExtra(network) {
182
+ const config = getNetworkConfig(network);
183
+ if (!config) return void 0;
184
+ const token = getDefaultToken(network);
185
+ return {
186
+ contractAddress: token?.contractAddress,
187
+ assetSymbol: token?.symbol,
188
+ assetDecimals: token?.decimals,
189
+ networkName: config.name
190
+ };
191
+ }
192
+ /**
193
+ * Get facilitator signer addresses for a network
194
+ */
195
+ getSigners(network) {
196
+ return this.signer.getAddresses(network);
197
+ }
198
+ /**
199
+ * Verify a payment payload
200
+ */
201
+ async verify(payload, requirements) {
202
+ const network = requirements.network;
203
+ if (payload.accepted.scheme !== SCHEME_EXACT_DIRECT) {
204
+ return {
205
+ isValid: false,
206
+ invalidReason: `invalid_scheme: expected ${SCHEME_EXACT_DIRECT}, got ${payload.accepted.scheme}`
207
+ };
208
+ }
209
+ if (payload.accepted.network !== network) {
210
+ return {
211
+ isValid: false,
212
+ invalidReason: `network_mismatch: expected ${network}, got ${payload.accepted.network}`
213
+ };
214
+ }
215
+ const stacksPayload = payload.payload;
216
+ if (!stacksPayload.txId) {
217
+ return {
218
+ isValid: false,
219
+ invalidReason: "missing_tx_id"
220
+ };
221
+ }
222
+ if (!isValidTxId(stacksPayload.txId)) {
223
+ return {
224
+ isValid: false,
225
+ invalidReason: "invalid_tx_id_format"
226
+ };
227
+ }
228
+ if (!stacksPayload.from) {
229
+ return {
230
+ isValid: false,
231
+ invalidReason: "missing_from_address"
232
+ };
233
+ }
234
+ if (!isValidPrincipal(stacksPayload.from)) {
235
+ return {
236
+ isValid: false,
237
+ invalidReason: "invalid_from_address",
238
+ payer: stacksPayload.from
239
+ };
240
+ }
241
+ if (this.isTransactionUsed(stacksPayload.txId)) {
242
+ return {
243
+ isValid: false,
244
+ invalidReason: "transaction_already_used",
245
+ payer: stacksPayload.from
246
+ };
247
+ }
248
+ const txResult = await this.signer.queryTransaction(stacksPayload.txId);
249
+ if (!txResult) {
250
+ return {
251
+ isValid: false,
252
+ invalidReason: "transaction_not_found",
253
+ payer: stacksPayload.from
254
+ };
255
+ }
256
+ if (txResult.txStatus !== "success") {
257
+ return {
258
+ isValid: false,
259
+ invalidReason: `transaction_failed: status=${txResult.txStatus}`,
260
+ payer: stacksPayload.from
261
+ };
262
+ }
263
+ if (this.config.maxTransactionAge > 0) {
264
+ const txTime = txResult.burnBlockTime * 1e3;
265
+ const age = (Date.now() - txTime) / 1e3;
266
+ if (age > this.config.maxTransactionAge) {
267
+ return {
268
+ isValid: false,
269
+ invalidReason: `transaction_too_old: ${Math.round(age)} seconds`,
270
+ payer: stacksPayload.from
271
+ };
272
+ }
273
+ }
274
+ const expectedContract = requirements.extra?.contractAddress ?? stacksPayload.contractAddress;
275
+ const transfer = extractTokenTransfer(txResult, expectedContract) || extractTokenTransferFromPostConditions(txResult, expectedContract);
276
+ if (!transfer) {
277
+ return {
278
+ isValid: false,
279
+ invalidReason: "not_token_transfer",
280
+ payer: stacksPayload.from
281
+ };
282
+ }
283
+ if (expectedContract && !comparePrincipals(transfer.contractAddress, expectedContract)) {
284
+ return {
285
+ isValid: false,
286
+ invalidReason: `contract_mismatch: expected ${expectedContract}, got ${transfer.contractAddress}`,
287
+ payer: stacksPayload.from
288
+ };
289
+ }
290
+ if (!comparePrincipals(transfer.to, requirements.payTo)) {
291
+ return {
292
+ isValid: false,
293
+ invalidReason: `recipient_mismatch: expected ${requirements.payTo}, got ${transfer.to}`,
294
+ payer: stacksPayload.from
295
+ };
296
+ }
297
+ const txAmount = BigInt(transfer.amount);
298
+ const requiredAmount = BigInt(requirements.amount);
299
+ if (txAmount < requiredAmount) {
300
+ return {
301
+ isValid: false,
302
+ invalidReason: `insufficient_amount: expected ${requirements.amount}, got ${transfer.amount}`,
303
+ payer: stacksPayload.from
304
+ };
305
+ }
306
+ this.markTransactionUsed(stacksPayload.txId);
307
+ return {
308
+ isValid: true,
309
+ payer: stacksPayload.from
310
+ };
311
+ }
312
+ /**
313
+ * Settle a payment (for exact-direct, the transfer is already complete)
314
+ */
315
+ async settle(payload, requirements) {
316
+ const verifyResult = await this.verify(payload, requirements);
317
+ if (!verifyResult.isValid) {
318
+ return {
319
+ success: false,
320
+ errorReason: verifyResult.invalidReason || "verification_failed",
321
+ payer: verifyResult.payer,
322
+ transaction: "",
323
+ network: requirements.network
324
+ };
325
+ }
326
+ const stacksPayload = payload.payload;
327
+ return {
328
+ success: true,
329
+ transaction: stacksPayload.txId,
330
+ network: requirements.network,
331
+ payer: verifyResult.payer
332
+ };
333
+ }
334
+ /**
335
+ * Check if a transaction has been used
336
+ */
337
+ isTransactionUsed(txId) {
338
+ return this.usedTransactions.has(txId);
339
+ }
340
+ /**
341
+ * Mark a transaction as used
342
+ */
343
+ markTransactionUsed(txId) {
344
+ this.usedTransactions.set(txId, Date.now());
345
+ }
346
+ /**
347
+ * Start the cleanup interval for used transactions cache
348
+ */
349
+ startCleanupInterval() {
350
+ setInterval(() => {
351
+ const cutoff = Date.now() - this.config.usedTxCacheDuration;
352
+ for (const [txId, timestamp] of this.usedTransactions) {
353
+ if (timestamp < cutoff) {
354
+ this.usedTransactions.delete(txId);
355
+ }
356
+ }
357
+ }, 60 * 60 * 1e3);
358
+ }
359
+ };
360
+ function createExactDirectStacksFacilitator(signer, config = {}) {
361
+ return new ExactDirectStacksFacilitator(signer, config);
362
+ }
363
+ export {
364
+ ExactDirectStacksFacilitator,
365
+ createExactDirectStacksFacilitator
366
+ };
367
+ //# sourceMappingURL=index.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/constants.ts","../../../src/tokens.ts","../../../src/utils.ts","../../../src/exact-direct/facilitator/scheme.ts"],"sourcesContent":["/**\n * Stacks T402 Constants\n *\n * Stacks is a Bitcoin Layer 2 that brings smart contracts and DeFi\n * to Bitcoin. SIP-010 is the fungible token standard on Stacks.\n */\n\n// CAIP-2 namespace for Stacks\nexport const STACKS_CAIP2_NAMESPACE = \"stacks\";\n\n// CAIP-2 network identifiers\n// Stacks Mainnet (chain ID: 1)\nexport const STACKS_MAINNET_CAIP2 = \"stacks:1\";\n\n// Stacks Testnet (chain ID: 2147483648)\nexport const STACKS_TESTNET_CAIP2 = \"stacks:2147483648\";\n\n// Scheme identifier\nexport const SCHEME_EXACT_DIRECT = \"exact-direct\";\n\n// Default API endpoints (Hiro API)\nexport const DEFAULT_MAINNET_API = \"https://api.mainnet.hiro.so\";\nexport const DEFAULT_TESTNET_API = \"https://api.testnet.hiro.so\";\n\n// Network configurations\nexport interface StacksNetworkConfig {\n readonly name: string;\n readonly caip2: string;\n readonly apiUrl: string;\n readonly chainId: number;\n readonly addressPrefix: string;\n readonly isTestnet: boolean;\n}\n\nexport const STACKS_NETWORKS: Record<string, StacksNetworkConfig> = {\n [STACKS_MAINNET_CAIP2]: {\n name: \"Stacks Mainnet\",\n caip2: STACKS_MAINNET_CAIP2,\n apiUrl: DEFAULT_MAINNET_API,\n chainId: 1,\n addressPrefix: \"SP\",\n isTestnet: false,\n },\n [STACKS_TESTNET_CAIP2]: {\n name: \"Stacks Testnet\",\n caip2: STACKS_TESTNET_CAIP2,\n apiUrl: DEFAULT_TESTNET_API,\n chainId: 2147483648,\n addressPrefix: \"ST\",\n isTestnet: true,\n },\n};\n\n/**\n * Get network configuration by CAIP-2 identifier\n */\nexport function getNetworkConfig(network: string): StacksNetworkConfig | undefined {\n return STACKS_NETWORKS[network];\n}\n\n/**\n * Check if a network identifier is a Stacks network\n */\nexport function isStacksNetwork(network: string): boolean {\n return network.startsWith(`${STACKS_CAIP2_NAMESPACE}:`);\n}\n","/**\n * Stacks Token Registry\n *\n * On Stacks, tokens are SIP-010 fungible tokens identified by\n * their contract address (principal.contract-name).\n */\n\nimport {\n STACKS_MAINNET_CAIP2,\n STACKS_TESTNET_CAIP2,\n} from \"./constants.js\";\n\n/**\n * Token configuration for Stacks SIP-010 tokens\n */\nexport interface TokenConfig {\n /** Contract address (principal.contract-name) */\n readonly contractAddress: string;\n /** Token symbol */\n readonly symbol: string;\n /** Token name */\n readonly name: string;\n /** Decimal places */\n readonly decimals: number;\n /** Token issuer */\n readonly issuer?: string;\n}\n\n/**\n * sUSDC on Stacks Mainnet\n * Contract: SP3Y2ZSH8P7D50B0VBTSX11S7XSG24M1VB9YFQA4K.token-susdc\n * Decimals: 6\n */\nexport const SUSDC_MAINNET: TokenConfig = {\n contractAddress: \"SP3Y2ZSH8P7D50B0VBTSX11S7XSG24M1VB9YFQA4K.token-susdc\",\n symbol: \"sUSDC\",\n name: \"Stacks USDC\",\n decimals: 6,\n issuer: \"Stacks\",\n};\n\n/**\n * sUSDC on Stacks Testnet\n * Contract: ST1PQHQKV0RJXZFY1DGX8MNSNYVE3VGZJSRTPGZGM.token-susdc\n * Decimals: 6\n */\nexport const SUSDC_TESTNET: TokenConfig = {\n contractAddress: \"ST1PQHQKV0RJXZFY1DGX8MNSNYVE3VGZJSRTPGZGM.token-susdc\",\n symbol: \"sUSDC\",\n name: \"Test Stacks USDC\",\n decimals: 6,\n};\n\n/**\n * Network-specific token registries\n */\nexport const TOKEN_REGISTRY: Record<string, Record<string, TokenConfig>> = {\n [STACKS_MAINNET_CAIP2]: {\n sUSDC: SUSDC_MAINNET,\n },\n [STACKS_TESTNET_CAIP2]: {\n sUSDC: SUSDC_TESTNET,\n },\n};\n\n/**\n * Default tokens per network\n */\nexport const DEFAULT_TOKENS: Record<string, TokenConfig> = {\n [STACKS_MAINNET_CAIP2]: SUSDC_MAINNET,\n [STACKS_TESTNET_CAIP2]: SUSDC_TESTNET,\n};\n\n/**\n * Get token configuration by network and symbol\n */\nexport function getTokenConfig(\n network: string,\n symbol: string = \"sUSDC\",\n): TokenConfig | undefined {\n return TOKEN_REGISTRY[network]?.[symbol];\n}\n\n/**\n * Get the default token for a network\n */\nexport function getDefaultToken(network: string): TokenConfig | undefined {\n return DEFAULT_TOKENS[network];\n}\n\n/**\n * Get contract address for a token on a network\n */\nexport function getContractAddress(network: string, symbol: string = \"sUSDC\"): string | undefined {\n return getTokenConfig(network, symbol)?.contractAddress;\n}\n","/**\n * Stacks Utility Functions\n */\n\nimport type { StacksTransactionResult, ParsedTokenTransfer } from \"./types.js\";\n\n/**\n * Validate a Stacks principal address format\n * Stacks addresses start with SP (mainnet) or ST (testnet)\n * followed by alphanumeric characters (base58-like encoding)\n */\nexport function isValidPrincipal(address: string): boolean {\n if (!address || typeof address !== \"string\") {\n return false;\n }\n\n // Standard principal: SP/ST prefix + base58 characters (33-41 chars total)\n // Contract principal: standard-principal.contract-name\n const parts = address.split(\".\");\n const principal = parts[0];\n\n // Check principal format: SP or ST prefix + alphanumeric (base58 chars)\n const principalRegex = /^(SP|ST)[0-9A-HJ-NP-Za-km-z]{33,41}$/;\n if (!principalRegex.test(principal)) {\n return false;\n }\n\n // If it's a contract principal, validate contract name\n if (parts.length === 2) {\n const contractName = parts[1];\n // Contract names: 1-128 chars, alphanumeric + hyphen + underscore\n const contractNameRegex = /^[a-zA-Z][a-zA-Z0-9\\-_]{0,127}$/;\n return contractNameRegex.test(contractName);\n }\n\n // Standard principal (no contract part) or exactly one dot for contract\n return parts.length === 1;\n}\n\n/**\n * Validate a Stacks transaction ID format\n * Transaction IDs are 0x-prefixed 64-character hex strings\n */\nexport function isValidTxId(hash: string): boolean {\n if (!hash || typeof hash !== \"string\") {\n return false;\n }\n return /^0x[a-fA-F0-9]{64}$/.test(hash);\n}\n\n/**\n * Compare two Stacks principals (case-sensitive)\n */\nexport function comparePrincipals(a: string, b: string): boolean {\n return a === b;\n}\n\n/**\n * Format an amount with decimals for display\n */\nexport function formatAmount(amount: string, decimals: number): string {\n const amountBigInt = BigInt(amount);\n const divisor = BigInt(10 ** decimals);\n const wholePart = amountBigInt / divisor;\n const fractionalPart = amountBigInt % divisor;\n\n if (fractionalPart === 0n) {\n return wholePart.toString();\n }\n\n const fractionalStr = fractionalPart.toString().padStart(decimals, \"0\");\n const trimmedFractional = fractionalStr.replace(/0+$/, \"\");\n return `${wholePart}.${trimmedFractional}`;\n}\n\n/**\n * Parse an amount string to the smallest unit (with decimals applied)\n */\nexport function parseAmount(amount: string, decimals: number): string {\n const parts = amount.split(\".\");\n const wholePart = parts[0] || \"0\";\n const fractionalPart = (parts[1] || \"\").padEnd(decimals, \"0\").slice(0, decimals);\n return BigInt(wholePart + fractionalPart).toString();\n}\n\n/**\n * Extract token transfer details from a Stacks transaction result\n * Looks for ft_transfer events matching the expected contract\n */\nexport function extractTokenTransfer(\n result: StacksTransactionResult,\n contractAddress?: string,\n): ParsedTokenTransfer | null {\n if (result.txStatus !== \"success\") {\n return null;\n }\n\n // Check if this is a contract-call transaction\n if (result.txType !== \"contract_call\") {\n return null;\n }\n\n // Check contract call is a transfer function\n if (result.contractCall) {\n const { contractId, functionName } = result.contractCall;\n\n if (functionName !== \"transfer\") {\n return null;\n }\n\n // If contractAddress specified, verify it matches\n if (contractAddress && contractId !== contractAddress) {\n return null;\n }\n\n // Look for ft_transfer event\n const transferEvent = result.events.find(\n (e) => e.eventType === \"fungible_token_asset\" && e.asset?.assetEventType === \"transfer\",\n );\n\n if (transferEvent?.asset) {\n return {\n contractAddress: contractId,\n from: transferEvent.asset.sender,\n to: transferEvent.asset.recipient,\n amount: transferEvent.asset.amount,\n success: true,\n };\n }\n\n // Fallback: extract from function args if events not available\n if (result.contractCall.functionArgs.length >= 3) {\n const amountArg = result.contractCall.functionArgs[0];\n const senderArg = result.contractCall.functionArgs[1];\n const recipientArg = result.contractCall.functionArgs[2];\n\n // Parse principal from repr (format: 'SP...')\n const senderMatch = senderArg?.repr?.match(/^'?(S[PT][0-9A-HJ-NP-Za-km-z]+)/);\n const recipientMatch = recipientArg?.repr?.match(/^'?(S[PT][0-9A-HJ-NP-Za-km-z]+)/);\n const amountMatch = amountArg?.repr?.match(/^u(\\d+)$/);\n\n if (senderMatch && recipientMatch && amountMatch) {\n return {\n contractAddress: contractId,\n from: senderMatch[1],\n to: recipientMatch[1],\n amount: amountMatch[1],\n success: true,\n };\n }\n }\n }\n\n return null;\n}\n\n/**\n * Extract token transfer from post conditions (alternative method)\n */\nexport function extractTokenTransferFromPostConditions(\n result: StacksTransactionResult,\n contractAddress?: string,\n): ParsedTokenTransfer | null {\n if (result.txStatus !== \"success\") {\n return null;\n }\n\n // Look for fungible token post conditions\n for (const pc of result.postConditions) {\n if (pc.asset) {\n const assetContractAddress = `${pc.asset.contractAddress}.${pc.asset.contractName}`;\n\n if (contractAddress && assetContractAddress !== contractAddress) {\n continue;\n }\n\n // Find corresponding ft_transfer event for recipient\n const transferEvent = result.events.find(\n (e) =>\n e.eventType === \"fungible_token_asset\" &&\n e.asset?.assetEventType === \"transfer\" &&\n e.asset?.sender === pc.principal.address,\n );\n\n if (transferEvent?.asset) {\n return {\n contractAddress: assetContractAddress,\n from: transferEvent.asset.sender,\n to: transferEvent.asset.recipient,\n amount: transferEvent.asset.amount,\n success: true,\n };\n }\n }\n }\n\n return null;\n}\n","/**\n * Stacks Exact-Direct Facilitator Scheme\n *\n * Verifies that a Stacks SIP-010 token transfer was executed correctly\n * by querying the Hiro API.\n */\n\nimport type {\n Network,\n PaymentPayload,\n PaymentRequirements,\n SchemeNetworkFacilitator,\n SettleResponse,\n VerifyResponse,\n} from \"@t402/core/types\";\nimport { STACKS_CAIP2_NAMESPACE, SCHEME_EXACT_DIRECT, getNetworkConfig } from \"../../constants.js\";\nimport { getDefaultToken } from \"../../tokens.js\";\nimport type {\n ExactDirectStacksPayload,\n FacilitatorStacksSigner,\n StacksFacilitatorConfig,\n} from \"../../types.js\";\nimport {\n comparePrincipals,\n extractTokenTransfer,\n extractTokenTransferFromPostConditions,\n isValidTxId,\n isValidPrincipal,\n} from \"../../utils.js\";\n\n// Default configuration\nconst DEFAULT_MAX_TRANSACTION_AGE = 3600; // 1 hour\nconst DEFAULT_CACHE_DURATION = 24 * 60 * 60 * 1000; // 24 hours in ms\n\n/**\n * Exact-direct facilitator scheme for Stacks\n */\nexport class ExactDirectStacksFacilitator implements SchemeNetworkFacilitator {\n readonly scheme = SCHEME_EXACT_DIRECT;\n readonly caipFamily = `${STACKS_CAIP2_NAMESPACE}:*`;\n\n private readonly signer: FacilitatorStacksSigner;\n private readonly config: Required<StacksFacilitatorConfig>;\n private readonly usedTransactions = new Map<string, number>();\n\n constructor(\n signer: FacilitatorStacksSigner,\n config: StacksFacilitatorConfig = {},\n ) {\n this.signer = signer;\n this.config = {\n maxTransactionAge: config.maxTransactionAge ?? DEFAULT_MAX_TRANSACTION_AGE,\n usedTxCacheDuration:\n config.usedTxCacheDuration ?? DEFAULT_CACHE_DURATION,\n };\n\n // Start cleanup interval\n this.startCleanupInterval();\n }\n\n /**\n * Get extra data for payment requirements\n */\n getExtra(network: Network): Record<string, unknown> | undefined {\n const config = getNetworkConfig(network);\n if (!config) return undefined;\n\n const token = getDefaultToken(network);\n return {\n contractAddress: token?.contractAddress,\n assetSymbol: token?.symbol,\n assetDecimals: token?.decimals,\n networkName: config.name,\n };\n }\n\n /**\n * Get facilitator signer addresses for a network\n */\n getSigners(network: Network): string[] {\n return this.signer.getAddresses(network);\n }\n\n /**\n * Verify a payment payload\n */\n async verify(\n payload: PaymentPayload,\n requirements: PaymentRequirements,\n ): Promise<VerifyResponse> {\n const network = requirements.network;\n\n // Validate scheme\n if (payload.accepted.scheme !== SCHEME_EXACT_DIRECT) {\n return {\n isValid: false,\n invalidReason: `invalid_scheme: expected ${SCHEME_EXACT_DIRECT}, got ${payload.accepted.scheme}`,\n };\n }\n\n // Validate network\n if (payload.accepted.network !== network) {\n return {\n isValid: false,\n invalidReason: `network_mismatch: expected ${network}, got ${payload.accepted.network}`,\n };\n }\n\n // Parse payload\n const stacksPayload = payload.payload as unknown as ExactDirectStacksPayload;\n\n // Validate required fields\n if (!stacksPayload.txId) {\n return {\n isValid: false,\n invalidReason: \"missing_tx_id\",\n };\n }\n\n // Validate tx ID format\n if (!isValidTxId(stacksPayload.txId)) {\n return {\n isValid: false,\n invalidReason: \"invalid_tx_id_format\",\n };\n }\n\n // Validate from address\n if (!stacksPayload.from) {\n return {\n isValid: false,\n invalidReason: \"missing_from_address\",\n };\n }\n\n if (!isValidPrincipal(stacksPayload.from)) {\n return {\n isValid: false,\n invalidReason: \"invalid_from_address\",\n payer: stacksPayload.from,\n };\n }\n\n // Check for replay attack\n if (this.isTransactionUsed(stacksPayload.txId)) {\n return {\n isValid: false,\n invalidReason: \"transaction_already_used\",\n payer: stacksPayload.from,\n };\n }\n\n // Query transaction\n const txResult = await this.signer.queryTransaction(stacksPayload.txId);\n\n if (!txResult) {\n return {\n isValid: false,\n invalidReason: \"transaction_not_found\",\n payer: stacksPayload.from,\n };\n }\n\n // Verify transaction was successful\n if (txResult.txStatus !== \"success\") {\n return {\n isValid: false,\n invalidReason: `transaction_failed: status=${txResult.txStatus}`,\n payer: stacksPayload.from,\n };\n }\n\n // Check transaction age\n if (this.config.maxTransactionAge > 0) {\n const txTime = txResult.burnBlockTime * 1000; // Convert to milliseconds\n const age = (Date.now() - txTime) / 1000;\n if (age > this.config.maxTransactionAge) {\n return {\n isValid: false,\n invalidReason: `transaction_too_old: ${Math.round(age)} seconds`,\n payer: stacksPayload.from,\n };\n }\n }\n\n // Extract transfer details\n const expectedContract = (requirements.extra?.contractAddress as string) ??\n stacksPayload.contractAddress;\n\n const transfer =\n extractTokenTransfer(txResult, expectedContract) ||\n extractTokenTransferFromPostConditions(txResult, expectedContract);\n\n if (!transfer) {\n return {\n isValid: false,\n invalidReason: \"not_token_transfer\",\n payer: stacksPayload.from,\n };\n }\n\n // Verify contract address\n if (expectedContract && !comparePrincipals(transfer.contractAddress, expectedContract)) {\n return {\n isValid: false,\n invalidReason: `contract_mismatch: expected ${expectedContract}, got ${transfer.contractAddress}`,\n payer: stacksPayload.from,\n };\n }\n\n // Verify recipient\n if (!comparePrincipals(transfer.to, requirements.payTo)) {\n return {\n isValid: false,\n invalidReason: `recipient_mismatch: expected ${requirements.payTo}, got ${transfer.to}`,\n payer: stacksPayload.from,\n };\n }\n\n // Verify amount\n const txAmount = BigInt(transfer.amount);\n const requiredAmount = BigInt(requirements.amount);\n if (txAmount < requiredAmount) {\n return {\n isValid: false,\n invalidReason: `insufficient_amount: expected ${requirements.amount}, got ${transfer.amount}`,\n payer: stacksPayload.from,\n };\n }\n\n // Mark transaction as used\n this.markTransactionUsed(stacksPayload.txId);\n\n return {\n isValid: true,\n payer: stacksPayload.from,\n };\n }\n\n /**\n * Settle a payment (for exact-direct, the transfer is already complete)\n */\n async settle(\n payload: PaymentPayload,\n requirements: PaymentRequirements,\n ): Promise<SettleResponse> {\n // Verify first\n const verifyResult = await this.verify(payload, requirements);\n\n if (!verifyResult.isValid) {\n return {\n success: false,\n errorReason: verifyResult.invalidReason || \"verification_failed\",\n payer: verifyResult.payer,\n transaction: \"\",\n network: requirements.network,\n };\n }\n\n const stacksPayload = payload.payload as unknown as ExactDirectStacksPayload;\n\n // For exact-direct, settlement is already complete\n return {\n success: true,\n transaction: stacksPayload.txId,\n network: requirements.network,\n payer: verifyResult.payer,\n };\n }\n\n /**\n * Check if a transaction has been used\n */\n private isTransactionUsed(txId: string): boolean {\n return this.usedTransactions.has(txId);\n }\n\n /**\n * Mark a transaction as used\n */\n private markTransactionUsed(txId: string): void {\n this.usedTransactions.set(txId, Date.now());\n }\n\n /**\n * Start the cleanup interval for used transactions cache\n */\n private startCleanupInterval(): void {\n setInterval(() => {\n const cutoff = Date.now() - this.config.usedTxCacheDuration;\n for (const [txId, timestamp] of this.usedTransactions) {\n if (timestamp < cutoff) {\n this.usedTransactions.delete(txId);\n }\n }\n }, 60 * 60 * 1000); // Run every hour\n }\n}\n\n/**\n * Create an exact-direct facilitator for Stacks\n */\nexport function createExactDirectStacksFacilitator(\n signer: FacilitatorStacksSigner,\n config: StacksFacilitatorConfig = {},\n): ExactDirectStacksFacilitator {\n return new ExactDirectStacksFacilitator(signer, config);\n}\n"],"mappings":"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","names":[]}
@@ -0,0 +1,247 @@
1
+ "use strict";
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __export = (target, all) => {
7
+ for (var name in all)
8
+ __defProp(target, name, { get: all[name], enumerable: true });
9
+ };
10
+ var __copyProps = (to, from, except, desc) => {
11
+ if (from && typeof from === "object" || typeof from === "function") {
12
+ for (let key of __getOwnPropNames(from))
13
+ if (!__hasOwnProp.call(to, key) && key !== except)
14
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
+ }
16
+ return to;
17
+ };
18
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
+
20
+ // src/exact-direct/server/index.ts
21
+ var server_exports = {};
22
+ __export(server_exports, {
23
+ ExactDirectStacksServer: () => ExactDirectStacksServer,
24
+ createExactDirectStacksServer: () => createExactDirectStacksServer,
25
+ registerExactDirectStacksServer: () => registerExactDirectStacksServer
26
+ });
27
+ module.exports = __toCommonJS(server_exports);
28
+
29
+ // src/constants.ts
30
+ var STACKS_CAIP2_NAMESPACE = "stacks";
31
+ var STACKS_MAINNET_CAIP2 = "stacks:1";
32
+ var STACKS_TESTNET_CAIP2 = "stacks:2147483648";
33
+ var SCHEME_EXACT_DIRECT = "exact-direct";
34
+ var DEFAULT_MAINNET_API = "https://api.mainnet.hiro.so";
35
+ var DEFAULT_TESTNET_API = "https://api.testnet.hiro.so";
36
+ var STACKS_NETWORKS = {
37
+ [STACKS_MAINNET_CAIP2]: {
38
+ name: "Stacks Mainnet",
39
+ caip2: STACKS_MAINNET_CAIP2,
40
+ apiUrl: DEFAULT_MAINNET_API,
41
+ chainId: 1,
42
+ addressPrefix: "SP",
43
+ isTestnet: false
44
+ },
45
+ [STACKS_TESTNET_CAIP2]: {
46
+ name: "Stacks Testnet",
47
+ caip2: STACKS_TESTNET_CAIP2,
48
+ apiUrl: DEFAULT_TESTNET_API,
49
+ chainId: 2147483648,
50
+ addressPrefix: "ST",
51
+ isTestnet: true
52
+ }
53
+ };
54
+ function isStacksNetwork(network) {
55
+ return network.startsWith(`${STACKS_CAIP2_NAMESPACE}:`);
56
+ }
57
+
58
+ // src/tokens.ts
59
+ var SUSDC_MAINNET = {
60
+ contractAddress: "SP3Y2ZSH8P7D50B0VBTSX11S7XSG24M1VB9YFQA4K.token-susdc",
61
+ symbol: "sUSDC",
62
+ name: "Stacks USDC",
63
+ decimals: 6,
64
+ issuer: "Stacks"
65
+ };
66
+ var SUSDC_TESTNET = {
67
+ contractAddress: "ST1PQHQKV0RJXZFY1DGX8MNSNYVE3VGZJSRTPGZGM.token-susdc",
68
+ symbol: "sUSDC",
69
+ name: "Test Stacks USDC",
70
+ decimals: 6
71
+ };
72
+ var TOKEN_REGISTRY = {
73
+ [STACKS_MAINNET_CAIP2]: {
74
+ sUSDC: SUSDC_MAINNET
75
+ },
76
+ [STACKS_TESTNET_CAIP2]: {
77
+ sUSDC: SUSDC_TESTNET
78
+ }
79
+ };
80
+ var DEFAULT_TOKENS = {
81
+ [STACKS_MAINNET_CAIP2]: SUSDC_MAINNET,
82
+ [STACKS_TESTNET_CAIP2]: SUSDC_TESTNET
83
+ };
84
+ function getTokenConfig(network, symbol = "sUSDC") {
85
+ return TOKEN_REGISTRY[network]?.[symbol];
86
+ }
87
+ function getDefaultToken(network) {
88
+ return DEFAULT_TOKENS[network];
89
+ }
90
+
91
+ // src/utils.ts
92
+ function parseAmount(amount, decimals) {
93
+ const parts = amount.split(".");
94
+ const wholePart = parts[0] || "0";
95
+ const fractionalPart = (parts[1] || "").padEnd(decimals, "0").slice(0, decimals);
96
+ return BigInt(wholePart + fractionalPart).toString();
97
+ }
98
+
99
+ // src/exact-direct/server/scheme.ts
100
+ var ExactDirectStacksServer = class {
101
+ scheme = SCHEME_EXACT_DIRECT;
102
+ moneyParsers = [];
103
+ config;
104
+ constructor(config = {}) {
105
+ this.config = config;
106
+ }
107
+ /**
108
+ * Register a custom money parser in the parser chain.
109
+ */
110
+ registerMoneyParser(parser) {
111
+ this.moneyParsers.push(parser);
112
+ return this;
113
+ }
114
+ /**
115
+ * Parse price into Stacks-specific amount
116
+ */
117
+ async parsePrice(price, network) {
118
+ if (!isStacksNetwork(network)) {
119
+ throw new Error(`Invalid Stacks network: ${network}`);
120
+ }
121
+ if (typeof price === "object" && price !== null && "amount" in price) {
122
+ if (!price.asset) {
123
+ throw new Error(`Asset must be specified for AssetAmount on network ${network}`);
124
+ }
125
+ return {
126
+ amount: price.amount,
127
+ asset: price.asset,
128
+ extra: price.extra || {}
129
+ };
130
+ }
131
+ const amount = this.parseMoneyToDecimal(price);
132
+ for (const parser of this.moneyParsers) {
133
+ const result = await parser(amount, network);
134
+ if (result !== null) {
135
+ return result;
136
+ }
137
+ }
138
+ return this.defaultMoneyConversion(amount, network);
139
+ }
140
+ /**
141
+ * Enhance payment requirements with Stacks-specific details
142
+ */
143
+ async enhancePaymentRequirements(paymentRequirements, supportedKind, facilitatorExtensions) {
144
+ void facilitatorExtensions;
145
+ const extra = { ...paymentRequirements.extra };
146
+ if (supportedKind.extra?.contractAddress) {
147
+ extra.contractAddress = supportedKind.extra.contractAddress;
148
+ }
149
+ if (supportedKind.extra?.assetSymbol) {
150
+ extra.assetSymbol = supportedKind.extra.assetSymbol;
151
+ }
152
+ if (supportedKind.extra?.assetDecimals) {
153
+ extra.assetDecimals = supportedKind.extra.assetDecimals;
154
+ }
155
+ if (supportedKind.extra?.networkName) {
156
+ extra.networkName = supportedKind.extra.networkName;
157
+ }
158
+ return {
159
+ ...paymentRequirements,
160
+ extra
161
+ };
162
+ }
163
+ /**
164
+ * Parse Money (string | number) to a decimal number.
165
+ */
166
+ parseMoneyToDecimal(money) {
167
+ if (typeof money === "number") {
168
+ return money;
169
+ }
170
+ const cleanMoney = money.replace(/^\$/, "").trim();
171
+ const amount = parseFloat(cleanMoney);
172
+ if (isNaN(amount)) {
173
+ throw new Error(`Invalid money format: ${money}`);
174
+ }
175
+ return amount;
176
+ }
177
+ /**
178
+ * Default money conversion implementation.
179
+ */
180
+ defaultMoneyConversion(amount, network) {
181
+ const token = this.getDefaultAsset(network);
182
+ const tokenAmount = parseAmount(amount.toString(), token.decimals);
183
+ return {
184
+ amount: tokenAmount.toString(),
185
+ asset: this.createAssetIdentifier(network, token.contractAddress),
186
+ extra: {
187
+ symbol: token.symbol,
188
+ name: token.name,
189
+ decimals: token.decimals,
190
+ contractAddress: token.contractAddress
191
+ }
192
+ };
193
+ }
194
+ /**
195
+ * Create a CAIP-19 asset identifier for Stacks tokens
196
+ */
197
+ createAssetIdentifier(network, contractAddress) {
198
+ return `${network}/sip010:${contractAddress}`;
199
+ }
200
+ /**
201
+ * Get the default asset info for a network.
202
+ */
203
+ getDefaultAsset(network) {
204
+ if (this.config.preferredToken) {
205
+ const preferred = getTokenConfig(network, this.config.preferredToken);
206
+ if (preferred) return preferred;
207
+ }
208
+ const defaultToken = getDefaultToken(network);
209
+ if (defaultToken) return defaultToken;
210
+ throw new Error(`No tokens configured for network ${network}`);
211
+ }
212
+ /**
213
+ * Get all supported networks
214
+ */
215
+ static getSupportedNetworks() {
216
+ return Object.keys(TOKEN_REGISTRY);
217
+ }
218
+ /**
219
+ * Check if a network is supported
220
+ */
221
+ static isNetworkSupported(network) {
222
+ return network in TOKEN_REGISTRY;
223
+ }
224
+ };
225
+ function createExactDirectStacksServer(config = {}) {
226
+ return new ExactDirectStacksServer(config);
227
+ }
228
+
229
+ // src/exact-direct/server/register.ts
230
+ function registerExactDirectStacksServer(server, config = {}) {
231
+ const scheme = new ExactDirectStacksServer(config);
232
+ if (config.networks && config.networks.length > 0) {
233
+ config.networks.forEach((network) => {
234
+ server.register(network, scheme);
235
+ });
236
+ } else {
237
+ server.register(`${STACKS_CAIP2_NAMESPACE}:*`, scheme);
238
+ }
239
+ return server;
240
+ }
241
+ // Annotate the CommonJS export names for ESM import in node:
242
+ 0 && (module.exports = {
243
+ ExactDirectStacksServer,
244
+ createExactDirectStacksServer,
245
+ registerExactDirectStacksServer
246
+ });
247
+ //# sourceMappingURL=index.cjs.map