@continuumdao/ctm-mpc-defi 0.1.4 → 0.2.1

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 (94) hide show
  1. package/README.md +20 -78
  2. package/dist/agent/catalog.cjs +1388 -144
  3. package/dist/agent/catalog.cjs.map +1 -1
  4. package/dist/agent/catalog.d.ts +881 -17
  5. package/dist/agent/catalog.js +1327 -145
  6. package/dist/agent/catalog.js.map +1 -1
  7. package/dist/agent/skills/aave-v4/SKILL.md +43 -0
  8. package/dist/agent/skills/curve-dao/SKILL.md +12 -0
  9. package/dist/agent/skills/ethena/SKILL.md +10 -0
  10. package/dist/agent/skills/euler-v2/SKILL.md +10 -0
  11. package/dist/agent/skills/lido/SKILL.md +22 -0
  12. package/dist/agent/skills/maple-syrup/SKILL.md +10 -0
  13. package/dist/agent/skills/sky/SKILL.md +10 -0
  14. package/dist/agent/skills/uniswap-v4/SKILL.md +22 -0
  15. package/dist/chains/evm/index.cjs +27 -213
  16. package/dist/chains/evm/index.cjs.map +1 -1
  17. package/dist/chains/evm/index.d.ts +15 -25
  18. package/dist/chains/evm/index.js +21 -199
  19. package/dist/chains/evm/index.js.map +1 -1
  20. package/dist/chains/near/index.d.ts +1 -1
  21. package/dist/chains/solana/index.d.ts +1 -1
  22. package/dist/core/index.cjs +8 -110
  23. package/dist/core/index.cjs.map +1 -1
  24. package/dist/core/index.d.ts +5 -39
  25. package/dist/core/index.js +6 -100
  26. package/dist/core/index.js.map +1 -1
  27. package/dist/{envelope-CcE5Cz_q.d.ts → envelope-CpBUh9eP.d.ts} +1 -1
  28. package/dist/index.cjs +238 -1184
  29. package/dist/index.cjs.map +1 -1
  30. package/dist/index.d.ts +7 -10
  31. package/dist/index.js +227 -1156
  32. package/dist/index.js.map +1 -1
  33. package/dist/protocols/evm/aave-v4/index.cjs +1710 -0
  34. package/dist/protocols/evm/aave-v4/index.cjs.map +1 -0
  35. package/dist/protocols/evm/aave-v4/index.d.ts +499 -0
  36. package/dist/protocols/evm/aave-v4/index.js +1666 -0
  37. package/dist/protocols/evm/aave-v4/index.js.map +1 -0
  38. package/dist/protocols/evm/curve-dao/index.cjs +24 -124
  39. package/dist/protocols/evm/curve-dao/index.cjs.map +1 -1
  40. package/dist/protocols/evm/curve-dao/index.d.ts +3 -4
  41. package/dist/protocols/evm/curve-dao/index.js +15 -115
  42. package/dist/protocols/evm/curve-dao/index.js.map +1 -1
  43. package/dist/protocols/evm/ethena/index.cjs +853 -0
  44. package/dist/protocols/evm/ethena/index.cjs.map +1 -0
  45. package/dist/protocols/evm/ethena/index.d.ts +160 -0
  46. package/dist/protocols/evm/ethena/index.js +831 -0
  47. package/dist/protocols/evm/ethena/index.js.map +1 -0
  48. package/dist/protocols/evm/euler-v2/index.cjs +1585 -0
  49. package/dist/protocols/evm/euler-v2/index.cjs.map +1 -0
  50. package/dist/protocols/evm/euler-v2/index.d.ts +316 -0
  51. package/dist/protocols/evm/euler-v2/index.js +1560 -0
  52. package/dist/protocols/evm/euler-v2/index.js.map +1 -0
  53. package/dist/protocols/evm/lido/index.cjs +839 -0
  54. package/dist/protocols/evm/lido/index.cjs.map +1 -0
  55. package/dist/protocols/evm/lido/index.d.ts +119 -0
  56. package/dist/protocols/evm/lido/index.js +814 -0
  57. package/dist/protocols/evm/lido/index.js.map +1 -0
  58. package/dist/protocols/evm/maple/index.cjs +619 -0
  59. package/dist/protocols/evm/maple/index.cjs.map +1 -0
  60. package/dist/protocols/evm/maple/index.d.ts +108 -0
  61. package/dist/protocols/evm/maple/index.js +605 -0
  62. package/dist/protocols/evm/maple/index.js.map +1 -0
  63. package/dist/protocols/evm/sky/index.cjs +1259 -0
  64. package/dist/protocols/evm/sky/index.cjs.map +1 -0
  65. package/dist/protocols/evm/sky/index.d.ts +217 -0
  66. package/dist/protocols/evm/sky/index.js +1234 -0
  67. package/dist/protocols/evm/sky/index.js.map +1 -0
  68. package/dist/protocols/evm/uniswap-v4/index.cjs +423 -658
  69. package/dist/protocols/evm/uniswap-v4/index.cjs.map +1 -1
  70. package/dist/protocols/evm/uniswap-v4/index.d.ts +3 -4
  71. package/dist/protocols/evm/uniswap-v4/index.js +422 -657
  72. package/dist/protocols/evm/uniswap-v4/index.js.map +1 -1
  73. package/dist/{registry-oMKlO_5z.d.ts → registry-Bv5o37_w.d.ts} +1 -1
  74. package/dist/{types-Ce2qNHai.d.cts → types-BfjWdw1j.d.ts} +3 -1
  75. package/dist/{types-5u863Fd9.d.ts → types-DUeNJLr9.d.ts} +1 -1
  76. package/package.json +43 -8
  77. package/dist/agent/catalog.d.cts +0 -195
  78. package/dist/chains/evm/index.d.cts +0 -62
  79. package/dist/chains/near/index.d.cts +0 -37
  80. package/dist/chains/solana/index.d.cts +0 -40
  81. package/dist/core/index.d.cts +0 -43
  82. package/dist/envelope-DYDPnrHZ.d.cts +0 -35
  83. package/dist/index.d.cts +0 -15
  84. package/dist/keygen-CfNp8yKJ.d.cts +0 -9
  85. package/dist/keygen-DsINazx8.d.ts +0 -9
  86. package/dist/nodeRead-BnmSaMGO.d.cts +0 -8
  87. package/dist/nodeRead-BnmSaMGO.d.ts +0 -8
  88. package/dist/protocols/evm/curve-dao/index.d.cts +0 -147
  89. package/dist/protocols/evm/uniswap-v4/index.d.cts +0 -324
  90. package/dist/registry-BwZoE668.d.cts +0 -8
  91. package/dist/txParams-BC7ogvdR.d.cts +0 -19
  92. package/dist/txParams-BC7ogvdR.d.ts +0 -19
  93. package/dist/types-B8idm_gu.d.cts +0 -34
  94. package/dist/types-Ce2qNHai.d.ts +0 -57
