@oobe-protocol-labs/synapse-sap-sdk 0.5.0 → 0.6.2

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 (93) hide show
  1. package/README.md +107 -3
  2. package/dist/cjs/index.js +44 -2
  3. package/dist/cjs/index.js.map +1 -1
  4. package/dist/cjs/modules/escrow.js +23 -8
  5. package/dist/cjs/modules/escrow.js.map +1 -1
  6. package/dist/cjs/plugin/index.js +18 -2
  7. package/dist/cjs/plugin/index.js.map +1 -1
  8. package/dist/cjs/plugin/schemas.js +32 -0
  9. package/dist/cjs/plugin/schemas.js.map +1 -1
  10. package/dist/cjs/registries/index.js.map +1 -1
  11. package/dist/cjs/registries/x402.js +54 -9
  12. package/dist/cjs/registries/x402.js.map +1 -1
  13. package/dist/cjs/types/endpoint.js +15 -0
  14. package/dist/cjs/types/endpoint.js.map +1 -0
  15. package/dist/cjs/utils/endpoint-validator.js +232 -0
  16. package/dist/cjs/utils/endpoint-validator.js.map +1 -0
  17. package/dist/cjs/utils/index.js +39 -1
  18. package/dist/cjs/utils/index.js.map +1 -1
  19. package/dist/cjs/utils/network-normalizer.js +236 -0
  20. package/dist/cjs/utils/network-normalizer.js.map +1 -0
  21. package/dist/cjs/utils/priority-fee.js +163 -0
  22. package/dist/cjs/utils/priority-fee.js.map +1 -0
  23. package/dist/cjs/utils/rpc-strategy.js +239 -0
  24. package/dist/cjs/utils/rpc-strategy.js.map +1 -0
  25. package/dist/cjs/utils/schemas.js +331 -0
  26. package/dist/cjs/utils/schemas.js.map +1 -0
  27. package/dist/esm/index.js +10 -0
  28. package/dist/esm/index.js.map +1 -1
  29. package/dist/esm/modules/escrow.js +23 -8
  30. package/dist/esm/modules/escrow.js.map +1 -1
  31. package/dist/esm/plugin/index.js +18 -2
  32. package/dist/esm/plugin/index.js.map +1 -1
  33. package/dist/esm/plugin/schemas.js +32 -0
  34. package/dist/esm/plugin/schemas.js.map +1 -1
  35. package/dist/esm/registries/index.js.map +1 -1
  36. package/dist/esm/registries/x402.js +54 -9
  37. package/dist/esm/registries/x402.js.map +1 -1
  38. package/dist/esm/types/endpoint.js +14 -0
  39. package/dist/esm/types/endpoint.js.map +1 -0
  40. package/dist/esm/utils/endpoint-validator.js +226 -0
  41. package/dist/esm/utils/endpoint-validator.js.map +1 -0
  42. package/dist/esm/utils/index.js +7 -0
  43. package/dist/esm/utils/index.js.map +1 -1
  44. package/dist/esm/utils/network-normalizer.js +229 -0
  45. package/dist/esm/utils/network-normalizer.js.map +1 -0
  46. package/dist/esm/utils/priority-fee.js +158 -0
  47. package/dist/esm/utils/priority-fee.js.map +1 -0
  48. package/dist/esm/utils/rpc-strategy.js +231 -0
  49. package/dist/esm/utils/rpc-strategy.js.map +1 -0
  50. package/dist/esm/utils/schemas.js +320 -0
  51. package/dist/esm/utils/schemas.js.map +1 -0
  52. package/dist/types/index.d.ts +10 -2
  53. package/dist/types/index.d.ts.map +1 -1
  54. package/dist/types/modules/escrow.d.ts +7 -2
  55. package/dist/types/modules/escrow.d.ts.map +1 -1
  56. package/dist/types/plugin/index.d.ts.map +1 -1
  57. package/dist/types/plugin/schemas.d.ts +12 -6
  58. package/dist/types/plugin/schemas.d.ts.map +1 -1
  59. package/dist/types/registries/index.d.ts +1 -1
  60. package/dist/types/registries/index.d.ts.map +1 -1
  61. package/dist/types/registries/x402.d.ts +34 -2
  62. package/dist/types/registries/x402.d.ts.map +1 -1
  63. package/dist/types/types/endpoint.d.ts +161 -0
  64. package/dist/types/types/endpoint.d.ts.map +1 -0
  65. package/dist/types/types/index.d.ts +1 -0
  66. package/dist/types/types/index.d.ts.map +1 -1
  67. package/dist/types/utils/endpoint-validator.d.ts +110 -0
  68. package/dist/types/utils/endpoint-validator.d.ts.map +1 -0
  69. package/dist/types/utils/index.d.ts +8 -0
  70. package/dist/types/utils/index.d.ts.map +1 -1
  71. package/dist/types/utils/network-normalizer.d.ts +120 -0
  72. package/dist/types/utils/network-normalizer.d.ts.map +1 -0
  73. package/dist/types/utils/priority-fee.d.ts +185 -0
  74. package/dist/types/utils/priority-fee.d.ts.map +1 -0
  75. package/dist/types/utils/rpc-strategy.d.ts +172 -0
  76. package/dist/types/utils/rpc-strategy.d.ts.map +1 -0
  77. package/dist/types/utils/schemas.d.ts +351 -0
  78. package/dist/types/utils/schemas.d.ts.map +1 -0
  79. package/package.json +1 -1
  80. package/src/index.ts +63 -0
  81. package/src/modules/escrow.ts +33 -6
  82. package/src/plugin/index.ts +20 -0
  83. package/src/plugin/schemas.ts +32 -0
  84. package/src/registries/index.ts +1 -0
  85. package/src/registries/x402.ts +68 -7
  86. package/src/types/endpoint.ts +181 -0
  87. package/src/types/index.ts +9 -0
  88. package/src/utils/endpoint-validator.ts +300 -0
  89. package/src/utils/index.ts +54 -0
  90. package/src/utils/network-normalizer.ts +240 -0
  91. package/src/utils/priority-fee.ts +270 -0
  92. package/src/utils/rpc-strategy.ts +322 -0
  93. package/src/utils/schemas.ts +359 -0
