@stableflow/utils-evm 3.0.0-beta-02a59f37a0-20260515

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,27 @@
1
+ # @stableflow/utils-evm
2
+
3
+ EVM utilities for the StableFlow SDK: RPC fallback providers and LayerZero compose helpers.
4
+
5
+ ## Installation
6
+
7
+ ```bash
8
+ pnpm add @stableflow/utils-evm @stableflow/core
9
+ ```
10
+
11
+ This package is primarily used internally by `@stableflow/bridges` and `@stableflow/wallet-evm`. Application code rarely imports it directly.
12
+
13
+ ## Main exports
14
+
15
+ | Export | Description |
16
+ |--------|-------------|
17
+ | `evmRpcFallbackProvider` | EVM JSON-RPC provider with fallback URLs |
18
+ | `buildEndpointV2LzComposePayload` | LayerZero v2 compose payload builder |
19
+ | `encodeUint`, `normalizeHex`, `toBytes32` | Encoding helpers |
20
+ | `NATIVE_MSG_FEE_BUFFER` | Native message fee buffer constant |
21
+
22
+ See `dist/index.d.ts` after `pnpm build`.
23
+
24
+ ## See also
25
+
26
+ - [Developer Guide](../../DEVELOPER_GUIDE.md)
27
+ - [@stableflow/wallet-evm](../wallet-evm/README.md)
@@ -0,0 +1,24 @@
1
+ import { ethers } from 'ethers';
2
+ import { TokenConfig } from '@stableflow/core';
3
+
4
+ declare class SequentialFallbackProvider extends ethers.AbstractProvider {
5
+ private providers;
6
+ constructor(providers: ethers.JsonRpcProvider[], chainId: number);
7
+ _detectNetwork(): Promise<ethers.Network>;
8
+ _perform(req: ethers.PerformActionRequest): Promise<any>;
9
+ }
10
+ declare function evmRpcFallbackProvider(chain: TokenConfig): ethers.AbstractProvider | SequentialFallbackProvider | ethers.JsonRpcProvider;
11
+
12
+ declare function normalizeHex(hex: string): string;
13
+ declare function toBytes32(addrOrBytes32: string): string;
14
+ declare function encodeUint(value: string, sizeBytes: number): string;
15
+ declare function buildEndpointV2LzComposePayload(params: {
16
+ nonce: string;
17
+ srcEid: string;
18
+ amountLD: string;
19
+ composeFrom: string;
20
+ composeMsg: string;
21
+ }): string;
22
+ declare const NATIVE_MSG_FEE_BUFFER: bigint;
23
+
24
+ export { NATIVE_MSG_FEE_BUFFER, buildEndpointV2LzComposePayload, encodeUint, evmRpcFallbackProvider, normalizeHex, toBytes32 };
@@ -0,0 +1,24 @@
1
+ import { ethers } from 'ethers';
2
+ import { TokenConfig } from '@stableflow/core';
3
+
4
+ declare class SequentialFallbackProvider extends ethers.AbstractProvider {
5
+ private providers;
6
+ constructor(providers: ethers.JsonRpcProvider[], chainId: number);
7
+ _detectNetwork(): Promise<ethers.Network>;
8
+ _perform(req: ethers.PerformActionRequest): Promise<any>;
9
+ }
10
+ declare function evmRpcFallbackProvider(chain: TokenConfig): ethers.AbstractProvider | SequentialFallbackProvider | ethers.JsonRpcProvider;
11
+
12
+ declare function normalizeHex(hex: string): string;
13
+ declare function toBytes32(addrOrBytes32: string): string;
14
+ declare function encodeUint(value: string, sizeBytes: number): string;
15
+ declare function buildEndpointV2LzComposePayload(params: {
16
+ nonce: string;
17
+ srcEid: string;
18
+ amountLD: string;
19
+ composeFrom: string;
20
+ composeMsg: string;
21
+ }): string;
22
+ declare const NATIVE_MSG_FEE_BUFFER: bigint;
23
+
24
+ export { NATIVE_MSG_FEE_BUFFER, buildEndpointV2LzComposePayload, encodeUint, evmRpcFallbackProvider, normalizeHex, toBytes32 };
package/dist/index.js ADDED
@@ -0,0 +1,99 @@
1
+ 'use strict';
2
+
3
+ var ethers = require('ethers');
4
+
5
+ // src/evm-rpc-providers.ts
6
+ var providerCache = /* @__PURE__ */ new Map();
7
+ var SequentialFallbackProvider = class extends ethers.ethers.AbstractProvider {
8
+ constructor(providers, chainId) {
9
+ super(chainId);
10
+ this.providers = providers;
11
+ }
12
+ async _detectNetwork() {
13
+ return this.providers[0]._detectNetwork();
14
+ }
15
+ async _perform(req) {
16
+ let lastError;
17
+ for (const provider of this.providers) {
18
+ try {
19
+ return await provider._perform(req);
20
+ } catch (err) {
21
+ if (ethers.ethers.isError(err, "CALL_EXCEPTION")) throw err;
22
+ lastError = err;
23
+ }
24
+ }
25
+ throw lastError;
26
+ }
27
+ };
28
+ function evmRpcFallbackProvider(chain) {
29
+ const { rpcUrls, chainId } = chain;
30
+ if (providerCache.has(chainId)) {
31
+ return providerCache.get(chainId);
32
+ }
33
+ const providers = rpcUrls.map(
34
+ (rpc) => {
35
+ return new ethers.ethers.JsonRpcProvider(rpc, chainId, { staticNetwork: true });
36
+ }
37
+ );
38
+ const provider = providers.length === 1 ? providers[0] : new SequentialFallbackProvider(providers, chainId);
39
+ providerCache.set(chainId, provider);
40
+ return provider;
41
+ }
42
+ function normalizeHex(hex) {
43
+ if (!hex) return "0x";
44
+ const s = String(hex);
45
+ const indices = [];
46
+ const re = /0x/gi;
47
+ let m;
48
+ while ((m = re.exec(s)) !== null) {
49
+ indices.push(m.index);
50
+ }
51
+ let best = "";
52
+ if (indices.length === 0) {
53
+ best = s.replace(/[^0-9a-fA-F]/g, "");
54
+ } else {
55
+ for (let i = 0; i < indices.length; i++) {
56
+ const start = indices[i] + 2;
57
+ const end = i + 1 < indices.length ? indices[i + 1] : s.length;
58
+ const candidate = s.slice(start, end).replace(/[^0-9a-fA-F]/g, "");
59
+ if (candidate.length > best.length) best = candidate;
60
+ }
61
+ }
62
+ if (!best) return "0x";
63
+ if (best.length % 2) throw new Error("hex length must be even");
64
+ return "0x" + best.toLowerCase();
65
+ }
66
+ function toBytes32(addrOrBytes32) {
67
+ const h = normalizeHex(addrOrBytes32);
68
+ if (h.length === 42) return ethers.ethers.zeroPadValue(h, 32);
69
+ if (h.length === 66) return h;
70
+ throw new Error("expected address(20B) or bytes32(32B)");
71
+ }
72
+ function encodeUint(value, sizeBytes) {
73
+ const bn = BigInt(value);
74
+ const max = (1n << BigInt(sizeBytes * 8)) - 1n;
75
+ if (bn < 0n || bn > max) {
76
+ throw new Error(`value does not fit uint${sizeBytes * 8}`);
77
+ }
78
+ return ethers.ethers.zeroPadValue(ethers.ethers.toBeHex(bn), sizeBytes);
79
+ }
80
+ function buildEndpointV2LzComposePayload(params) {
81
+ const { nonce, srcEid, amountLD, composeFrom, composeMsg } = params;
82
+ return ethers.ethers.concat([
83
+ encodeUint(nonce, 8),
84
+ encodeUint(srcEid, 4),
85
+ encodeUint(amountLD, 32),
86
+ toBytes32(composeFrom),
87
+ normalizeHex(composeMsg)
88
+ ]);
89
+ }
90
+ var NATIVE_MSG_FEE_BUFFER = 105n;
91
+
92
+ exports.NATIVE_MSG_FEE_BUFFER = NATIVE_MSG_FEE_BUFFER;
93
+ exports.buildEndpointV2LzComposePayload = buildEndpointV2LzComposePayload;
94
+ exports.encodeUint = encodeUint;
95
+ exports.evmRpcFallbackProvider = evmRpcFallbackProvider;
96
+ exports.normalizeHex = normalizeHex;
97
+ exports.toBytes32 = toBytes32;
98
+ //# sourceMappingURL=index.js.map
99
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/evm-rpc-providers.ts","../src/layerzero-compose.ts"],"names":["ethers"],"mappings":"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","file":"index.js","sourcesContent":["import { ethers } from \"ethers\";\nimport type { TokenConfig } from \"@stableflow/core\";\n\nconst providerCache = new Map<number, ethers.AbstractProvider>();\n\nclass SequentialFallbackProvider extends ethers.AbstractProvider {\n private providers: ethers.JsonRpcProvider[];\n\n constructor(providers: ethers.JsonRpcProvider[], chainId: number) {\n super(chainId);\n this.providers = providers;\n }\n\n async _detectNetwork(): Promise<ethers.Network> {\n return this.providers[0]._detectNetwork();\n }\n\n async _perform(req: ethers.PerformActionRequest): Promise<any> {\n let lastError: unknown;\n for (const provider of this.providers) {\n try {\n return await provider._perform(req);\n } catch (err) {\n if (ethers.isError(err, \"CALL_EXCEPTION\")) throw err;\n lastError = err;\n }\n }\n throw lastError;\n }\n}\n\nexport function evmRpcFallbackProvider(chain: TokenConfig) {\n const { rpcUrls, chainId } = chain;\n\n if (providerCache.has(chainId!)) {\n return providerCache.get(chainId!)!;\n }\n\n const providers = rpcUrls.map(\n (rpc: string) => {\n return new ethers.JsonRpcProvider(rpc, chainId, { staticNetwork: true });\n }\n );\n\n const provider =\n providers.length === 1\n ? providers[0]\n : new SequentialFallbackProvider(providers, chainId!);\n\n providerCache.set(chainId!, provider);\n return provider;\n}\n","import { ethers } from 'ethers';\n\nexport function normalizeHex(hex: string) {\n if (!hex) return '0x';\n const s = String(hex);\n\n const indices = [];\n const re = /0x/gi;\n let m;\n while ((m = re.exec(s)) !== null) {\n indices.push(m.index);\n }\n\n let best = '';\n if (indices.length === 0) {\n best = s.replace(/[^0-9a-fA-F]/g, '');\n } else {\n for (let i = 0; i < indices.length; i++) {\n const start = indices[i] + 2;\n const end = i + 1 < indices.length ? indices[i + 1] : s.length;\n const candidate = s.slice(start, end).replace(/[^0-9a-fA-F]/g, '');\n if (candidate.length > best.length) best = candidate;\n }\n }\n\n if (!best) return '0x';\n if (best.length % 2) throw new Error('hex length must be even');\n return '0x' + best.toLowerCase();\n}\n\nexport function toBytes32(addrOrBytes32: string) {\n const h = normalizeHex(addrOrBytes32);\n if (h.length === 42) return ethers.zeroPadValue(h, 32);\n if (h.length === 66) return h;\n throw new Error('expected address(20B) or bytes32(32B)');\n}\n\nexport function encodeUint(value: string, sizeBytes: number) {\n const bn = BigInt(value);\n const max = (1n << BigInt(sizeBytes * 8)) - 1n;\n\n if (bn < 0n || bn > max) {\n throw new Error(`value does not fit uint${sizeBytes * 8}`);\n }\n\n return ethers.zeroPadValue(ethers.toBeHex(bn), sizeBytes);\n}\n\nexport function buildEndpointV2LzComposePayload(params: {\n nonce: string;\n srcEid: string;\n amountLD: string;\n composeFrom: string;\n composeMsg: string;\n}) {\n const { nonce, srcEid, amountLD, composeFrom, composeMsg } = params;\n return ethers.concat([\n encodeUint(nonce, 8),\n encodeUint(srcEid, 4),\n encodeUint(amountLD, 32),\n toBytes32(composeFrom),\n normalizeHex(composeMsg),\n ]);\n}\n\nexport const NATIVE_MSG_FEE_BUFFER: bigint = 105n;\n"]}
package/dist/index.mjs ADDED
@@ -0,0 +1,92 @@
1
+ import { ethers } from 'ethers';
2
+
3
+ // src/evm-rpc-providers.ts
4
+ var providerCache = /* @__PURE__ */ new Map();
5
+ var SequentialFallbackProvider = class extends ethers.AbstractProvider {
6
+ constructor(providers, chainId) {
7
+ super(chainId);
8
+ this.providers = providers;
9
+ }
10
+ async _detectNetwork() {
11
+ return this.providers[0]._detectNetwork();
12
+ }
13
+ async _perform(req) {
14
+ let lastError;
15
+ for (const provider of this.providers) {
16
+ try {
17
+ return await provider._perform(req);
18
+ } catch (err) {
19
+ if (ethers.isError(err, "CALL_EXCEPTION")) throw err;
20
+ lastError = err;
21
+ }
22
+ }
23
+ throw lastError;
24
+ }
25
+ };
26
+ function evmRpcFallbackProvider(chain) {
27
+ const { rpcUrls, chainId } = chain;
28
+ if (providerCache.has(chainId)) {
29
+ return providerCache.get(chainId);
30
+ }
31
+ const providers = rpcUrls.map(
32
+ (rpc) => {
33
+ return new ethers.JsonRpcProvider(rpc, chainId, { staticNetwork: true });
34
+ }
35
+ );
36
+ const provider = providers.length === 1 ? providers[0] : new SequentialFallbackProvider(providers, chainId);
37
+ providerCache.set(chainId, provider);
38
+ return provider;
39
+ }
40
+ function normalizeHex(hex) {
41
+ if (!hex) return "0x";
42
+ const s = String(hex);
43
+ const indices = [];
44
+ const re = /0x/gi;
45
+ let m;
46
+ while ((m = re.exec(s)) !== null) {
47
+ indices.push(m.index);
48
+ }
49
+ let best = "";
50
+ if (indices.length === 0) {
51
+ best = s.replace(/[^0-9a-fA-F]/g, "");
52
+ } else {
53
+ for (let i = 0; i < indices.length; i++) {
54
+ const start = indices[i] + 2;
55
+ const end = i + 1 < indices.length ? indices[i + 1] : s.length;
56
+ const candidate = s.slice(start, end).replace(/[^0-9a-fA-F]/g, "");
57
+ if (candidate.length > best.length) best = candidate;
58
+ }
59
+ }
60
+ if (!best) return "0x";
61
+ if (best.length % 2) throw new Error("hex length must be even");
62
+ return "0x" + best.toLowerCase();
63
+ }
64
+ function toBytes32(addrOrBytes32) {
65
+ const h = normalizeHex(addrOrBytes32);
66
+ if (h.length === 42) return ethers.zeroPadValue(h, 32);
67
+ if (h.length === 66) return h;
68
+ throw new Error("expected address(20B) or bytes32(32B)");
69
+ }
70
+ function encodeUint(value, sizeBytes) {
71
+ const bn = BigInt(value);
72
+ const max = (1n << BigInt(sizeBytes * 8)) - 1n;
73
+ if (bn < 0n || bn > max) {
74
+ throw new Error(`value does not fit uint${sizeBytes * 8}`);
75
+ }
76
+ return ethers.zeroPadValue(ethers.toBeHex(bn), sizeBytes);
77
+ }
78
+ function buildEndpointV2LzComposePayload(params) {
79
+ const { nonce, srcEid, amountLD, composeFrom, composeMsg } = params;
80
+ return ethers.concat([
81
+ encodeUint(nonce, 8),
82
+ encodeUint(srcEid, 4),
83
+ encodeUint(amountLD, 32),
84
+ toBytes32(composeFrom),
85
+ normalizeHex(composeMsg)
86
+ ]);
87
+ }
88
+ var NATIVE_MSG_FEE_BUFFER = 105n;
89
+
90
+ export { NATIVE_MSG_FEE_BUFFER, buildEndpointV2LzComposePayload, encodeUint, evmRpcFallbackProvider, normalizeHex, toBytes32 };
91
+ //# sourceMappingURL=index.mjs.map
92
+ //# sourceMappingURL=index.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/evm-rpc-providers.ts","../src/layerzero-compose.ts"],"names":["ethers"],"mappings":"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","file":"index.mjs","sourcesContent":["import { ethers } from \"ethers\";\nimport type { TokenConfig } from \"@stableflow/core\";\n\nconst providerCache = new Map<number, ethers.AbstractProvider>();\n\nclass SequentialFallbackProvider extends ethers.AbstractProvider {\n private providers: ethers.JsonRpcProvider[];\n\n constructor(providers: ethers.JsonRpcProvider[], chainId: number) {\n super(chainId);\n this.providers = providers;\n }\n\n async _detectNetwork(): Promise<ethers.Network> {\n return this.providers[0]._detectNetwork();\n }\n\n async _perform(req: ethers.PerformActionRequest): Promise<any> {\n let lastError: unknown;\n for (const provider of this.providers) {\n try {\n return await provider._perform(req);\n } catch (err) {\n if (ethers.isError(err, \"CALL_EXCEPTION\")) throw err;\n lastError = err;\n }\n }\n throw lastError;\n }\n}\n\nexport function evmRpcFallbackProvider(chain: TokenConfig) {\n const { rpcUrls, chainId } = chain;\n\n if (providerCache.has(chainId!)) {\n return providerCache.get(chainId!)!;\n }\n\n const providers = rpcUrls.map(\n (rpc: string) => {\n return new ethers.JsonRpcProvider(rpc, chainId, { staticNetwork: true });\n }\n );\n\n const provider =\n providers.length === 1\n ? providers[0]\n : new SequentialFallbackProvider(providers, chainId!);\n\n providerCache.set(chainId!, provider);\n return provider;\n}\n","import { ethers } from 'ethers';\n\nexport function normalizeHex(hex: string) {\n if (!hex) return '0x';\n const s = String(hex);\n\n const indices = [];\n const re = /0x/gi;\n let m;\n while ((m = re.exec(s)) !== null) {\n indices.push(m.index);\n }\n\n let best = '';\n if (indices.length === 0) {\n best = s.replace(/[^0-9a-fA-F]/g, '');\n } else {\n for (let i = 0; i < indices.length; i++) {\n const start = indices[i] + 2;\n const end = i + 1 < indices.length ? indices[i + 1] : s.length;\n const candidate = s.slice(start, end).replace(/[^0-9a-fA-F]/g, '');\n if (candidate.length > best.length) best = candidate;\n }\n }\n\n if (!best) return '0x';\n if (best.length % 2) throw new Error('hex length must be even');\n return '0x' + best.toLowerCase();\n}\n\nexport function toBytes32(addrOrBytes32: string) {\n const h = normalizeHex(addrOrBytes32);\n if (h.length === 42) return ethers.zeroPadValue(h, 32);\n if (h.length === 66) return h;\n throw new Error('expected address(20B) or bytes32(32B)');\n}\n\nexport function encodeUint(value: string, sizeBytes: number) {\n const bn = BigInt(value);\n const max = (1n << BigInt(sizeBytes * 8)) - 1n;\n\n if (bn < 0n || bn > max) {\n throw new Error(`value does not fit uint${sizeBytes * 8}`);\n }\n\n return ethers.zeroPadValue(ethers.toBeHex(bn), sizeBytes);\n}\n\nexport function buildEndpointV2LzComposePayload(params: {\n nonce: string;\n srcEid: string;\n amountLD: string;\n composeFrom: string;\n composeMsg: string;\n}) {\n const { nonce, srcEid, amountLD, composeFrom, composeMsg } = params;\n return ethers.concat([\n encodeUint(nonce, 8),\n encodeUint(srcEid, 4),\n encodeUint(amountLD, 32),\n toBytes32(composeFrom),\n normalizeHex(composeMsg),\n ]);\n}\n\nexport const NATIVE_MSG_FEE_BUFFER: bigint = 105n;\n"]}
package/package.json ADDED
@@ -0,0 +1,37 @@
1
+ {
2
+ "name": "@stableflow/utils-evm",
3
+ "version": "3.0.0-beta-02a59f37a0-20260515",
4
+ "description": "EVM utilities for StableFlow SDK (RPC fallback, LayerZero compose)",
5
+ "license": "MIT",
6
+ "repository": {
7
+ "type": "git",
8
+ "url": "https://github.com/stableflow-ai/stableflow-sdk.git",
9
+ "directory": "packages/utils-evm"
10
+ },
11
+ "main": "./dist/index.js",
12
+ "module": "./dist/index.mjs",
13
+ "types": "./dist/index.d.ts",
14
+ "exports": {
15
+ ".": {
16
+ "types": "./dist/index.d.ts",
17
+ "import": "./dist/index.mjs",
18
+ "require": "./dist/index.js"
19
+ }
20
+ },
21
+ "files": [
22
+ "dist"
23
+ ],
24
+ "sideEffects": false,
25
+ "dependencies": {
26
+ "ethers": "^6.16.0",
27
+ "@stableflow/core": "3.0.0-beta-02a59f37a0-20260515"
28
+ },
29
+ "devDependencies": {
30
+ "tsup": "^8.5.1",
31
+ "typescript": "^5.9.3"
32
+ },
33
+ "scripts": {
34
+ "build": "tsup",
35
+ "clean": "rm -rf dist"
36
+ }
37
+ }