@@ -0,0 +1,1585 @@
1
+ 'use strict';
2
+
3
+ var viem = require('viem');
4
+ var continuumNodeSdk = require('@continuumdao/continuum-node-sdk');
5
+
6
+ // src/core/registry.ts
7
+ var modules = [];
8
+ function registerProtocolModule(mod) {
9
+ const existing = modules.findIndex((m) => m.id === mod.id);
10
+ if (existing >= 0) {
11
+ modules[existing] = mod;
12
+ } else {
13
+ modules.push(mod);
14
+ }
15
+ }
16
+ var EULER_SAME_ASSET_BORROW_PROTOCOL_HEADROOM_BPS = 50n;
17
+ var EULER_SAME_ASSET_BORROW_APPROVAL_BUFFER_BPS = 100n;
18
+ var EULER_SAME_ASSET_BORROW_MAX_ROUNDS = 48;
19
+ var EULER_SAME_ASSET_BORROW_RATIO_STOP_EPS_BPS = 15n;
20
+ function eulerBorrowAndCollateralSameAsset(row) {
21
+ const c = (row.collateralAssetAddressLower ?? "").trim().toLowerCase();
22
+ const b = (row.borrowAssetAddressLower ?? "").trim().toLowerCase();
23
+ if (!c.startsWith("0x") || !b.startsWith("0x")) return false;
24
+ try {
25
+ return viem.getAddress(c).toLowerCase() === viem.getAddress(b).toLowerCase();
26
+ } catch {
27
+ return false;
28
+ }
29
+ }
30
+ function planSameAssetLeveragedBorrows(args) {
31
+ const { initialCollateralWei: C0wei, oneBorrowUnitUoAWei, borrowDecimals, maxNewBorrowWei } = args;
32
+ const targetRaw = Number(args.targetLtvBps);
33
+ const maxLtvBps = Number(args.maxLtvBps);
34
+ if (C0wei <= 0n || oneBorrowUnitUoAWei <= 0n || maxNewBorrowWei <= 0n) return null;
35
+ if (!Number.isFinite(targetRaw) || !(targetRaw > 0) || targetRaw >= 1e4) return null;
36
+ if (!Number.isFinite(maxLtvBps) || !(maxLtvBps > 0) || maxLtvBps > 1e4) return null;
37
+ const unit = 10n ** BigInt(borrowDecimals);
38
+ const effectiveMax = maxLtvBps > Number(EULER_SAME_ASSET_BORROW_PROTOCOL_HEADROOM_BPS) ? maxLtvBps - Number(EULER_SAME_ASSET_BORROW_PROTOCOL_HEADROOM_BPS) : maxLtvBps;
39
+ let targetBps = Math.min(targetRaw, effectiveMax);
40
+ if (targetBps <= 0) return null;
41
+ const workingCeiling = Math.max(1, effectiveMax - Number(EULER_SAME_ASSET_BORROW_RATIO_STOP_EPS_BPS));
42
+ targetBps = Math.min(targetBps, workingCeiling);
43
+ const uoaFromBorrowWei = (w) => {
44
+ if (w <= 0n) return 0n;
45
+ return w * oneBorrowUnitUoAWei / unit;
46
+ };
47
+ let C = uoaFromBorrowWei(C0wei);
48
+ let D = 0n;
49
+ if (C <= 0n) return null;
50
+ const borrowWeis = [];
51
+ let liquidityLeft = maxNewBorrowWei;
52
+ const targetBn = BigInt(targetBps);
53
+ const stopBn = targetBn - BigInt(Number(EULER_SAME_ASSET_BORROW_RATIO_STOP_EPS_BPS));
54
+ const capBn = BigInt(effectiveMax);
55
+ for (let i = 0; i < EULER_SAME_ASSET_BORROW_MAX_ROUNDS; i++) {
56
+ if (C <= 0n) break;
57
+ const ratioBps = D * 10000n / C;
58
+ if (ratioBps >= stopBn) break;
59
+ const headroomUoa = capBn * C / 10000n - D;
60
+ if (headroomUoa <= 0n) break;
61
+ const den = 10000n - targetBn;
62
+ if (den <= 0n) break;
63
+ let bUoa = (targetBn * C - 10000n * D) / den;
64
+ if (bUoa <= 0n) break;
65
+ if (bUoa > headroomUoa) bUoa = headroomUoa;
66
+ let borrowWei = bUoa * unit / oneBorrowUnitUoAWei;
67
+ if (borrowWei <= 0n) break;
68
+ if (borrowWei > liquidityLeft) borrowWei = liquidityLeft;
69
+ if (borrowWei <= 0n) break;
70
+ const bUoaActual = uoaFromBorrowWei(borrowWei);
71
+ borrowWeis.push(borrowWei);
72
+ D += bUoaActual;
73
+ C += bUoaActual;
74
+ liquidityLeft -= borrowWei;
75
+ }
76
+ if (borrowWeis.length === 0) return null;
77
+ const totalBorrowWei = borrowWeis.reduce((a, w) => a + w, 0n);
78
+ const projectedFinalLtvBps = C > 0n ? D * 10000n / C : 0n;
79
+ return { borrowWeis, totalBorrowWei, projectedFinalLtvBps };
80
+ }
81
+ function eulerSameAssetTotalCollateralPullWei(args) {
82
+ let s = args.initialCollateralWei;
83
+ for (const w of args.loopBorrowWeis) s += w;
84
+ return s;
85
+ }
86
+ function eulerSameAssetApproveAmountWithBuffer(args) {
87
+ const extra = args.totalPullWei * EULER_SAME_ASSET_BORROW_APPROVAL_BUFFER_BPS / 10000n;
88
+ return args.totalPullWei + extra + 1n;
89
+ }
90
+
91
+ // src/core/purpose.ts
92
+ function mergePurposeText(purposeText, purposeSuffix) {
93
+ const t = purposeText.trim();
94
+ const suffix = (purposeSuffix ?? "").trim();
95
+ if (!suffix) return t;
96
+ return t ? `${t}
97
+
98
+ ${suffix}` : suffix;
99
+ }
100
+
101
+ // src/core/envelope.ts
102
+ function finalizeMultisign(input) {
103
+ const { keyGen, destinationChainID, legs } = input;
104
+ if (legs.length === 0) {
105
+ throw new Error("finalizeMultisign requires at least one leg");
106
+ }
107
+ const ph = (keyGen.pubkeyhex ?? "").trim();
108
+ if (!ph) throw new Error("keyGen pubKey (pubkeyhex) is required");
109
+ const keyList = keyGen.keylist ?? [];
110
+ const clientId = continuumNodeSdk.getClientIdFromKeyGenResult(keyGen);
111
+ const first = legs[0];
112
+ const messageHashes = legs.map((l) => l.msgHash);
113
+ const messageRawBatch = legs.map((l) => l.msgRaw);
114
+ const batchMeta = legs.map((l) => ({
115
+ destinationAddress: l.destinationAddress,
116
+ signatureText: l.signatureText,
117
+ ...l.audit
118
+ }));
119
+ const proposalTxParams = legs.map((l) => l.proposalTxParams).filter((p) => p != null && typeof p === "object");
120
+ const extraPayload = {
121
+ batchMeta,
122
+ ...input.extraJSON ?? {}
123
+ };
124
+ const extraJSON = JSON.stringify(extraPayload);
125
+ const bodyForSign = {
126
+ keyList,
127
+ pubKey: ph,
128
+ msgHash: messageHashes[0],
129
+ msgRaw: first.msgRaw,
130
+ destinationChainID,
131
+ destinationAddress: input.destinationAddress ?? first.destinationAddress,
132
+ extraJSON,
133
+ signatureText: first.signatureText,
134
+ purpose: mergePurposeText(input.purposeText, input.purposeSuffix),
135
+ ...first.feeSnapshot
136
+ };
137
+ if (legs.length > 1) {
138
+ bodyForSign.messageHashes = messageHashes;
139
+ bodyForSign.messageRawBatch = messageRawBatch;
140
+ }
141
+ if (proposalTxParams.length > 0) {
142
+ bodyForSign.proposalTxParams = proposalTxParams;
143
+ }
144
+ const valueWei = first.valueWei;
145
+ if (valueWei != null && valueWei > 0n) {
146
+ bodyForSign.value = valueWei.toString();
147
+ }
148
+ if (clientId) bodyForSign.clientId = clientId;
149
+ return { bodyForSign, messageToSign: JSON.stringify(bodyForSign) };
150
+ }
151
+ function routerSwapGasLimitFromEstimate(estimatedGas, chainGasLimit) {
152
+ if (chainGasLimit != null && Number.isFinite(chainGasLimit) && chainGasLimit > 0) {
153
+ return continuumNodeSdk.gasLimitFromEstimateAndChainConfig(estimatedGas, chainGasLimit);
154
+ }
155
+ return (estimatedGas * 12n + 9n) / 10n;
156
+ }
157
+
158
+ // src/chains/evm/buildBatch.ts
159
+ async function buildEvmMultisignBatch(args) {
160
+ const { context, steps } = args;
161
+ const {
162
+ chainId,
163
+ rpcUrl,
164
+ executorAddress,
165
+ chainDetail,
166
+ useCustomGas,
167
+ customGasChainDetails,
168
+ keyGen,
169
+ purposeText
170
+ } = context;
171
+ if (steps.length === 0) throw new Error("buildEvmMultisignBatch requires at least one step");
172
+ const ch = viem.defineChain({
173
+ id: chainId,
174
+ name: "Destination",
175
+ nativeCurrency: { decimals: 18, name: "Ether", symbol: "ETH" },
176
+ rpcUrls: { default: { http: [rpcUrl] } }
177
+ });
178
+ const publicClient = viem.createPublicClient({ chain: ch, transport: viem.http(rpcUrl) });
179
+ const feeParams = await continuumNodeSdk.fetchChainFeeParams(rpcUrl, chainId);
180
+ const legacy = Boolean(chainDetail?.legacy) || !feeParams.isEip1559;
181
+ const latestBaseFeeWei = !legacy ? (await publicClient.getBlock({ blockTag: "latest" })).baseFeePerGas ?? 0n : 0n;
182
+ const gasLimitConfig = useCustomGas && chainDetail?.gasLimit != null ? Number(chainDetail.gasLimit) : void 0;
183
+ const chainGasLimitRouter = chainDetail?.gasLimit != null && Number.isFinite(Number(chainDetail.gasLimit)) && Number(chainDetail.gasLimit) > 0 ? Number(chainDetail.gasLimit) : void 0;
184
+ const gasFeeMultiplier = useCustomGas && chainDetail?.gasMultiplier != null ? Number(chainDetail.gasMultiplier) : void 0;
185
+ const executor = viem.getAddress(executorAddress);
186
+ const baseNonce = await publicClient.getTransactionCount({ address: executor, blockTag: "pending" });
187
+ const legs = [];
188
+ for (let i = 0; i < steps.length; i++) {
189
+ const step = steps[i];
190
+ const currentNonce = baseNonce + i;
191
+ let estimatedGas;
192
+ if (args.estimateGasForStep) {
193
+ estimatedGas = await args.estimateGasForStep({ step, index: i, publicClient, executor });
194
+ } else {
195
+ try {
196
+ estimatedGas = await publicClient.estimateGas({
197
+ to: step.to,
198
+ data: step.data,
199
+ value: step.value,
200
+ account: executor
201
+ });
202
+ } catch {
203
+ estimatedGas = step.fallbackGas ?? 100000n;
204
+ }
205
+ }
206
+ let gasLimitI;
207
+ if (args.resolveGasLimit) {
208
+ gasLimitI = await args.resolveGasLimit({ step, index: i, estimatedGas, publicClient });
209
+ } else if (step.routerSwap) {
210
+ gasLimitI = routerSwapGasLimitFromEstimate(estimatedGas, chainGasLimitRouter);
211
+ } else {
212
+ gasLimitI = useCustomGas ? continuumNodeSdk.gasLimitFromEstimateAndChainConfig(estimatedGas, gasLimitConfig) : estimatedGas;
213
+ }
214
+ let proposalTxParams;
215
+ let feeSnapshot;
216
+ let serialized;
217
+ if (legacy) {
218
+ let gasPriceWei = await publicClient.getGasPrice();
219
+ if (useCustomGas && gasFeeMultiplier != null && gasFeeMultiplier > 0) {
220
+ gasPriceWei = gasPriceWei * BigInt(100 + gasFeeMultiplier) / 100n;
221
+ }
222
+ if (useCustomGas && chainDetail?.gasPrice != null && chainDetail.gasPrice > 0) {
223
+ const configured = viem.parseGwei(continuumNodeSdk.gweiToDecimalString(Number(chainDetail.gasPrice)));
224
+ if (configured > gasPriceWei) gasPriceWei = configured;
225
+ }
226
+ serialized = viem.serializeTransaction({
227
+ type: "legacy",
228
+ to: step.to,
229
+ data: step.data,
230
+ value: step.value,
231
+ gas: gasLimitI,
232
+ gasPrice: gasPriceWei,
233
+ nonce: currentNonce,
234
+ chainId
235
+ });
236
+ proposalTxParams = {
237
+ nonce: currentNonce,
238
+ gasLimit: gasLimitI.toString(),
239
+ txType: "legacy",
240
+ gasPrice: gasPriceWei.toString()
241
+ };
242
+ feeSnapshot = continuumNodeSdk.proposalTxParamsToFeeSnapshot(proposalTxParams);
243
+ } else {
244
+ const fetchedBase = feeParams.baseFeeGwei ?? 0;
245
+ const fetchedPriority = feeParams.priorityFeeGwei ?? 0;
246
+ const configuredBase = useCustomGas && chainDetail?.baseFee != null ? Number(chainDetail.baseFee) : 0;
247
+ const configuredPriority = useCustomGas && chainDetail?.priorityFee != null ? Number(chainDetail.priorityFee) : 0;
248
+ const effectiveBaseFeeGwei = Math.max(fetchedBase, configuredBase);
249
+ const effectivePriorityFeeGwei = Math.max(fetchedPriority, configuredPriority);
250
+ const baseFeeMultiplierPct = useCustomGas && chainDetail?.baseFeeMultiplier != null ? Math.max(100, Number(chainDetail.baseFeeMultiplier)) : 100;
251
+ const baseComponentGwei = effectiveBaseFeeGwei * baseFeeMultiplierPct / 100;
252
+ const maxFeePerGasGwei = baseComponentGwei + effectivePriorityFeeGwei;
253
+ let maxPriorityFeePerGas = effectivePriorityFeeGwei > 0 ? viem.parseGwei(continuumNodeSdk.gweiToDecimalString(effectivePriorityFeeGwei)) : viem.parseGwei("1");
254
+ let maxFeePerGas = viem.parseGwei(continuumNodeSdk.gweiToDecimalString(maxFeePerGasGwei));
255
+ if (useCustomGas && gasFeeMultiplier != null && gasFeeMultiplier > 0) {
256
+ maxPriorityFeePerGas = maxPriorityFeePerGas * BigInt(100 + gasFeeMultiplier) / 100n;
257
+ maxFeePerGas = maxFeePerGas * BigInt(100 + gasFeeMultiplier) / 100n;
258
+ }
259
+ ({ maxFeePerGas, maxPriorityFeePerGas } = continuumNodeSdk.alignEip1559FeesWithLatestBase(
260
+ maxFeePerGas,
261
+ maxPriorityFeePerGas,
262
+ latestBaseFeeWei
263
+ ));
264
+ serialized = viem.serializeTransaction({
265
+ type: "eip1559",
266
+ to: step.to,
267
+ data: step.data,
268
+ value: step.value,
269
+ gas: gasLimitI,
270
+ maxFeePerGas,
271
+ maxPriorityFeePerGas,
272
+ nonce: currentNonce,
273
+ chainId
274
+ });
275
+ proposalTxParams = {
276
+ nonce: currentNonce,
277
+ gasLimit: gasLimitI.toString(),
278
+ txType: "eip1559",
279
+ maxFeePerGas: maxFeePerGas.toString(),
280
+ maxPriorityFeePerGas: maxPriorityFeePerGas.toString()
281
+ };
282
+ feeSnapshot = i === 0 ? continuumNodeSdk.proposalTxParamsToFeeSnapshot(proposalTxParams) : {};
283
+ }
284
+ const h = viem.keccak256(serialized);
285
+ const msgHash = h.startsWith("0x") ? h.slice(2) : h;
286
+ const batchMetaExtra = args.buildBatchMeta({ step, index: i, gasLimit: gasLimitI });
287
+ legs.push({
288
+ msgHash,
289
+ msgRaw: i === 0 && args.firstMsgRawNo0x != null ? args.firstMsgRawNo0x : serialized,
290
+ destinationAddress: step.to,
291
+ signatureText: typeof batchMetaExtra.signatureText === "string" ? batchMetaExtra.signatureText : JSON.stringify(batchMetaExtra.signatureText ?? {}),
292
+ audit: batchMetaExtra,
293
+ feeSnapshot: i === 0 ? feeSnapshot : {},
294
+ proposalTxParams,
295
+ valueWei: i === 0 ? step.value : void 0
296
+ });
297
+ if (i === 0 && args.firstMsgRawNo0x != null) {
298
+ legs[0].msgRaw = args.firstMsgRawNo0x;
299
+ }
300
+ }
301
+ const extraJSON = {};
302
+ if (useCustomGas && customGasChainDetails && Object.keys(customGasChainDetails).length > 0) {
303
+ extraJSON.customGasChainDetails = customGasChainDetails;
304
+ }
305
+ const result = finalizeMultisign({
306
+ keyGen,
307
+ purposeText,
308
+ purposeSuffix: args.purposeSuffix,
309
+ destinationChainID: String(chainId),
310
+ destinationAddress: args.destinationAddress ?? steps[0].to,
311
+ legs,
312
+ extraJSON: Object.keys(extraJSON).length > 0 ? extraJSON : void 0
313
+ });
314
+ const pv = args.payableValueWei;
315
+ if (pv != null && pv > 0n) {
316
+ result.bodyForSign.value = pv.toString();
317
+ }
318
+ return result;
319
+ }
320
+
321
+ // src/protocols/evm/euler-v2/vaultWithdrawMultisign.ts
322
+ var EULER_V2_VAULT_WITHDRAW_FALLBACK_GAS = 900000n;
323
+ var erc20DecimalsAbi = viem.parseAbi(["function decimals() view returns (uint8)"]);
324
+ var erc4626AssetAbi = viem.parseAbi(["function asset() view returns (address)"]);
325
+ var erc4626MaxWithdrawAbi = viem.parseAbi(["function maxWithdraw(address owner) view returns (uint256)"]);
326
+ var erc4626WithdrawAbi = viem.parseAbi([
327
+ "function withdraw(uint256 assets, address receiver, address owner) returns (uint256 shares)"
328
+ ]);
329
+ async function fetchEulerVaultUnderlyingMeta(args) {
330
+ const ch = viem.defineChain({
331
+ id: args.chainId,
332
+ name: "Destination",
333
+ nativeCurrency: { decimals: 18, name: "Ether", symbol: "ETH" },
334
+ rpcUrls: { default: { http: [args.rpcUrl] } }
335
+ });
336
+ const publicClient = viem.createPublicClient({ chain: ch, transport: viem.http(args.rpcUrl) });
337
+ const assetAddrRaw = await publicClient.readContract({
338
+ address: args.evault,
339
+ abi: erc4626AssetAbi,
340
+ functionName: "asset"
341
+ });
342
+ const assetAddr = viem.getAddress(assetAddrRaw);
343
+ const d = await publicClient.readContract({
344
+ address: assetAddr,
345
+ abi: erc20DecimalsAbi,
346
+ functionName: "decimals"
347
+ });
348
+ const n = typeof d === "bigint" ? Number(d) : Number(d);
349
+ const decimals = !Number.isFinite(n) || n < 0 || n > 36 ? 18 : n;
350
+ return { asset: assetAddr, decimals };
351
+ }
352
+ function clampEulerUnderlyingDecimalsForEulerUi(args) {
353
+ const a = args.underlyingAssetLower.trim().toLowerCase();
354
+ let d = args.fetchedDecimals;
355
+ if (args.wrappedGasAliasesLower.has(a)) {
356
+ return Math.max(d, 18);
357
+ }
358
+ if (d >= 18) return d;
359
+ const label = `${args.underlyingSymbol} ${args.marketName}`;
360
+ const ethish = /\b(WETH|wstETH|stETH|rETH|weETH|eETH)\b/i.test(label) || /(^|[^A-Z0-9])ETH([^A-Z0-9]|$)/i.test(label);
361
+ if (ethish) return 18;
362
+ return d;
363
+ }
364
+ async function fetchEulerVaultAssetDecimals(args) {
365
+ const m = await fetchEulerVaultUnderlyingMeta(args);
366
+ return m.decimals;
367
+ }
368
+ async function fetchEulerVaultMaxWithdrawWei(args) {
369
+ const ch = viem.defineChain({
370
+ id: args.chainId,
371
+ name: "Destination",
372
+ nativeCurrency: { decimals: 18, name: "Ether", symbol: "ETH" },
373
+ rpcUrls: { default: { http: [args.rpcUrl] } }
374
+ });
375
+ const publicClient = viem.createPublicClient({ chain: ch, transport: viem.http(args.rpcUrl) });
376
+ return publicClient.readContract({
377
+ address: args.evault,
378
+ abi: erc4626MaxWithdrawAbi,
379
+ functionName: "maxWithdraw",
380
+ args: [args.owner]
381
+ });
382
+ }
383
+ var eulerVaultSharesCashAbi = viem.parseAbi([
384
+ "function balanceOf(address account) view returns (uint256)",
385
+ "function convertToAssets(uint256 shares) view returns (uint256)",
386
+ "function cash() view returns (uint256)"
387
+ ]);
388
+ async function eulerDirectWithdrawSimulatesOk(args) {
389
+ const withdrawData = viem.encodeFunctionData({
390
+ abi: erc4626WithdrawAbi,
391
+ functionName: "withdraw",
392
+ args: [args.assetsWei, args.receiver, args.vaultShareOwner]
393
+ });
394
+ try {
395
+ await args.publicClient.call({
396
+ account: args.txSender,
397
+ to: args.evault,
398
+ data: withdrawData,
399
+ gas: 8000000n
400
+ });
401
+ return true;
402
+ } catch {
403
+ return false;
404
+ }
405
+ }
406
+ async function eulerLendEarnEffectiveMaxWeiBySimulation(args) {
407
+ const ch = viem.defineChain({
408
+ id: args.chainId,
409
+ name: "EulerLendRedeem",
410
+ nativeCurrency: { decimals: 18, name: "Ether", symbol: "ETH" },
411
+ rpcUrls: { default: { http: [args.rpcUrl] } }
412
+ });
413
+ const publicClient = viem.createPublicClient({ chain: ch, transport: viem.http(args.rpcUrl) });
414
+ let shareBal;
415
+ let vaultCash;
416
+ try {
417
+ ;
418
+ [shareBal, vaultCash] = await Promise.all([
419
+ publicClient.readContract({
420
+ address: args.evault,
421
+ abi: eulerVaultSharesCashAbi,
422
+ functionName: "balanceOf",
423
+ args: [args.vaultShareOwner]
424
+ }),
425
+ publicClient.readContract({
426
+ address: args.evault,
427
+ abi: eulerVaultSharesCashAbi,
428
+ functionName: "cash"
429
+ })
430
+ ]);
431
+ } catch {
432
+ return 0n;
433
+ }
434
+ if (shareBal === 0n) return 0n;
435
+ let assetsOwned;
436
+ try {
437
+ assetsOwned = await publicClient.readContract({
438
+ address: args.evault,
439
+ abi: eulerVaultSharesCashAbi,
440
+ functionName: "convertToAssets",
441
+ args: [shareBal]
442
+ });
443
+ } catch {
444
+ return 0n;
445
+ }
446
+ const hi = vaultCash <= 0n ? 0n : vaultCash <= assetsOwned ? vaultCash : assetsOwned;
447
+ if (hi === 0n) return 0n;
448
+ let lo = 0n;
449
+ let hiProbe = hi;
450
+ let best = 0n;
451
+ while (lo <= hiProbe) {
452
+ const mid = lo + (hiProbe - lo) / 2n;
453
+ const ok = await eulerDirectWithdrawSimulatesOk({
454
+ publicClient,
455
+ evault: args.evault,
456
+ vaultShareOwner: args.vaultShareOwner,
457
+ receiver: args.receiver,
458
+ txSender: args.txSender,
459
+ assetsWei: mid
460
+ });
461
+ if (ok) {
462
+ best = mid;
463
+ lo = mid + 1n;
464
+ } else {
465
+ hiProbe = mid - 1n;
466
+ }
467
+ }
468
+ return best;
469
+ }
470
+ async function fetchEulerLendEarnVaultEffectiveMaxWithdrawWei(args) {
471
+ const rpc = args.rpcUrl.trim();
472
+ if (!rpc) return 0n;
473
+ const evault = viem.getAddress(args.evault);
474
+ const vaultShareOwner = viem.getAddress(args.vaultShareOwner);
475
+ const txSender = viem.getAddress(args.txSender);
476
+ const receiver = txSender;
477
+ const std = await fetchEulerVaultMaxWithdrawWei({
478
+ rpcUrl: rpc,
479
+ chainId: args.chainId,
480
+ evault,
481
+ owner: vaultShareOwner
482
+ });
483
+ if (std > 0n) return std;
484
+ return eulerLendEarnEffectiveMaxWeiBySimulation({
485
+ rpcUrl: rpc,
486
+ chainId: args.chainId,
487
+ evault,
488
+ vaultShareOwner,
489
+ receiver,
490
+ txSender
491
+ });
492
+ }
493
+ async function buildEvmMultisignBodyEulerV2VaultWithdraw(args) {
494
+ const evault = viem.getAddress(args.evault);
495
+ const receiver = viem.getAddress(args.owner);
496
+ const shareOwner = viem.getAddress(args.vaultShareOwner ?? args.owner);
497
+ const executor = viem.getAddress(args.executorAddress);
498
+ const dec = await fetchEulerVaultAssetDecimals({ rpcUrl: args.rpcUrl, chainId: args.chainId, evault });
499
+ const amountWei = viem.parseUnits(args.amountHuman, dec);
500
+ if (amountWei === 0n) throw new Error("Withdraw amount is zero after converting with token decimals.");
501
+ const maxW = await fetchEulerLendEarnVaultEffectiveMaxWithdrawWei({
502
+ rpcUrl: args.rpcUrl,
503
+ chainId: args.chainId,
504
+ evault,
505
+ vaultShareOwner: shareOwner,
506
+ txSender: executor
507
+ });
508
+ if (amountWei > maxW) {
509
+ throw new Error("Withdraw amount exceeds simulated max redeem for this vault (LTV / vault cash). Try a smaller amount.");
510
+ }
511
+ const withdrawData = viem.encodeFunctionData({
512
+ abi: erc4626WithdrawAbi,
513
+ functionName: "withdraw",
514
+ args: [amountWei, receiver, shareOwner]
515
+ });
516
+ const vaultLabel = (args.vaultMarketLabel ?? "").trim() || "Euler vault";
517
+ const purposeSuffix = `Euler v2: 1-tx \u2014 eVault.withdraw (${args.amountHuman} underlying) from isolated vault "${vaultLabel}" (ERC-4626).`;
518
+ const evmSteps = [
519
+ {
520
+ to: evault,
521
+ data: withdrawData,
522
+ value: 0n,
523
+ fallbackGas: EULER_V2_VAULT_WITHDRAW_FALLBACK_GAS
524
+ }
525
+ ];
526
+ const firstDataNo0x = withdrawData.startsWith("0x") ? withdrawData.slice(2) : withdrawData;
527
+ return buildEvmMultisignBatch({
528
+ context: {
529
+ chainCategory: "evm",
530
+ keyGen: args.keyGen,
531
+ purposeText: args.purposeText,
532
+ chainId: args.chainId,
533
+ rpcUrl: args.rpcUrl,
534
+ executorAddress: executor,
535
+ chainDetail: args.chainDetail,
536
+ useCustomGas: args.useCustomGas,
537
+ customGasChainDetails: args.customGasChainDetails
538
+ },
539
+ steps: evmSteps,
540
+ purposeSuffix,
541
+ firstMsgRawNo0x: firstDataNo0x,
542
+ destinationAddress: evault,
543
+ buildBatchMeta: ({ gasLimit }) => ({
544
+ signatureText: JSON.stringify({
545
+ kind: "EulerV2",
546
+ name: "EVault.withdraw",
547
+ function: "withdraw(uint256 assets, address receiver, address owner)",
548
+ evault,
549
+ receiver,
550
+ owner: shareOwner,
551
+ vaultMarket: vaultLabel,
552
+ amountHuman: args.amountHuman
553
+ }),
554
+ evm: { type: "euler_v2_vault_withdraw", version: 1, chainId: String(args.chainId) },
555
+ eulerV2: {
556
+ vaultMarket: vaultLabel,
557
+ amountHuman: args.amountHuman,
558
+ evault,
559
+ owner: shareOwner,
560
+ gasBuildWithdraw: { baseGasUnits: gasLimit.toString() }
561
+ }
562
+ })
563
+ });
564
+ }
565
+
566
+ // src/protocols/evm/euler-v2/borrowCollateralMaxWithdrawWei.ts
567
+ var erc4626WithdrawAbi2 = viem.parseAbi([
568
+ "function withdraw(uint256 assets, address receiver, address owner) returns (uint256 shares)"
569
+ ]);
570
+ var evcBatchAbi = viem.parseAbi([
571
+ "function batch((address targetContract, address onBehalfOfAccount, uint256 value, bytes data)[])"
572
+ ]);
573
+ var eulerCollateralVaultReadAbi = viem.parseAbi([
574
+ "function balanceOf(address account) view returns (uint256)",
575
+ "function convertToAssets(uint256 shares) view returns (uint256)",
576
+ "function cash() view returns (uint256)"
577
+ ]);
578
+ async function evcWithdrawSimulatesOk(args) {
579
+ const withdrawData = viem.encodeFunctionData({
580
+ abi: erc4626WithdrawAbi2,
581
+ functionName: "withdraw",
582
+ args: [args.assetsWei, args.caller, args.subAccount]
583
+ });
584
+ const batchData = viem.encodeFunctionData({
585
+ abi: evcBatchAbi,
586
+ functionName: "batch",
587
+ args: [
588
+ [
589
+ {
590
+ targetContract: args.collateralVault,
591
+ onBehalfOfAccount: args.subAccount,
592
+ value: 0n,
593
+ data: withdrawData
594
+ }
595
+ ]
596
+ ]
597
+ });
598
+ try {
599
+ await args.publicClient.call({
600
+ account: args.caller,
601
+ to: args.evc,
602
+ data: batchData,
603
+ gas: 8000000n
604
+ });
605
+ return true;
606
+ } catch {
607
+ return false;
608
+ }
609
+ }
610
+ async function collateralMaxWithdrawBySimulation(args) {
611
+ const ch = viem.defineChain({
612
+ id: args.chainId,
613
+ name: "EulerColMaxW",
614
+ nativeCurrency: { decimals: 18, name: "Ether", symbol: "ETH" },
615
+ rpcUrls: { default: { http: [args.rpcUrl] } }
616
+ });
617
+ const publicClient = viem.createPublicClient({ chain: ch, transport: viem.http(args.rpcUrl) });
618
+ let shareBal;
619
+ let assetsOwned;
620
+ let vaultCash;
621
+ try {
622
+ ;
623
+ [shareBal, vaultCash] = await Promise.all([
624
+ publicClient.readContract({
625
+ address: args.collateralVault,
626
+ abi: eulerCollateralVaultReadAbi,
627
+ functionName: "balanceOf",
628
+ args: [args.subAccount]
629
+ }),
630
+ publicClient.readContract({
631
+ address: args.collateralVault,
632
+ abi: eulerCollateralVaultReadAbi,
633
+ functionName: "cash"
634
+ })
635
+ ]);
636
+ } catch {
637
+ return 0n;
638
+ }
639
+ if (shareBal === 0n) return 0n;
640
+ try {
641
+ assetsOwned = await publicClient.readContract({
642
+ address: args.collateralVault,
643
+ abi: eulerCollateralVaultReadAbi,
644
+ functionName: "convertToAssets",
645
+ args: [shareBal]
646
+ });
647
+ } catch {
648
+ return 0n;
649
+ }
650
+ const hi = vaultCash <= 0n ? 0n : vaultCash <= assetsOwned ? vaultCash : assetsOwned;
651
+ if (hi === 0n) return 0n;
652
+ let lo = 0n;
653
+ let hiProbe = hi;
654
+ let best = 0n;
655
+ while (lo <= hiProbe) {
656
+ const mid = lo + (hiProbe - lo) / 2n;
657
+ const ok = await evcWithdrawSimulatesOk({
658
+ publicClient,
659
+ evc: args.evc,
660
+ collateralVault: args.collateralVault,
661
+ subAccount: args.subAccount,
662
+ caller: args.caller,
663
+ assetsWei: mid
664
+ });
665
+ if (ok) {
666
+ best = mid;
667
+ lo = mid + 1n;
668
+ } else {
669
+ hiProbe = mid - 1n;
670
+ }
671
+ }
672
+ return best;
673
+ }
674
+ async function fetchEulerBorrowCollateralMaxWithdrawAssetsWei(args) {
675
+ const rpcUrl = args.rpcUrl.trim();
676
+ if (!rpcUrl) return 0n;
677
+ const evc = viem.getAddress(args.evc);
678
+ const collateralVault = viem.getAddress(args.collateralVault);
679
+ const subAccount = viem.getAddress(args.subAccount);
680
+ const caller = viem.getAddress(args.caller);
681
+ const stdMax = await fetchEulerVaultMaxWithdrawWei({
682
+ rpcUrl,
683
+ chainId: args.chainId,
684
+ evault: collateralVault,
685
+ owner: subAccount
686
+ });
687
+ if (stdMax > 0n) return stdMax;
688
+ return collateralMaxWithdrawBySimulation({
689
+ rpcUrl,
690
+ chainId: args.chainId,
691
+ evc,
692
+ collateralVault,
693
+ subAccount,
694
+ caller
695
+ });
696
+ }
697
+ var EULER_V2_ISOLATED_VAULT_DEPOSIT_FALLBACK_GAS = 950000n;
698
+ var EULER_VAULT_DEPOSIT_ESTIMATE_FALLBACK = EULER_V2_ISOLATED_VAULT_DEPOSIT_FALLBACK_GAS;
699
+ var EULER_ERC20_APPROVE_FALLBACK = 100000n;
700
+ var EULER_WETH_DEPOSIT_FALLBACK = 120000n;
701
+ var wethDepositAbi = viem.parseAbi(["function deposit() payable"]);
702
+ var erc20AllowanceAbi = viem.parseAbi([
703
+ "function allowance(address owner, address spender) view returns (uint256)",
704
+ "function decimals() view returns (uint8)"
705
+ ]);
706
+ var erc20ApproveAbi = viem.parseAbi(["function approve(address spender, uint256 amount) returns (bool)"]);
707
+ var erc4626DepositAbi = viem.parseAbi([
708
+ "function deposit(uint256 assets, address receiver) returns (uint256 shares)"
709
+ ]);
710
+ async function buildEvmMultisignBodyEulerV2IsolatedLendDepositBatch(args) {
711
+ const asset = viem.getAddress(args.underlying);
712
+ const evault = viem.getAddress(args.evault);
713
+ const weth = viem.getAddress(args.nativeWrapped);
714
+ const executor = viem.getAddress(args.executorAddress);
715
+ const receiver = viem.getAddress(args.receiver);
716
+ if (args.isNativeIn && asset.toLowerCase() !== weth.toLowerCase()) {
717
+ throw new Error("Native lend path: underlying asset must match the chain wrapped native token.");
718
+ }
719
+ const ch = viem.defineChain({
720
+ id: args.chainId,
721
+ name: "Destination",
722
+ nativeCurrency: { decimals: 18, name: "Ether", symbol: "ETH" },
723
+ rpcUrls: { default: { http: [args.rpcUrl] } }
724
+ });
725
+ const publicClient = viem.createPublicClient({ chain: ch, transport: viem.http(args.rpcUrl) });
726
+ const dec = await publicClient.readContract({
727
+ address: asset,
728
+ abi: erc20AllowanceAbi,
729
+ functionName: "decimals"
730
+ });
731
+ const amountWei = viem.parseUnits(args.amountHuman, Number(dec));
732
+ if (amountWei === 0n) throw new Error("Amount is zero after converting with token decimals.");
733
+ const steps = [];
734
+ if (args.isNativeIn) {
735
+ const dataDeposit = viem.encodeFunctionData({ abi: wethDepositAbi, functionName: "deposit", args: [] });
736
+ steps.push({ kind: "weth_deposit", to: weth, data: dataDeposit, value: amountWei });
737
+ const wethAllowance = await publicClient.readContract({
738
+ address: weth,
739
+ abi: erc20AllowanceAbi,
740
+ functionName: "allowance",
741
+ args: [executor, evault]
742
+ });
743
+ if (wethAllowance < amountWei) {
744
+ if (wethAllowance > 0n) {
745
+ const dataReset = viem.encodeFunctionData({
746
+ abi: erc20ApproveAbi,
747
+ functionName: "approve",
748
+ args: [evault, 0n]
749
+ });
750
+ steps.push({ kind: "approve", to: weth, data: dataReset, value: 0n });
751
+ }
752
+ const dataApprove = viem.encodeFunctionData({
753
+ abi: erc20ApproveAbi,
754
+ functionName: "approve",
755
+ args: [evault, amountWei]
756
+ });
757
+ steps.push({ kind: "approve", to: weth, data: dataApprove, value: 0n });
758
+ }
759
+ } else {
760
+ const currentAllowance = await publicClient.readContract({
761
+ address: asset,
762
+ abi: erc20AllowanceAbi,
763
+ functionName: "allowance",
764
+ args: [executor, evault]
765
+ });
766
+ if (currentAllowance < amountWei) {
767
+ if (currentAllowance > 0n) {
768
+ const dataReset = viem.encodeFunctionData({
769
+ abi: erc20ApproveAbi,
770
+ functionName: "approve",
771
+ args: [evault, 0n]
772
+ });
773
+ steps.push({ kind: "approve", to: asset, data: dataReset, value: 0n });
774
+ }
775
+ const dataApprove = viem.encodeFunctionData({
776
+ abi: erc20ApproveAbi,
777
+ functionName: "approve",
778
+ args: [evault, amountWei]
779
+ });
780
+ steps.push({ kind: "approve", to: asset, data: dataApprove, value: 0n });
781
+ }
782
+ }
783
+ const depositData = viem.encodeFunctionData({
784
+ abi: erc4626DepositAbi,
785
+ functionName: "deposit",
786
+ args: [amountWei, receiver]
787
+ });
788
+ steps.push({ kind: "vault_deposit", to: evault, data: depositData, value: 0n });
789
+ const vaultLabel = (args.vaultMarketLabel ?? "").trim() || "Euler vault";
790
+ const evmSteps = steps.map((s) => ({
791
+ to: s.to,
792
+ data: s.data,
793
+ value: s.value,
794
+ fallbackGas: s.kind === "weth_deposit" ? EULER_WETH_DEPOSIT_FALLBACK : s.kind === "approve" ? EULER_ERC20_APPROVE_FALLBACK : EULER_VAULT_DEPOSIT_ESTIMATE_FALLBACK
795
+ }));
796
+ const n = steps.length;
797
+ const hasWrap = args.isNativeIn;
798
+ const purposeSuffix = (() => {
799
+ if (hasWrap) {
800
+ return `Euler v2: ${n}-tx batch \u2014 wrap native to WETH (if needed), approve eVault for the exact amount, then ERC-4626 deposit into isolated vault "${vaultLabel}".`;
801
+ }
802
+ if (n === 1) {
803
+ return `Euler v2: 1-tx \u2014 eVault.deposit (allowance already sufficient) into "${vaultLabel}".`;
804
+ }
805
+ return `Euler v2: ${n}-tx batch \u2014 approve eVault for the exact amount, then ERC-4626 deposit into "${vaultLabel}".`;
806
+ })();
807
+ const firstDataNo0x = evmSteps[0].data.startsWith("0x") ? evmSteps[0].data.slice(2) : evmSteps[0].data;
808
+ return buildEvmMultisignBatch({
809
+ context: {
810
+ chainCategory: "evm",
811
+ keyGen: args.keyGen,
812
+ purposeText: args.purposeText,
813
+ chainId: args.chainId,
814
+ rpcUrl: args.rpcUrl,
815
+ executorAddress: executor,
816
+ chainDetail: args.chainDetail,
817
+ useCustomGas: args.useCustomGas,
818
+ customGasChainDetails: args.customGasChainDetails
819
+ },
820
+ steps: evmSteps,
821
+ purposeSuffix,
822
+ firstMsgRawNo0x: firstDataNo0x,
823
+ destinationAddress: steps[0].to,
824
+ buildBatchMeta: ({ index, gasLimit }) => {
825
+ const s = steps[index];
826
+ if (s.kind === "weth_deposit") {
827
+ return {
828
+ signatureText: JSON.stringify({
829
+ kind: "EulerV2",
830
+ name: "WETH.deposit",
831
+ function: "deposit()",
832
+ valueWei: amountWei.toString(),
833
+ vaultMarket: vaultLabel,
834
+ note: "Wrap native for Euler v2 isolated vault deposit (same batch)."
835
+ }),
836
+ evm: { type: "euler_v2_weth_deposit", version: 1, chainId: String(args.chainId) },
837
+ eulerV2: {
838
+ step: "weth_deposit",
839
+ vaultMarket: vaultLabel,
840
+ amountHuman: args.amountHuman,
841
+ evault,
842
+ underlying: asset
843
+ }
844
+ };
845
+ }
846
+ if (s.kind === "approve") {
847
+ return {
848
+ signatureText: JSON.stringify({
849
+ kind: "EulerV2",
850
+ name: "ERC20.approve",
851
+ to: "Euler eVault",
852
+ function: "approve(address spender, uint256 amount)",
853
+ evault,
854
+ amountHuman: args.amountHuman,
855
+ note: "Allowance for this deposit amount only (not unlimited)."
856
+ }),
857
+ evm: { type: "euler_v2_erc20_approve", version: 1, chainId: String(args.chainId) },
858
+ eulerV2: { vaultMarket: vaultLabel, amountHuman: args.amountHuman, evault, underlying: asset }
859
+ };
860
+ }
861
+ return {
862
+ signatureText: JSON.stringify({
863
+ kind: "EulerV2",
864
+ name: "EVault.deposit",
865
+ function: "deposit(uint256 assets, address receiver)",
866
+ evault,
867
+ underlying: asset,
868
+ receiver,
869
+ vaultMarket: vaultLabel,
870
+ amountHuman: args.amountHuman
871
+ }),
872
+ evm: { type: "euler_v2_vault_deposit", version: 1, chainId: String(args.chainId) },
873
+ eulerV2: {
874
+ vaultMarket: vaultLabel,
875
+ amountHuman: args.amountHuman,
876
+ evault,
877
+ underlying: asset,
878
+ receiver,
879
+ gasBuildDeposit: { baseGasUnits: gasLimit.toString() }
880
+ }
881
+ };
882
+ }
883
+ });
884
+ }
885
+ var EULER_BORROW_BATCH_FALLBACK_GAS = 2500000n;
886
+ var EULER_BORROW_BATCH_FALLBACK_GAS_PER_ROUND = 350000n;
887
+ var EULER_ERC20_APPROVE_FALLBACK2 = 100000n;
888
+ var EULER_WETH_DEPOSIT_FALLBACK2 = 120000n;
889
+ var wethDepositAbi2 = viem.parseAbi(["function deposit() payable"]);
890
+ var erc20AllowanceAbi2 = viem.parseAbi([
891
+ "function allowance(address owner, address spender) view returns (uint256)",
892
+ "function decimals() view returns (uint8)"
893
+ ]);
894
+ var erc20ApproveAbi2 = viem.parseAbi(["function approve(address spender, uint256 amount) returns (bool)"]);
895
+ var erc4626DepositAbi2 = viem.parseAbi([
896
+ "function deposit(uint256 assets, address receiver) returns (uint256 shares)"
897
+ ]);
898
+ var evaultBorrowAbi = viem.parseAbi(["function borrow(uint256 amount, address receiver) returns (uint256)"]);
899
+ var evcAbi = viem.parseAbi([
900
+ "function batch((address targetContract, address onBehalfOfAccount, uint256 value, bytes data)[])"
901
+ ]);
902
+ async function buildEvmMultisignBodyEulerV2IsolatedBorrowBatch(args) {
903
+ const evc = viem.getAddress(args.evc);
904
+ const borrowVault = viem.getAddress(args.borrowVault);
905
+ const collateralVault = viem.getAddress(args.collateralVault);
906
+ const collateralAsset = viem.getAddress(args.collateralUnderlying);
907
+ const borrowAsset = viem.getAddress(args.borrowUnderlying);
908
+ const weth = viem.getAddress(args.nativeWrapped);
909
+ const executor = viem.getAddress(args.executorAddress);
910
+ const receiver = viem.getAddress(args.receiver);
911
+ if (args.isNativeCollateralIn && collateralAsset.toLowerCase() !== weth.toLowerCase()) {
912
+ throw new Error("Native collateral path: collateral underlying must be the chain wrapped native token.");
913
+ }
914
+ const ch = viem.defineChain({
915
+ id: args.chainId,
916
+ name: "Destination",
917
+ nativeCurrency: { decimals: 18, name: "Ether", symbol: "ETH" },
918
+ rpcUrls: { default: { http: [args.rpcUrl] } }
919
+ });
920
+ const publicClient = viem.createPublicClient({ chain: ch, transport: viem.http(args.rpcUrl) });
921
+ const cDecRaw = await publicClient.readContract({
922
+ address: collateralAsset,
923
+ abi: erc20AllowanceAbi2,
924
+ functionName: "decimals"
925
+ });
926
+ const collateralDecimals = Number(cDecRaw);
927
+ const collateralWei = viem.parseUnits(args.collateralAmountHuman, collateralDecimals);
928
+ if (collateralWei === 0n) throw new Error("Collateral amount is zero after converting with token decimals.");
929
+ await publicClient.readContract({
930
+ address: borrowAsset,
931
+ abi: erc20AllowanceAbi2,
932
+ functionName: "decimals"
933
+ });
934
+ const loops = [...args.loopBorrowWeis];
935
+ if (loops.length === 0) throw new Error("At least one borrow round is required.");
936
+ for (const w of loops) {
937
+ if (w <= 0n) throw new Error("Each borrow round must be positive.");
938
+ }
939
+ if (!args.redepositBorrowedToCollateral && loops.length !== 1) {
940
+ throw new Error("Multiple borrow rounds require redepositBorrowedToCollateral (same borrow and collateral asset).");
941
+ }
942
+ const borrowWeiTotal = loops.reduce((a, w) => a + w, 0n);
943
+ const totalPullWei = eulerSameAssetTotalCollateralPullWei({
944
+ initialCollateralWei: collateralWei,
945
+ loopBorrowWeis: args.redepositBorrowedToCollateral ? loops : []
946
+ });
947
+ const approveTargetWei = eulerSameAssetApproveAmountWithBuffer({ totalPullWei });
948
+ const steps = [];
949
+ if (args.isNativeCollateralIn) {
950
+ const dataDeposit = viem.encodeFunctionData({ abi: wethDepositAbi2, functionName: "deposit", args: [] });
951
+ steps.push({ kind: "weth_deposit", to: weth, data: dataDeposit, value: collateralWei });
952
+ const wethAllowance = await publicClient.readContract({
953
+ address: weth,
954
+ abi: erc20AllowanceAbi2,
955
+ functionName: "allowance",
956
+ args: [executor, collateralVault]
957
+ });
958
+ if (wethAllowance < approveTargetWei) {
959
+ if (wethAllowance > 0n) {
960
+ const dataReset = viem.encodeFunctionData({
961
+ abi: erc20ApproveAbi2,
962
+ functionName: "approve",
963
+ args: [collateralVault, 0n]
964
+ });
965
+ steps.push({ kind: "approve", to: weth, data: dataReset, value: 0n });
966
+ }
967
+ const dataApprove = viem.encodeFunctionData({
968
+ abi: erc20ApproveAbi2,
969
+ functionName: "approve",
970
+ args: [collateralVault, approveTargetWei]
971
+ });
972
+ steps.push({ kind: "approve", to: weth, data: dataApprove, value: 0n });
973
+ }
974
+ } else {
975
+ const currentAllowance = await publicClient.readContract({
976
+ address: collateralAsset,
977
+ abi: erc20AllowanceAbi2,
978
+ functionName: "allowance",
979
+ args: [executor, collateralVault]
980
+ });
981
+ if (currentAllowance < approveTargetWei) {
982
+ if (currentAllowance > 0n) {
983
+ const dataReset = viem.encodeFunctionData({
984
+ abi: erc20ApproveAbi2,
985
+ functionName: "approve",
986
+ args: [collateralVault, 0n]
987
+ });
988
+ steps.push({ kind: "approve", to: collateralAsset, data: dataReset, value: 0n });
989
+ }
990
+ const dataApprove = viem.encodeFunctionData({
991
+ abi: erc20ApproveAbi2,
992
+ functionName: "approve",
993
+ args: [collateralVault, approveTargetWei]
994
+ });
995
+ steps.push({ kind: "approve", to: collateralAsset, data: dataApprove, value: 0n });
996
+ }
997
+ }
998
+ const depositData = viem.encodeFunctionData({
999
+ abi: erc4626DepositAbi2,
1000
+ functionName: "deposit",
1001
+ args: [collateralWei, receiver]
1002
+ });
1003
+ const enableCollData = viem.encodeFunctionData({
1004
+ abi: viem.parseAbi(["function enableCollateral(address account, address vault)"]),
1005
+ functionName: "enableCollateral",
1006
+ args: [receiver, collateralVault]
1007
+ });
1008
+ const enableCtrlData = viem.encodeFunctionData({
1009
+ abi: viem.parseAbi(["function enableController(address account, address vault)"]),
1010
+ functionName: "enableController",
1011
+ args: [receiver, borrowVault]
1012
+ });
1013
+ const batchItems = [
1014
+ { targetContract: collateralVault, onBehalfOfAccount: receiver, value: 0n, data: depositData },
1015
+ { targetContract: evc, onBehalfOfAccount: viem.zeroAddress, value: 0n, data: enableCollData },
1016
+ { targetContract: evc, onBehalfOfAccount: viem.zeroAddress, value: 0n, data: enableCtrlData }
1017
+ ];
1018
+ for (const bw of loops) {
1019
+ const borrowData = viem.encodeFunctionData({
1020
+ abi: evaultBorrowAbi,
1021
+ functionName: "borrow",
1022
+ args: [bw, receiver]
1023
+ });
1024
+ batchItems.push({
1025
+ targetContract: borrowVault,
1026
+ onBehalfOfAccount: receiver,
1027
+ value: 0n,
1028
+ data: borrowData
1029
+ });
1030
+ if (args.redepositBorrowedToCollateral) {
1031
+ const redepositData = viem.encodeFunctionData({
1032
+ abi: erc4626DepositAbi2,
1033
+ functionName: "deposit",
1034
+ args: [bw, receiver]
1035
+ });
1036
+ batchItems.push({
1037
+ targetContract: collateralVault,
1038
+ onBehalfOfAccount: receiver,
1039
+ value: 0n,
1040
+ data: redepositData
1041
+ });
1042
+ }
1043
+ }
1044
+ const batchData = viem.encodeFunctionData({
1045
+ abi: evcAbi,
1046
+ functionName: "batch",
1047
+ args: [batchItems]
1048
+ });
1049
+ steps.push({ kind: "evc_batch", to: evc, data: batchData, value: 0n });
1050
+ const borrowRoundsExtraGas = EULER_BORROW_BATCH_FALLBACK_GAS_PER_ROUND * BigInt(Math.max(0, loops.length - 1 + (args.redepositBorrowedToCollateral ? loops.length : 0)));
1051
+ const vaultLabel = (args.vaultMarketLabel ?? "").trim() || "Euler vault";
1052
+ const evmSteps = steps.map((s) => ({
1053
+ to: s.to,
1054
+ data: s.data,
1055
+ value: s.value,
1056
+ fallbackGas: s.kind === "weth_deposit" ? EULER_WETH_DEPOSIT_FALLBACK2 : s.kind === "approve" ? EULER_ERC20_APPROVE_FALLBACK2 : EULER_BORROW_BATCH_FALLBACK_GAS + borrowRoundsExtraGas
1057
+ }));
1058
+ const n = steps.length;
1059
+ const hasWrap = args.isNativeCollateralIn;
1060
+ const purposeSuffix = (() => {
1061
+ const tail = args.redepositBorrowedToCollateral ? `deposit collateral, enableCollateral, enableController, then ${loops.length}\xD7 borrow+redeposit on "${vaultLabel}" (same-asset target LTV loop).` : `deposit collateral, enableCollateral, enableController, borrow from "${vaultLabel}".`;
1062
+ if (hasWrap) {
1063
+ return `Euler v2: ${n}-tx batch \u2014 wrap native collateral (if needed), approve collateral eVault (buffered for all deposits), then EVC batch: ${tail}`;
1064
+ }
1065
+ return `Euler v2: ${n}-tx batch \u2014 approve collateral (if needed) with buffer for all deposits, then EVC batch: ${tail}`;
1066
+ })();
1067
+ const firstDataNo0x = evmSteps[0].data.startsWith("0x") ? evmSteps[0].data.slice(2) : evmSteps[0].data;
1068
+ return buildEvmMultisignBatch({
1069
+ context: {
1070
+ chainCategory: "evm",
1071
+ keyGen: args.keyGen,
1072
+ purposeText: args.purposeText,
1073
+ chainId: args.chainId,
1074
+ rpcUrl: args.rpcUrl,
1075
+ executorAddress: executor,
1076
+ chainDetail: args.chainDetail,
1077
+ useCustomGas: args.useCustomGas,
1078
+ customGasChainDetails: args.customGasChainDetails
1079
+ },
1080
+ steps: evmSteps,
1081
+ purposeSuffix,
1082
+ firstMsgRawNo0x: firstDataNo0x,
1083
+ destinationAddress: steps[0].to,
1084
+ buildBatchMeta: ({ index }) => {
1085
+ const s = steps[index];
1086
+ if (s.kind === "weth_deposit") {
1087
+ return {
1088
+ signatureText: JSON.stringify({
1089
+ kind: "EulerV2",
1090
+ name: "WETH.deposit",
1091
+ function: "deposit()",
1092
+ valueWei: collateralWei.toString(),
1093
+ vaultMarket: vaultLabel,
1094
+ note: "Wrap native for Euler v2 isolated borrow collateral (same batch as borrow flow)."
1095
+ }),
1096
+ evm: { type: "euler_v2_weth_deposit", version: 1, chainId: String(args.chainId) },
1097
+ eulerV2: { step: "weth_deposit", vaultMarket: vaultLabel, flow: "borrow" }
1098
+ };
1099
+ }
1100
+ if (s.kind === "approve") {
1101
+ return {
1102
+ signatureText: JSON.stringify({
1103
+ kind: "EulerV2",
1104
+ name: "ERC20.approve",
1105
+ to: "Euler collateral eVault",
1106
+ function: "approve(address spender, uint256 amount)",
1107
+ collateralVault,
1108
+ amountHuman: args.collateralAmountHuman,
1109
+ note: "Allowance for initial and follow-on collateral deposits (buffered).",
1110
+ approveTotalWei: approveTargetWei.toString()
1111
+ }),
1112
+ evm: { type: "euler_v2_erc20_approve", version: 1, chainId: String(args.chainId) },
1113
+ eulerV2: { vaultMarket: vaultLabel, flow: "borrow_collateral_approve" }
1114
+ };
1115
+ }
1116
+ const borrowNote = args.redepositBorrowedToCollateral ? `Same-asset leverage: ${loops.length} borrow\u2192deposit round(s); total borrow wei ${borrowWeiTotal.toString()}.` : "Deposit collateral, enableCollateral, enableController, borrow in one EVC batch.";
1117
+ return {
1118
+ signatureText: JSON.stringify({
1119
+ kind: "EulerV2",
1120
+ name: "EVC.batch",
1121
+ function: "batch(BatchItem[])",
1122
+ evc,
1123
+ borrowVault,
1124
+ collateralVault,
1125
+ collateralAmountHuman: args.collateralAmountHuman,
1126
+ borrowAmountHuman: args.borrowAmountHuman,
1127
+ vaultMarket: vaultLabel,
1128
+ loopBorrowRounds: loops.length,
1129
+ redepositBorrowedToCollateral: args.redepositBorrowedToCollateral,
1130
+ borrowWeiTotal: borrowWeiTotal.toString(),
1131
+ note: borrowNote
1132
+ }),
1133
+ evm: { type: "euler_v2_evc_borrow_batch", version: 1, chainId: String(args.chainId) },
1134
+ eulerV2: {
1135
+ flow: "borrow_batch",
1136
+ vaultMarket: vaultLabel,
1137
+ borrowVault,
1138
+ collateralVault,
1139
+ collateralAmountHuman: args.collateralAmountHuman,
1140
+ borrowAmountHuman: args.borrowAmountHuman,
1141
+ loopBorrowRounds: loops.length,
1142
+ redepositBorrowedToCollateral: args.redepositBorrowedToCollateral,
1143
+ borrowWeiTotal: borrowWeiTotal.toString()
1144
+ }
1145
+ };
1146
+ }
1147
+ });
1148
+ }
1149
+ var EULER_REPAY_BATCH_FALLBACK = 1200000n;
1150
+ var EULER_ERC20_APPROVE_FALLBACK3 = 100000n;
1151
+ var erc20AllowanceAbi3 = viem.parseAbi([
1152
+ "function allowance(address owner, address spender) view returns (uint256)",
1153
+ "function decimals() view returns (uint8)"
1154
+ ]);
1155
+ var erc20ApproveAbi3 = viem.parseAbi(["function approve(address spender, uint256 amount) returns (bool)"]);
1156
+ var evaultRepayAbi = viem.parseAbi(["function repay(uint256 amount, address receiver) returns (uint256)"]);
1157
+ var evcAbi2 = viem.parseAbi([
1158
+ "function batch((address targetContract, address onBehalfOfAccount, uint256 value, bytes data)[])"
1159
+ ]);
1160
+ async function buildEvmMultisignBodyEulerV2BorrowRepayBatch(args) {
1161
+ const evc = viem.getAddress(args.evc);
1162
+ const borrowVault = viem.getAddress(args.borrowVault);
1163
+ const borrowAsset = viem.getAddress(args.borrowUnderlying);
1164
+ const subAccount = viem.getAddress(args.subAccount);
1165
+ const executor = viem.getAddress(args.executorAddress);
1166
+ const ch = viem.defineChain({
1167
+ id: args.chainId,
1168
+ name: "EulerRepay",
1169
+ nativeCurrency: { decimals: 18, name: "Ether", symbol: "ETH" },
1170
+ rpcUrls: { default: { http: [args.rpcUrl] } }
1171
+ });
1172
+ const publicClient = viem.createPublicClient({ chain: ch, transport: viem.http(args.rpcUrl) });
1173
+ const bDecRaw = await publicClient.readContract({
1174
+ address: borrowAsset,
1175
+ abi: erc20AllowanceAbi3,
1176
+ functionName: "decimals"
1177
+ });
1178
+ const borrowDecimals = Number(bDecRaw);
1179
+ const debtAbi = viem.parseAbi(["function debtOf(address account) view returns (uint256)"]);
1180
+ const owed = await publicClient.readContract({
1181
+ address: borrowVault,
1182
+ abi: debtAbi,
1183
+ functionName: "debtOf",
1184
+ args: [subAccount]
1185
+ });
1186
+ let repayWei;
1187
+ if (args.repayAll) {
1188
+ repayWei = viem.maxUint256;
1189
+ } else {
1190
+ repayWei = viem.parseUnits(args.amountHuman, borrowDecimals);
1191
+ if (repayWei === 0n) throw new Error("Repay amount is zero after converting with token decimals.");
1192
+ if (repayWei > owed) {
1193
+ repayWei = owed;
1194
+ }
1195
+ }
1196
+ const steps = [];
1197
+ const allowanceTarget = repayWei === viem.maxUint256 ? owed : repayWei;
1198
+ const currentAllowance = await publicClient.readContract({
1199
+ address: borrowAsset,
1200
+ abi: erc20AllowanceAbi3,
1201
+ functionName: "allowance",
1202
+ args: [executor, borrowVault]
1203
+ });
1204
+ if (currentAllowance < allowanceTarget) {
1205
+ if (currentAllowance > 0n) {
1206
+ const dataReset = viem.encodeFunctionData({
1207
+ abi: erc20ApproveAbi3,
1208
+ functionName: "approve",
1209
+ args: [borrowVault, 0n]
1210
+ });
1211
+ steps.push({ kind: "approve", to: borrowAsset, data: dataReset, value: 0n });
1212
+ }
1213
+ const approveAmt = repayWei === viem.maxUint256 ? viem.maxUint256 : repayWei;
1214
+ const dataApprove = viem.encodeFunctionData({
1215
+ abi: erc20ApproveAbi3,
1216
+ functionName: "approve",
1217
+ args: [borrowVault, approveAmt]
1218
+ });
1219
+ steps.push({ kind: "approve", to: borrowAsset, data: dataApprove, value: 0n });
1220
+ }
1221
+ const repayData = viem.encodeFunctionData({
1222
+ abi: evaultRepayAbi,
1223
+ functionName: "repay",
1224
+ args: [repayWei, subAccount]
1225
+ });
1226
+ const batchItems = [
1227
+ { targetContract: borrowVault, onBehalfOfAccount: subAccount, value: 0n, data: repayData }
1228
+ ];
1229
+ const batchData = viem.encodeFunctionData({
1230
+ abi: evcAbi2,
1231
+ functionName: "batch",
1232
+ args: [batchItems]
1233
+ });
1234
+ steps.push({ kind: "evc_batch", to: evc, data: batchData, value: 0n });
1235
+ const vaultLabel = (args.vaultMarketLabel ?? "").trim() || "Euler vault";
1236
+ const evmSteps = steps.map((s) => ({
1237
+ to: s.to,
1238
+ data: s.data,
1239
+ value: s.value,
1240
+ fallbackGas: s.kind === "approve" ? EULER_ERC20_APPROVE_FALLBACK3 : EULER_REPAY_BATCH_FALLBACK
1241
+ }));
1242
+ const n = steps.length;
1243
+ const purposeSuffix = `Euler v2: ${n}-tx batch \u2014 repay borrow on "${vaultLabel}" (approve if needed, then EVC repay).`;
1244
+ const firstDataNo0x = evmSteps[0].data.startsWith("0x") ? evmSteps[0].data.slice(2) : evmSteps[0].data;
1245
+ return buildEvmMultisignBatch({
1246
+ context: {
1247
+ chainCategory: "evm",
1248
+ keyGen: args.keyGen,
1249
+ purposeText: args.purposeText,
1250
+ chainId: args.chainId,
1251
+ rpcUrl: args.rpcUrl,
1252
+ executorAddress: executor,
1253
+ chainDetail: args.chainDetail,
1254
+ useCustomGas: args.useCustomGas,
1255
+ customGasChainDetails: args.customGasChainDetails
1256
+ },
1257
+ steps: evmSteps,
1258
+ purposeSuffix,
1259
+ firstMsgRawNo0x: firstDataNo0x,
1260
+ destinationAddress: steps[0].to,
1261
+ buildBatchMeta: ({ index }) => {
1262
+ const s = steps[index];
1263
+ if (s.kind === "approve") {
1264
+ return {
1265
+ signatureText: JSON.stringify({
1266
+ kind: "EulerV2",
1267
+ name: "ERC20.approve",
1268
+ function: "approve(address spender, uint256 amount)",
1269
+ spender: borrowVault,
1270
+ borrowUnderlying: borrowAsset,
1271
+ note: "Allow Euler liability vault to pull assets for repay."
1272
+ }),
1273
+ evm: { type: "euler_v2_erc20_approve", version: 1, chainId: String(args.chainId) },
1274
+ eulerV2: { vaultMarket: vaultLabel, flow: "borrow_repay_approve" }
1275
+ };
1276
+ }
1277
+ return {
1278
+ signatureText: JSON.stringify({
1279
+ kind: "EulerV2",
1280
+ name: "EVC.batch",
1281
+ function: "batch(BatchItem[]) \u2014 repay",
1282
+ evc,
1283
+ borrowVault,
1284
+ subAccount,
1285
+ amountHuman: args.repayAll ? "max" : args.amountHuman,
1286
+ vaultMarket: vaultLabel,
1287
+ note: "Repay borrow on Euler v2 liability vault via EVC."
1288
+ }),
1289
+ evm: { type: "euler_v2_borrow_repay_batch", version: 1, chainId: String(args.chainId) },
1290
+ eulerV2: {
1291
+ flow: "borrow_repay",
1292
+ vaultMarket: vaultLabel,
1293
+ borrowVault,
1294
+ subAccount,
1295
+ repayAll: args.repayAll,
1296
+ amountHuman: args.amountHuman
1297
+ }
1298
+ };
1299
+ }
1300
+ });
1301
+ }
1302
+ var EULER_COLLATERAL_DEPOSIT_BATCH_FALLBACK = 1600000n;
1303
+ var EULER_ERC20_APPROVE_FALLBACK4 = 100000n;
1304
+ var erc20AllowanceAbi4 = viem.parseAbi([
1305
+ "function allowance(address owner, address spender) view returns (uint256)"
1306
+ ]);
1307
+ var erc20ApproveAbi4 = viem.parseAbi(["function approve(address spender, uint256 amount) returns (bool)"]);
1308
+ var erc4626DepositAbi3 = viem.parseAbi([
1309
+ "function deposit(uint256 assets, address receiver) returns (uint256 shares)"
1310
+ ]);
1311
+ var evcAbi3 = viem.parseAbi([
1312
+ "function batch((address targetContract, address onBehalfOfAccount, uint256 value, bytes data)[])"
1313
+ ]);
1314
+ async function buildEvmMultisignBodyEulerV2BorrowCollateralDepositBatch(args) {
1315
+ const evc = viem.getAddress(args.evc);
1316
+ const collateralVault = viem.getAddress(args.collateralVault);
1317
+ const collateralAsset = viem.getAddress(args.collateralUnderlying);
1318
+ const subAccount = viem.getAddress(args.subAccount);
1319
+ const executor = viem.getAddress(args.executorAddress);
1320
+ const ch = viem.defineChain({
1321
+ id: args.chainId,
1322
+ name: "EulerColDeposit",
1323
+ nativeCurrency: { decimals: 18, name: "Ether", symbol: "ETH" },
1324
+ rpcUrls: { default: { http: [args.rpcUrl] } }
1325
+ });
1326
+ const publicClient = viem.createPublicClient({ chain: ch, transport: viem.http(args.rpcUrl) });
1327
+ const dec = await fetchEulerVaultAssetDecimals({
1328
+ rpcUrl: args.rpcUrl,
1329
+ chainId: args.chainId,
1330
+ evault: collateralVault
1331
+ });
1332
+ const amountWei = viem.parseUnits(args.amountHuman, dec);
1333
+ if (amountWei === 0n) throw new Error("Deposit amount is zero after converting with token decimals.");
1334
+ const steps = [];
1335
+ const currentAllowance = await publicClient.readContract({
1336
+ address: collateralAsset,
1337
+ abi: erc20AllowanceAbi4,
1338
+ functionName: "allowance",
1339
+ args: [executor, collateralVault]
1340
+ });
1341
+ if (currentAllowance < amountWei) {
1342
+ if (currentAllowance > 0n) {
1343
+ const dataReset = viem.encodeFunctionData({
1344
+ abi: erc20ApproveAbi4,
1345
+ functionName: "approve",
1346
+ args: [collateralVault, 0n]
1347
+ });
1348
+ steps.push({ kind: "approve", to: collateralAsset, data: dataReset, value: 0n });
1349
+ }
1350
+ const dataApprove = viem.encodeFunctionData({
1351
+ abi: erc20ApproveAbi4,
1352
+ functionName: "approve",
1353
+ args: [collateralVault, amountWei]
1354
+ });
1355
+ steps.push({ kind: "approve", to: collateralAsset, data: dataApprove, value: 0n });
1356
+ }
1357
+ const depositData = viem.encodeFunctionData({
1358
+ abi: erc4626DepositAbi3,
1359
+ functionName: "deposit",
1360
+ args: [amountWei, subAccount]
1361
+ });
1362
+ const batchItems = [
1363
+ {
1364
+ targetContract: collateralVault,
1365
+ onBehalfOfAccount: subAccount,
1366
+ value: 0n,
1367
+ data: depositData
1368
+ }
1369
+ ];
1370
+ const batchData = viem.encodeFunctionData({
1371
+ abi: evcAbi3,
1372
+ functionName: "batch",
1373
+ args: [batchItems]
1374
+ });
1375
+ steps.push({ kind: "evc_batch", to: evc, data: batchData, value: 0n });
1376
+ const vaultLabel = (args.vaultMarketLabel ?? "").trim() || "Euler vault";
1377
+ const evmSteps = steps.map((s) => ({
1378
+ to: s.to,
1379
+ data: s.data,
1380
+ value: s.value,
1381
+ fallbackGas: s.kind === "approve" ? EULER_ERC20_APPROVE_FALLBACK4 : EULER_COLLATERAL_DEPOSIT_BATCH_FALLBACK
1382
+ }));
1383
+ const n = steps.length;
1384
+ const purposeSuffix = `Euler v2: ${n}-tx batch \u2014 deposit ${args.amountHuman} collateral into "${vaultLabel}" (approve if needed, then EVC deposit).`;
1385
+ const firstDataNo0x = evmSteps[0].data.startsWith("0x") ? evmSteps[0].data.slice(2) : evmSteps[0].data;
1386
+ return buildEvmMultisignBatch({
1387
+ context: {
1388
+ chainCategory: "evm",
1389
+ keyGen: args.keyGen,
1390
+ purposeText: args.purposeText,
1391
+ chainId: args.chainId,
1392
+ rpcUrl: args.rpcUrl,
1393
+ executorAddress: executor,
1394
+ chainDetail: args.chainDetail,
1395
+ useCustomGas: args.useCustomGas,
1396
+ customGasChainDetails: args.customGasChainDetails
1397
+ },
1398
+ steps: evmSteps,
1399
+ purposeSuffix,
1400
+ firstMsgRawNo0x: firstDataNo0x,
1401
+ destinationAddress: steps[0].to,
1402
+ buildBatchMeta: ({ index }) => {
1403
+ const s = steps[index];
1404
+ if (s.kind === "approve") {
1405
+ return {
1406
+ signatureText: JSON.stringify({
1407
+ kind: "EulerV2",
1408
+ name: "ERC20.approve",
1409
+ function: "approve(address spender, uint256 amount)",
1410
+ spender: collateralVault,
1411
+ collateralUnderlying: collateralAsset,
1412
+ note: "Allow Euler collateral eVault to pull assets for collateral deposit."
1413
+ }),
1414
+ evm: { type: "euler_v2_erc20_approve", version: 1, chainId: String(args.chainId) },
1415
+ eulerV2: { vaultMarket: vaultLabel, flow: "borrow_collateral_deposit_approve" }
1416
+ };
1417
+ }
1418
+ return {
1419
+ signatureText: JSON.stringify({
1420
+ kind: "EulerV2",
1421
+ name: "EVC.batch",
1422
+ function: "batch(BatchItem[]) \u2014 collateral deposit",
1423
+ evc,
1424
+ collateralVault,
1425
+ subAccount,
1426
+ amountHuman: args.amountHuman,
1427
+ vaultMarket: vaultLabel,
1428
+ note: "Deposit collateral into Euler v2 collateral eVault via EVC (reduces LTV)."
1429
+ }),
1430
+ evm: { type: "euler_v2_borrow_collateral_deposit_batch", version: 1, chainId: String(args.chainId) },
1431
+ eulerV2: {
1432
+ flow: "borrow_collateral_deposit",
1433
+ vaultMarket: vaultLabel,
1434
+ collateralVault,
1435
+ subAccount,
1436
+ amountHuman: args.amountHuman
1437
+ }
1438
+ };
1439
+ }
1440
+ });
1441
+ }
1442
+ var EULER_COLLATERAL_WITHDRAW_BATCH_FALLBACK = 1400000n;
1443
+ var erc4626WithdrawAbi3 = viem.parseAbi([
1444
+ "function withdraw(uint256 assets, address receiver, address owner) returns (uint256 shares)"
1445
+ ]);
1446
+ var evcAbi4 = viem.parseAbi([
1447
+ "function batch((address targetContract, address onBehalfOfAccount, uint256 value, bytes data)[])"
1448
+ ]);
1449
+ async function buildEvmMultisignBodyEulerV2BorrowCollateralWithdrawBatch(args) {
1450
+ const evc = viem.getAddress(args.evc);
1451
+ const collateralVault = viem.getAddress(args.collateralVault);
1452
+ const subAccount = viem.getAddress(args.subAccount);
1453
+ const receiver = viem.getAddress(args.receiver);
1454
+ const executor = viem.getAddress(args.executorAddress);
1455
+ const dec = await fetchEulerVaultAssetDecimals({ rpcUrl: args.rpcUrl, chainId: args.chainId, evault: collateralVault });
1456
+ const amountWei = viem.parseUnits(args.amountHuman, dec);
1457
+ if (amountWei === 0n) throw new Error("Withdraw amount is zero after converting with token decimals.");
1458
+ const maxW = await fetchEulerBorrowCollateralMaxWithdrawAssetsWei({
1459
+ rpcUrl: args.rpcUrl,
1460
+ chainId: args.chainId,
1461
+ evc,
1462
+ collateralVault,
1463
+ subAccount,
1464
+ caller: executor
1465
+ });
1466
+ if (amountWei > maxW) {
1467
+ throw new Error(
1468
+ "Withdraw amount exceeds simulated max for this collateral vault (LTV cap, vault cash, or health check). Try a smaller amount."
1469
+ );
1470
+ }
1471
+ const withdrawData = viem.encodeFunctionData({
1472
+ abi: erc4626WithdrawAbi3,
1473
+ functionName: "withdraw",
1474
+ args: [amountWei, receiver, subAccount]
1475
+ });
1476
+ const batchItems = [
1477
+ {
1478
+ targetContract: collateralVault,
1479
+ onBehalfOfAccount: subAccount,
1480
+ value: 0n,
1481
+ data: withdrawData
1482
+ }
1483
+ ];
1484
+ const batchData = viem.encodeFunctionData({
1485
+ abi: evcAbi4,
1486
+ functionName: "batch",
1487
+ args: [batchItems]
1488
+ });
1489
+ const steps = [{ kind: "evc_batch", to: evc, data: batchData, value: 0n }];
1490
+ const vaultLabel = (args.vaultMarketLabel ?? "").trim() || "Euler vault";
1491
+ const purposeSuffix = `Euler v2: 1-tx \u2014 withdraw ${args.amountHuman} collateral from "${vaultLabel}" via EVC (sub-account).`;
1492
+ const firstDataNo0x = batchData.startsWith("0x") ? batchData.slice(2) : batchData;
1493
+ const evmSteps = [
1494
+ { to: evc, data: batchData, value: 0n, fallbackGas: EULER_COLLATERAL_WITHDRAW_BATCH_FALLBACK }
1495
+ ];
1496
+ return buildEvmMultisignBatch({
1497
+ context: {
1498
+ chainCategory: "evm",
1499
+ keyGen: args.keyGen,
1500
+ purposeText: args.purposeText,
1501
+ chainId: args.chainId,
1502
+ rpcUrl: args.rpcUrl,
1503
+ executorAddress: executor,
1504
+ chainDetail: args.chainDetail,
1505
+ useCustomGas: args.useCustomGas,
1506
+ customGasChainDetails: args.customGasChainDetails
1507
+ },
1508
+ steps: evmSteps,
1509
+ purposeSuffix,
1510
+ firstMsgRawNo0x: firstDataNo0x,
1511
+ destinationAddress: steps[0].to,
1512
+ buildBatchMeta: () => ({
1513
+ signatureText: JSON.stringify({
1514
+ kind: "EulerV2",
1515
+ name: "EVC.batch",
1516
+ function: "batch \u2014 collateral withdraw",
1517
+ evc,
1518
+ collateralVault,
1519
+ subAccount,
1520
+ receiver,
1521
+ amountHuman: args.amountHuman,
1522
+ vaultMarket: vaultLabel,
1523
+ note: "Withdraw collateral from Euler v2 eVault via EVC (max from RPC simulate + LTV / vault cash)."
1524
+ }),
1525
+ evm: { type: "euler_v2_borrow_collateral_withdraw_batch", version: 1, chainId: String(args.chainId) },
1526
+ eulerV2: {
1527
+ flow: "borrow_collateral_withdraw",
1528
+ vaultMarket: vaultLabel,
1529
+ collateralVault,
1530
+ subAccount,
1531
+ receiver,
1532
+ amountHuman: args.amountHuman
1533
+ }
1534
+ })
1535
+ });
1536
+ }
1537
+
1538
+ // src/protocols/evm/euler-v2/index.ts
1539
+ var EULER_V2_PROTOCOL_ID = "euler-v2";
1540
+ var eulerV2ProtocolModule = {
1541
+ id: EULER_V2_PROTOCOL_ID,
1542
+ chainCategory: "evm",
1543
+ isChainSupported(ctx) {
1544
+ return ctx.chainCategory === "evm";
1545
+ },
1546
+ isTokenSupported(token) {
1547
+ return token.category === "evm" && (token.kind === "native" || token.kind === "erc20");
1548
+ },
1549
+ actions: [
1550
+ { id: "euler-v2.isolated-lend", protocolId: EULER_V2_PROTOCOL_ID, chainCategory: "evm", description: "Deposit into Euler vault", commonParams: ["keyGen", "purposeText", "useCustomGas"], params: {} },
1551
+ { id: "euler-v2.isolated-borrow", protocolId: EULER_V2_PROTOCOL_ID, chainCategory: "evm", description: "Borrow from Euler vault", commonParams: ["keyGen", "purposeText", "useCustomGas"], params: {} },
1552
+ { id: "euler-v2.vault-withdraw", protocolId: EULER_V2_PROTOCOL_ID, chainCategory: "evm", description: "Withdraw from Euler vault", commonParams: ["keyGen", "purposeText", "useCustomGas"], params: {} },
1553
+ { id: "euler-v2.borrow-repay", protocolId: EULER_V2_PROTOCOL_ID, chainCategory: "evm", description: "Repay Euler borrow", commonParams: ["keyGen", "purposeText", "useCustomGas"], params: {} },
1554
+ { id: "euler-v2.collateral-deposit", protocolId: EULER_V2_PROTOCOL_ID, chainCategory: "evm", description: "Deposit borrow collateral", commonParams: ["keyGen", "purposeText", "useCustomGas"], params: {} },
1555
+ { id: "euler-v2.collateral-withdraw", protocolId: EULER_V2_PROTOCOL_ID, chainCategory: "evm", description: "Withdraw borrow collateral", commonParams: ["keyGen", "purposeText", "useCustomGas"], params: {} }
1556
+ ]
1557
+ };
1558
+ registerProtocolModule(eulerV2ProtocolModule);
1559
+
1560
+ exports.EULER_SAME_ASSET_BORROW_APPROVAL_BUFFER_BPS = EULER_SAME_ASSET_BORROW_APPROVAL_BUFFER_BPS;
1561
+ exports.EULER_SAME_ASSET_BORROW_MAX_ROUNDS = EULER_SAME_ASSET_BORROW_MAX_ROUNDS;
1562
+ exports.EULER_SAME_ASSET_BORROW_PROTOCOL_HEADROOM_BPS = EULER_SAME_ASSET_BORROW_PROTOCOL_HEADROOM_BPS;
1563
+ exports.EULER_SAME_ASSET_BORROW_RATIO_STOP_EPS_BPS = EULER_SAME_ASSET_BORROW_RATIO_STOP_EPS_BPS;
1564
+ exports.EULER_V2_ISOLATED_VAULT_DEPOSIT_FALLBACK_GAS = EULER_V2_ISOLATED_VAULT_DEPOSIT_FALLBACK_GAS;
1565
+ exports.EULER_V2_PROTOCOL_ID = EULER_V2_PROTOCOL_ID;
1566
+ exports.EULER_V2_VAULT_WITHDRAW_FALLBACK_GAS = EULER_V2_VAULT_WITHDRAW_FALLBACK_GAS;
1567
+ exports.buildEvmMultisignBodyEulerV2BorrowCollateralDepositBatch = buildEvmMultisignBodyEulerV2BorrowCollateralDepositBatch;
1568
+ exports.buildEvmMultisignBodyEulerV2BorrowCollateralWithdrawBatch = buildEvmMultisignBodyEulerV2BorrowCollateralWithdrawBatch;
1569
+ exports.buildEvmMultisignBodyEulerV2BorrowRepayBatch = buildEvmMultisignBodyEulerV2BorrowRepayBatch;
1570
+ exports.buildEvmMultisignBodyEulerV2IsolatedBorrowBatch = buildEvmMultisignBodyEulerV2IsolatedBorrowBatch;
1571
+ exports.buildEvmMultisignBodyEulerV2IsolatedLendDepositBatch = buildEvmMultisignBodyEulerV2IsolatedLendDepositBatch;
1572
+ exports.buildEvmMultisignBodyEulerV2VaultWithdraw = buildEvmMultisignBodyEulerV2VaultWithdraw;
1573
+ exports.clampEulerUnderlyingDecimalsForEulerUi = clampEulerUnderlyingDecimalsForEulerUi;
1574
+ exports.eulerBorrowAndCollateralSameAsset = eulerBorrowAndCollateralSameAsset;
1575
+ exports.eulerSameAssetApproveAmountWithBuffer = eulerSameAssetApproveAmountWithBuffer;
1576
+ exports.eulerSameAssetTotalCollateralPullWei = eulerSameAssetTotalCollateralPullWei;
1577
+ exports.eulerV2ProtocolModule = eulerV2ProtocolModule;
1578
+ exports.fetchEulerBorrowCollateralMaxWithdrawAssetsWei = fetchEulerBorrowCollateralMaxWithdrawAssetsWei;
1579
+ exports.fetchEulerLendEarnVaultEffectiveMaxWithdrawWei = fetchEulerLendEarnVaultEffectiveMaxWithdrawWei;
1580
+ exports.fetchEulerVaultAssetDecimals = fetchEulerVaultAssetDecimals;
1581
+ exports.fetchEulerVaultMaxWithdrawWei = fetchEulerVaultMaxWithdrawWei;
1582
+ exports.fetchEulerVaultUnderlyingMeta = fetchEulerVaultUnderlyingMeta;
1583
+ exports.planSameAssetLeveragedBorrows = planSameAssetLeveragedBorrows;
1584
+ //# sourceMappingURL=index.cjs.map
1585
+ //# sourceMappingURL=index.cjs.map