@@ -0,0 +1,229 @@
1
+ /**
2
+ * @module utils/network-normalizer
3
+ * @description Network identifier normalization for x402 payment headers.
4
+ *
5
+ * Solves the canonical-string mismatch between SAP clients and servers:
6
+ * some providers accept `solana:mainnet-beta` while others require
7
+ * the genesis-hash form `solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp`.
8
+ *
9
+ * This module provides a single source of truth for normalizing
10
+ * network identifiers so that both clients and agents canonicalize
11
+ * before comparing. Catches the Kamiyo "sap network mismatch" error
12
+ * at the SDK level.
13
+ *
14
+ * @category Utils
15
+ * @since v0.6.0
16
+ */
17
+ import { SapNetwork } from "../constants/network";
18
+ // ═══════════════════════════════════════════════════════════════════
19
+ // Canonical Mapping
20
+ // ═══════════════════════════════════════════════════════════════════
21
+ /**
22
+ * Bidirectional alias map: for every known network string, stores the
23
+ * canonical SapNetworkId it resolves to, plus all known aliases.
24
+ */
25
+ const NETWORK_ALIASES = new Map([
26
+ // ── Mainnet ───────────────────────────────
27
+ // Canonical: genesis-hash form
28
+ ["solana:mainnet-beta", SapNetwork.SOLANA_MAINNET],
29
+ ["solana:mainnet", SapNetwork.SOLANA_MAINNET],
30
+ ["mainnet-beta", SapNetwork.SOLANA_MAINNET],
31
+ ["mainnet", SapNetwork.SOLANA_MAINNET],
32
+ ["solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp", SapNetwork.SOLANA_MAINNET_GENESIS],
33
+ ["5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp", SapNetwork.SOLANA_MAINNET_GENESIS],
34
+ // ── Devnet ────────────────────────────────
35
+ ["solana:devnet", SapNetwork.SOLANA_DEVNET_NAMED],
36
+ ["devnet", SapNetwork.SOLANA_DEVNET_NAMED],
37
+ ["solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1", SapNetwork.SOLANA_DEVNET],
38
+ ["EtWTRABZaYq6iMfeYKouRu166VU2xqa1", SapNetwork.SOLANA_DEVNET],
39
+ ]);
40
+ /**
41
+ * Mainnet equivalence set: all strings that refer to Solana mainnet-beta,
42
+ * regardless of format.
43
+ */
44
+ const MAINNET_EQUIVALENTS = new Set([
45
+ SapNetwork.SOLANA_MAINNET,
46
+ SapNetwork.SOLANA_MAINNET_GENESIS,
47
+ "solana:mainnet",
48
+ "mainnet-beta",
49
+ "mainnet",
50
+ "5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp",
51
+ ]);
52
+ /**
53
+ * Devnet equivalence set.
54
+ */
55
+ const DEVNET_EQUIVALENTS = new Set([
56
+ SapNetwork.SOLANA_DEVNET,
57
+ SapNetwork.SOLANA_DEVNET_NAMED,
58
+ "devnet",
59
+ "EtWTRABZaYq6iMfeYKouRu166VU2xqa1",
60
+ ]);
61
+ // ═══════════════════════════════════════════════════════════════════
62
+ // Public API
63
+ // ═══════════════════════════════════════════════════════════════════
64
+ /**
65
+ * @name normalizeNetworkId
66
+ * @description Normalize a raw network identifier string to its canonical
67
+ * {@link SapNetworkId} form.
68
+ *
69
+ * Handles:
70
+ * - Case-insensitive matching
71
+ * - Stripping whitespace
72
+ * - Resolving genesis-hash vs. cluster-name aliases
73
+ * - Unknown strings are returned as-is (passthrough)
74
+ *
75
+ * @param raw - Raw network identifier string from headers, env vars, or config.
76
+ * @returns The canonical {@link SapNetworkId}, or the trimmed input if unknown.
77
+ *
78
+ * @category Utils
79
+ * @since v0.6.0
80
+ *
81
+ * @example
82
+ * ```ts
83
+ * import { normalizeNetworkId } from "@synapse-sap/sdk";
84
+ *
85
+ * normalizeNetworkId("solana:mainnet-beta");
86
+ * // → "solana:mainnet-beta"
87
+ *
88
+ * normalizeNetworkId("5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp");
89
+ * // → "solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp"
90
+ *
91
+ * normalizeNetworkId(" MAINNET ");
92
+ * // → "solana:mainnet-beta"
93
+ * ```
94
+ */
95
+ export function normalizeNetworkId(raw) {
96
+ const trimmed = raw.trim();
97
+ // Exact match first
98
+ const exact = NETWORK_ALIASES.get(trimmed);
99
+ if (exact)
100
+ return exact;
101
+ // Case-insensitive lookup
102
+ const lower = trimmed.toLowerCase();
103
+ for (const [alias, canonical] of NETWORK_ALIASES) {
104
+ if (alias.toLowerCase() === lower)
105
+ return canonical;
106
+ }
107
+ // Passthrough for unknown networks (custom chains, etc.)
108
+ return trimmed;
109
+ }
110
+ /**
111
+ * @name isNetworkEquivalent
112
+ * @description Check if two network identifier strings refer to the same network,
113
+ * even if they use different formats (cluster-name vs. genesis-hash).
114
+ *
115
+ * This is the key function that prevents the Kamiyo "sap network mismatch"
116
+ * error — instead of comparing strings literally, we compare their
117
+ * canonical equivalence class.
118
+ *
119
+ * @param a - First network identifier.
120
+ * @param b - Second network identifier.
121
+ * @returns `true` if both identifiers refer to the same Solana network.
122
+ *
123
+ * @category Utils
124
+ * @since v0.6.0
125
+ *
126
+ * @example
127
+ * ```ts
128
+ * import { isNetworkEquivalent } from "@synapse-sap/sdk";
129
+ *
130
+ * isNetworkEquivalent("solana:mainnet-beta", "solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp");
131
+ * // → true
132
+ *
133
+ * isNetworkEquivalent("solana:devnet", "solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1");
134
+ * // → true
135
+ *
136
+ * isNetworkEquivalent("solana:mainnet-beta", "solana:devnet");
137
+ * // → false
138
+ * ```
139
+ */
140
+ export function isNetworkEquivalent(a, b) {
141
+ const normA = a.trim();
142
+ const normB = b.trim();
143
+ // Fast path: identical strings
144
+ if (normA === normB)
145
+ return true;
146
+ // Check if both are in the same equivalence set
147
+ if (MAINNET_EQUIVALENTS.has(normA) && MAINNET_EQUIVALENTS.has(normB))
148
+ return true;
149
+ if (DEVNET_EQUIVALENTS.has(normA) && DEVNET_EQUIVALENTS.has(normB))
150
+ return true;
151
+ // Case-insensitive match
152
+ const lowerA = normA.toLowerCase();
153
+ const lowerB = normB.toLowerCase();
154
+ if (lowerA === lowerB)
155
+ return true;
156
+ // Resolve both to canonical and compare
157
+ const canonA = normalizeNetworkId(normA);
158
+ const canonB = normalizeNetworkId(normB);
159
+ if (canonA === canonB)
160
+ return true;
161
+ // Final check: both resolve to the same equivalence set
162
+ if (MAINNET_EQUIVALENTS.has(canonA) && MAINNET_EQUIVALENTS.has(canonB))
163
+ return true;
164
+ if (DEVNET_EQUIVALENTS.has(canonA) && DEVNET_EQUIVALENTS.has(canonB))
165
+ return true;
166
+ return false;
167
+ }
168
+ /**
169
+ * @name getNetworkGenesisHash
170
+ * @description Get the genesis-hash form of a network identifier.
171
+ * Returns the genesis-hash variant for known networks, or the input as-is.
172
+ *
173
+ * Useful for agents that require the genesis-hash form (Kamiyo, Helius x402).
174
+ *
175
+ * @param networkId - Any network identifier.
176
+ * @returns The genesis-hash form, or the input if unknown.
177
+ *
178
+ * @category Utils
179
+ * @since v0.6.0
180
+ */
181
+ export function getNetworkGenesisHash(networkId) {
182
+ const norm = normalizeNetworkId(networkId);
183
+ if (MAINNET_EQUIVALENTS.has(networkId) || MAINNET_EQUIVALENTS.has(norm)) {
184
+ return SapNetwork.SOLANA_MAINNET_GENESIS;
185
+ }
186
+ if (DEVNET_EQUIVALENTS.has(networkId) || DEVNET_EQUIVALENTS.has(norm)) {
187
+ return SapNetwork.SOLANA_DEVNET;
188
+ }
189
+ return norm;
190
+ }
191
+ /**
192
+ * @name getNetworkClusterName
193
+ * @description Get the cluster-name form of a network identifier.
194
+ * Returns the human-readable cluster name for known networks.
195
+ *
196
+ * Useful for providers that accept cluster names (Coinbase, Phantom).
197
+ *
198
+ * @param networkId - Any network identifier.
199
+ * @returns The cluster-name form, or the input if unknown.
200
+ *
201
+ * @category Utils
202
+ * @since v0.6.0
203
+ */
204
+ export function getNetworkClusterName(networkId) {
205
+ const norm = normalizeNetworkId(networkId);
206
+ if (MAINNET_EQUIVALENTS.has(networkId) || MAINNET_EQUIVALENTS.has(norm)) {
207
+ return SapNetwork.SOLANA_MAINNET;
208
+ }
209
+ if (DEVNET_EQUIVALENTS.has(networkId) || DEVNET_EQUIVALENTS.has(norm)) {
210
+ return SapNetwork.SOLANA_DEVNET_NAMED;
211
+ }
212
+ return norm;
213
+ }
214
+ /**
215
+ * @name isKnownNetwork
216
+ * @description Check if a network identifier is recognized by the SDK.
217
+ *
218
+ * @param networkId - The network identifier to check.
219
+ * @returns `true` if the identifier maps to a known Solana network.
220
+ *
221
+ * @category Utils
222
+ * @since v0.6.0
223
+ */
224
+ export function isKnownNetwork(networkId) {
225
+ const trimmed = networkId.trim();
226
+ return MAINNET_EQUIVALENTS.has(trimmed) || DEVNET_EQUIVALENTS.has(trimmed) ||
227
+ NETWORK_ALIASES.has(trimmed);
228
+ }
229
+ //# sourceMappingURL=network-normalizer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"network-normalizer.js","sourceRoot":"","sources":["../../../src/utils/network-normalizer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,158 @@
1
+ /**
2
+ * @module utils/priority-fee
3
+ * @description Compute budget and priority fee utilities for SAP transactions.
4
+ *
5
+ * Solana transactions that include a priority fee (via the Compute Budget program)
6
+ * land faster because validators prefer higher-fee transactions. This module
7
+ * provides a clean, composable API for building priority-fee instructions
8
+ * that can be prepended to any Anchor method builder via `.preInstructions()`.
9
+ *
10
+ * Typical use: x402 settlement transactions where the receiving agent's RPC
11
+ * has a short confirmation window (e.g., 30 seconds).
12
+ *
13
+ * @category Utils
14
+ * @since v0.6.2
15
+ *
16
+ * @example
17
+ * ```ts
18
+ * import { buildPriorityFeeIxs, DEFAULT_SETTLE_PRIORITY } from "@synapse-sap/sdk";
19
+ *
20
+ * // Append to any Anchor method builder:
21
+ * await program.methods
22
+ * .settleCalls(calls, hash)
23
+ * .accounts({ ... })
24
+ * .preInstructions(buildPriorityFeeIxs({ priorityFeeMicroLamports: 5000 }))
25
+ * .rpc({ skipPreflight: true });
26
+ * ```
27
+ */
28
+ import { ComputeBudgetProgram, } from "@solana/web3.js";
29
+ // ═══════════════════════════════════════════════════════════════════
30
+ // Constants
31
+ // ═══════════════════════════════════════════════════════════════════
32
+ /**
33
+ * Default priority fee for settlement transactions (in microlamports).
34
+ * 5000 µL ≈ 0.0005 SOL per 200k CU — fast enough for most agent RPCs.
35
+ *
36
+ * @since v0.6.2
37
+ */
38
+ export const DEFAULT_SETTLE_PRIORITY_FEE = 5_000;
39
+ /**
40
+ * Default compute unit limit for settlement transactions.
41
+ * `settle_calls` uses ~60k CU; 100k provides a safe margin.
42
+ *
43
+ * @since v0.6.2
44
+ */
45
+ export const DEFAULT_SETTLE_COMPUTE_UNITS = 100_000;
46
+ /**
47
+ * Default compute unit limit for batch settlement transactions.
48
+ * `settle_batch` with 10 entries uses ~200k CU; 300k provides margin.
49
+ *
50
+ * @since v0.6.2
51
+ */
52
+ export const DEFAULT_BATCH_SETTLE_COMPUTE_UNITS = 300_000;
53
+ // ═══════════════════════════════════════════════════════════════════
54
+ // Presets
55
+ // ═══════════════════════════════════════════════════════════════════
56
+ /**
57
+ * Recommended preset for fast x402 settlements.
58
+ * Priority fee 5000 µL, 100k CU, skip preflight, confirmed commitment.
59
+ *
60
+ * @since v0.6.2
61
+ */
62
+ export const FAST_SETTLE_OPTIONS = Object.freeze({
63
+ priorityFeeMicroLamports: DEFAULT_SETTLE_PRIORITY_FEE,
64
+ computeUnits: DEFAULT_SETTLE_COMPUTE_UNITS,
65
+ skipPreflight: true,
66
+ commitment: "confirmed",
67
+ });
68
+ /**
69
+ * Recommended preset for fast batch settlements.
70
+ * Priority fee 5000 µL, 300k CU, skip preflight, confirmed commitment.
71
+ *
72
+ * @since v0.6.2
73
+ */
74
+ export const FAST_BATCH_SETTLE_OPTIONS = Object.freeze({
75
+ priorityFeeMicroLamports: DEFAULT_SETTLE_PRIORITY_FEE,
76
+ computeUnits: DEFAULT_BATCH_SETTLE_COMPUTE_UNITS,
77
+ skipPreflight: true,
78
+ commitment: "confirmed",
79
+ });
80
+ // ═══════════════════════════════════════════════════════════════════
81
+ // Builder
82
+ // ═══════════════════════════════════════════════════════════════════
83
+ /**
84
+ * @name buildPriorityFeeIxs
85
+ * @description Build compute budget instructions for priority fee transactions.
86
+ *
87
+ * Returns an array of 0–2 `TransactionInstruction`s:
88
+ * - `SetComputeUnitPrice` (if `priorityFeeMicroLamports > 0`)
89
+ * - `SetComputeUnitLimit` (if `computeUnits` provided)
90
+ *
91
+ * The returned array is designed to be passed directly to
92
+ * Anchor's `.preInstructions()` builder method.
93
+ *
94
+ * @param config - Priority fee configuration.
95
+ * @returns Array of compute budget instructions (may be empty).
96
+ *
97
+ * @category Utils
98
+ * @since v0.6.2
99
+ *
100
+ * @example
101
+ * ```ts
102
+ * const ixs = buildPriorityFeeIxs({
103
+ * priorityFeeMicroLamports: 5000,
104
+ * computeUnits: 100_000,
105
+ * });
106
+ *
107
+ * await program.methods
108
+ * .settleCalls(calls, hash)
109
+ * .accounts({ ... })
110
+ * .preInstructions(ixs)
111
+ * .rpc({ skipPreflight: true });
112
+ * ```
113
+ */
114
+ export function buildPriorityFeeIxs(config) {
115
+ if (!config)
116
+ return [];
117
+ const ixs = [];
118
+ const fee = config.priorityFeeMicroLamports ?? 0;
119
+ if (fee > 0) {
120
+ ixs.push(ComputeBudgetProgram.setComputeUnitPrice({
121
+ microLamports: fee,
122
+ }));
123
+ }
124
+ const cu = config.computeUnits;
125
+ if (cu !== undefined && cu > 0) {
126
+ ixs.push(ComputeBudgetProgram.setComputeUnitLimit({
127
+ units: cu,
128
+ }));
129
+ }
130
+ return ixs;
131
+ }
132
+ /**
133
+ * @name buildRpcOptions
134
+ * @description Build Anchor `.rpc()` options from {@link SettleOptions}.
135
+ *
136
+ * @param opts - Settle options.
137
+ * @returns Options object suitable for Anchor `.rpc(opts)`.
138
+ *
139
+ * @category Utils
140
+ * @since v0.6.2
141
+ * @internal
142
+ */
143
+ export function buildRpcOptions(opts) {
144
+ if (!opts)
145
+ return undefined;
146
+ const rpcOpts = {};
147
+ if (opts.skipPreflight !== undefined) {
148
+ rpcOpts.skipPreflight = opts.skipPreflight;
149
+ }
150
+ if (opts.commitment !== undefined) {
151
+ rpcOpts.commitment = opts.commitment;
152
+ }
153
+ if (opts.maxRetries !== undefined) {
154
+ rpcOpts.maxRetries = opts.maxRetries;
155
+ }
156
+ return Object.keys(rpcOpts).length > 0 ? rpcOpts : undefined;
157
+ }
158
+ //# sourceMappingURL=priority-fee.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"priority-fee.js","sourceRoot":"","sources":["../../../src/utils/priority-fee.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BG;AAEH,OAAO,EACL,oBAAoB,GAErB,MAAM,iBAAiB,CAAC;AAEzB,sEAAsE;AACtE,aAAa;AACb,sEAAsE;AAEtE;;;;;GAKG;AACH,MAAM,CAAC,MAAM,2BAA2B,GAAG,KAAK,CAAC;AAEjD;;;;;GAKG;AACH,MAAM,CAAC,MAAM,4BAA4B,GAAG,OAAO,CAAC;AAEpD;;;;;GAKG;AACH,MAAM,CAAC,MAAM,kCAAkC,GAAG,OAAO,CAAC;AAwF1D,sEAAsE;AACtE,WAAW;AACX,sEAAsE;AAEtE;;;;;GAKG;AACH,MAAM,CAAC,MAAM,mBAAmB,GAA4B,MAAM,CAAC,MAAM,CAAC;IACxE,wBAAwB,EAAE,2BAA2B;IACrD,YAAY,EAAE,4BAA4B;IAC1C,aAAa,EAAE,IAAI;IACnB,UAAU,EAAE,WAAW;CACxB,CAAC,CAAC;AAEH;;;;;GAKG;AACH,MAAM,CAAC,MAAM,yBAAyB,GAA4B,MAAM,CAAC,MAAM,CAAC;IAC9E,wBAAwB,EAAE,2BAA2B;IACrD,YAAY,EAAE,kCAAkC;IAChD,aAAa,EAAE,IAAI;IACnB,UAAU,EAAE,WAAW;CACxB,CAAC,CAAC;AAEH,sEAAsE;AACtE,WAAW;AACX,sEAAsE;AAEtE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8BG;AACH,MAAM,UAAU,mBAAmB,CACjC,MAA0B;IAE1B,IAAI,CAAC,MAAM;QAAE,OAAO,EAAE,CAAC;IAEvB,MAAM,GAAG,GAA6B,EAAE,CAAC;IAEzC,MAAM,GAAG,GAAG,MAAM,CAAC,wBAAwB,IAAI,CAAC,CAAC;IACjD,IAAI,GAAG,GAAG,CAAC,EAAE,CAAC;QACZ,GAAG,CAAC,IAAI,CACN,oBAAoB,CAAC,mBAAmB,CAAC;YACvC,aAAa,EAAE,GAAG;SACnB,CAAC,CACH,CAAC;IACJ,CAAC;IAED,MAAM,EAAE,GAAG,MAAM,CAAC,YAAY,CAAC;IAC/B,IAAI,EAAE,KAAK,SAAS,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC;QAC/B,GAAG,CAAC,IAAI,CACN,oBAAoB,CAAC,mBAAmB,CAAC;YACvC,KAAK,EAAE,EAAE;SACV,CAAC,CACH,CAAC;IACJ,CAAC;IAED,OAAO,GAAG,CAAC;AACb,CAAC;AAED;;;;;;;;;;GAUG;AACH,MAAM,UAAU,eAAe,CAC7B,IAAoB;IAEpB,IAAI,CAAC,IAAI;QAAE,OAAO,SAAS,CAAC;IAE5B,MAAM,OAAO,GAA4B,EAAE,CAAC;IAE5C,IAAI,IAAI,CAAC,aAAa,KAAK,SAAS,EAAE,CAAC;QACrC,OAAO,CAAC,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC;IAC7C,CAAC;IACD,IAAI,IAAI,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;QAClC,OAAO,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC;IACvC,CAAC;IACD,IAAI,IAAI,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;QAClC,OAAO,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC;IACvC,CAAC;IAED,OAAO,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC;AAC/D,CAAC"}
@@ -0,0 +1,231 @@
1
+ /**
2
+ * @module utils/rpc-strategy
3
+ * @description Dual-connection RPC strategy and idempotent ATA creation.
4
+ *
5
+ * Solves two interoperability problems:
6
+ *
7
+ * 1. **WebSocket 400 loop**: Some authenticated RPCs reject WebSocket
8
+ * connections for SPL token operations. This module exposes a
9
+ * dual-connection strategy: primary RPC for SAP program calls,
10
+ * fallback public RPC for token operations.
11
+ *
12
+ * 2. **Idempotent ATA creation**: Wraps `getOrCreateAssociatedTokenAccount`
13
+ * with retries so "account already exists" doesn't surface as a
14
+ * hard error.
15
+ *
16
+ * @category Utils
17
+ * @since v0.6.0
18
+ */
19
+ import { Connection, PublicKey, } from "@solana/web3.js";
20
+ // ═══════════════════════════════════════════════════════════════════
21
+ // Default Public RPCs
22
+ // ═══════════════════════════════════════════════════════════════════
23
+ /** Well-known public fallback RPCs by cluster. */
24
+ const PUBLIC_RPCS = {
25
+ "mainnet-beta": "https://api.mainnet-beta.solana.com",
26
+ devnet: "https://api.devnet.solana.com",
27
+ localnet: "http://localhost:8899",
28
+ };
29
+ // ═══════════════════════════════════════════════════════════════════
30
+ // Public API
31
+ // ═══════════════════════════════════════════════════════════════════
32
+ /**
33
+ * @name getRpcUrl
34
+ * @description Get the primary RPC URL from environment or config.
35
+ *
36
+ * Resolution order:
37
+ * 1. Explicit `config.primaryUrl`
38
+ * 2. `SOLANA_RPC_URL` env var
39
+ * 3. Cluster-appropriate public RPC
40
+ *
41
+ * @param config - Optional RPC configuration.
42
+ * @param cluster - Cluster hint (defaults to "mainnet-beta").
43
+ * @returns The resolved primary RPC URL.
44
+ *
45
+ * @category Utils
46
+ * @since v0.6.0
47
+ */
48
+ export function getRpcUrl(config, cluster = "mainnet-beta") {
49
+ if (config?.primaryUrl)
50
+ return config.primaryUrl;
51
+ // Environment variable
52
+ const envUrl = typeof process !== "undefined" ? process.env?.SOLANA_RPC_URL : undefined;
53
+ if (envUrl)
54
+ return envUrl;
55
+ return PUBLIC_RPCS[cluster] ?? PUBLIC_RPCS["mainnet-beta"];
56
+ }
57
+ /**
58
+ * @name getFallbackRpcUrl
59
+ * @description Get the fallback RPC URL for SPL token operations.
60
+ *
61
+ * This avoids the WebSocket-400 loop when the primary RPC is
62
+ * an authenticated endpoint that rejects token-related WebSocket
63
+ * subscriptions.
64
+ *
65
+ * @param config - Optional RPC configuration.
66
+ * @param cluster - Cluster hint (defaults to "mainnet-beta").
67
+ * @returns The resolved fallback RPC URL.
68
+ *
69
+ * @category Utils
70
+ * @since v0.6.0
71
+ */
72
+ export function getFallbackRpcUrl(config, cluster = "mainnet-beta") {
73
+ if (config?.fallbackUrl)
74
+ return config.fallbackUrl;
75
+ // Env fallback
76
+ const envUrl = typeof process !== "undefined"
77
+ ? process.env?.SOLANA_FALLBACK_RPC_URL
78
+ : undefined;
79
+ if (envUrl)
80
+ return envUrl;
81
+ return PUBLIC_RPCS[cluster] ?? PUBLIC_RPCS["mainnet-beta"];
82
+ }
83
+ /**
84
+ * @name createDualConnection
85
+ * @description Create a dual-connection pair: primary for SAP program calls,
86
+ * fallback for SPL token operations.
87
+ *
88
+ * @param config - RPC configuration.
89
+ * @param cluster - Cluster hint.
90
+ * @returns A {@link DualConnection} with both connections.
91
+ *
92
+ * @category Utils
93
+ * @since v0.6.0
94
+ *
95
+ * @example
96
+ * ```ts
97
+ * const { primary, fallback } = createDualConnection({
98
+ * primaryUrl: "https://my-rpc.example.com",
99
+ * }, "mainnet-beta");
100
+ *
101
+ * // Use primary for SAP calls
102
+ * const provider = new AnchorProvider(primary, wallet, {});
103
+ *
104
+ * // Use fallback for SPL token account creation
105
+ * const ata = await getOrCreateATA(fallback, mint, owner);
106
+ * ```
107
+ */
108
+ export function createDualConnection(config, cluster = "mainnet-beta") {
109
+ const commitment = config.commitment ?? "confirmed";
110
+ const primaryUrl = getRpcUrl(config, cluster);
111
+ const fallbackUrl = getFallbackRpcUrl(config, cluster);
112
+ return {
113
+ primary: new Connection(primaryUrl, { commitment }),
114
+ fallback: new Connection(fallbackUrl, { commitment }),
115
+ };
116
+ }
117
+ /**
118
+ * @name findATA
119
+ * @description Derive the Associated Token Account address.
120
+ * Uses the standard ATA PDA derivation without importing the full
121
+ * `@solana/spl-token` package.
122
+ *
123
+ * @param owner - The token account owner.
124
+ * @param mint - The token mint.
125
+ * @param programId - Token program ID (defaults to TOKEN_PROGRAM_ID).
126
+ * @returns The derived ATA public key.
127
+ *
128
+ * @category Utils
129
+ * @since v0.6.0
130
+ */
131
+ export function findATA(owner, mint, programId = new PublicKey("TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA")) {
132
+ const ASSOCIATED_TOKEN_PROGRAM_ID = new PublicKey("ATokenGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL");
133
+ const [ata] = PublicKey.findProgramAddressSync([owner.toBuffer(), programId.toBuffer(), mint.toBuffer()], ASSOCIATED_TOKEN_PROGRAM_ID);
134
+ return ata;
135
+ }
136
+ // ═══════════════════════════════════════════════════════════════════
137
+ // Error Classification
138
+ // ═══════════════════════════════════════════════════════════════════
139
+ /**
140
+ * Anchor error code to friendly message mapping.
141
+ * Converts cryptic Anchor errors into actionable messages.
142
+ */
143
+ const ANCHOR_ERROR_MESSAGES = {
144
+ 6000: "Agent already registered for this wallet",
145
+ 6001: "Agent not found — register first",
146
+ 6002: "Name exceeds maximum length (64 bytes)",
147
+ 6003: "Description exceeds maximum length (256 bytes)",
148
+ 6004: "Too many capabilities (max 10)",
149
+ 6005: "Too many pricing tiers (max 5)",
150
+ 6006: "Too many protocols (max 5)",
151
+ 6007: "Feedback score out of range (0-1000)",
152
+ 6008: "Unauthorized — only the agent owner can perform this action",
153
+ 6009: "Escrow expired",
154
+ 6010: "Insufficient escrow balance",
155
+ 6011: "Max calls exceeded",
156
+ 6012: "Invalid settlement — calls must be > 0",
157
+ 6013: "Escrow not empty — withdraw balance before closing",
158
+ 6014: "Invalid token program — make sure you passed TOKEN_PROGRAM_ID as the 3rd remaining account",
159
+ 6015: "Vault already initialized",
160
+ 6016: "Session already exists",
161
+ 6017: "Session closed — cannot write to closed session",
162
+ 6018: "Data exceeds maximum write size (750 bytes)",
163
+ 6019: "Ring buffer overflow — seal before writing more",
164
+ };
165
+ /**
166
+ * @name classifyAnchorError
167
+ * @description Convert an Anchor error code into a friendly, actionable message.
168
+ *
169
+ * @param errorCode - The numeric Anchor error code.
170
+ * @returns A human-readable error message, or a generic message for unknown codes.
171
+ *
172
+ * @category Utils
173
+ * @since v0.6.0
174
+ *
175
+ * @example
176
+ * ```ts
177
+ * try {
178
+ * await client.escrow.create(...);
179
+ * } catch (err) {
180
+ * const code = extractAnchorErrorCode(err);
181
+ * if (code !== null) {
182
+ * console.error(classifyAnchorError(code));
183
+ * }
184
+ * }
185
+ * ```
186
+ */
187
+ export function classifyAnchorError(errorCode) {
188
+ return (ANCHOR_ERROR_MESSAGES[errorCode] ??
189
+ `Unknown SAP program error (code ${errorCode}). Check the IDL for details.`);
190
+ }
191
+ /**
192
+ * @name extractAnchorErrorCode
193
+ * @description Attempt to extract an Anchor error code from an Error object.
194
+ *
195
+ * Anchor errors typically have the structure `{ code: number, msg: string }`.
196
+ * This function handles both the direct `error.code` pattern and the
197
+ * `error.error.errorCode.number` nested pattern.
198
+ *
199
+ * @param err - The caught error object.
200
+ * @returns The numeric error code, or `null` if not an Anchor error.
201
+ *
202
+ * @category Utils
203
+ * @since v0.6.0
204
+ */
205
+ export function extractAnchorErrorCode(err) {
206
+ if (err == null || typeof err !== "object")
207
+ return null;
208
+ // Direct code property
209
+ const direct = err.code;
210
+ if (typeof direct === "number")
211
+ return direct;
212
+ // Nested Anchor format: error.error.errorCode.number
213
+ const nested = err.error;
214
+ if (nested && typeof nested === "object") {
215
+ const errorCode = nested.errorCode;
216
+ if (errorCode && typeof errorCode === "object") {
217
+ const num = errorCode.number;
218
+ if (typeof num === "number")
219
+ return num;
220
+ }
221
+ }
222
+ // AnchorError pattern: { logs: [...], error: { errorCode: { number } } }
223
+ const msg = err.message;
224
+ if (typeof msg === "string") {
225
+ const match = msg.match(/custom program error:\s*0x([0-9a-fA-F]+)/);
226
+ if (match?.[1])
227
+ return parseInt(match[1], 16);
228
+ }
229
+ return null;
230
+ }
231
+ //# sourceMappingURL=rpc-strategy.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rpc-strategy.js","sourceRoot":"","sources":["../../../src/utils/rpc-strategy.ts"],"names":[],"mappings":"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"}