quantumcoin 7.0.1 → 7.0.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 (36) hide show
  1. package/.gitignore +3 -0
  2. package/README-SDK.md +62 -6
  3. package/README.md +25 -0
  4. package/SPEC.md +3845 -0
  5. package/examples/AllSolidityTypes.sol +184 -0
  6. package/examples/SimpleIERC20.sol +74 -0
  7. package/examples/example-generator-sdk-js.js +95 -0
  8. package/examples/example-generator-sdk-ts.js +95 -0
  9. package/examples/example.js +2 -2
  10. package/examples/package-lock.json +10 -1103
  11. package/examples/package.json +1 -2
  12. package/examples/read-operations.js +1 -2
  13. package/examples/sdk-generator-erc20.inline.json +251 -0
  14. package/examples/solidity-types.ts +43 -0
  15. package/generate-sdk.js +579 -84
  16. package/package.json +27 -6
  17. package/src/abi/interface.d.ts +18 -0
  18. package/src/abi/interface.js +247 -9
  19. package/src/abi/js-abi-coder.js +474 -0
  20. package/src/contract/contract-factory.d.ts +1 -1
  21. package/src/contract/contract-factory.js +14 -2
  22. package/src/contract/contract.d.ts +1 -1
  23. package/src/generator/index.js +834 -59
  24. package/src/index.d.ts +16 -0
  25. package/src/providers/provider.d.ts +13 -11
  26. package/src/types/index.d.ts +462 -0
  27. package/src/types/index.js +9 -0
  28. package/test/e2e/all-solidity-types.dynamic.test.js +200 -0
  29. package/test/e2e/all-solidity-types.fixtures.js +231 -0
  30. package/test/e2e/all-solidity-types.generated-sdks.e2e.test.js +361 -0
  31. package/test/e2e/simple-erc20.generated-sdks.e2e.test.js +144 -0
  32. package/test/e2e/typed-generator.e2e.test.js +6 -6
  33. package/test/unit/generate-contract-cli.test.js +1 -0
  34. package/test/unit/generate-sdk-artifacts-json.test.js +45 -0
  35. package/test/unit/solidity-types.test.js +46 -0
  36. package/test/unit/utils.test.js +1 -1
@@ -0,0 +1,200 @@
1
+ /**
2
+ * @testCategory e2e
3
+ * @blockchainRequired write
4
+ * @transactional true
5
+ * @description Dynamic Contract + ContractFactory test for AllSolidityTypes
6
+ *
7
+ * This deploys `examples/AllSolidityTypes.sol` and invokes every method,
8
+ * verifying roundtrip outputs match the expected values.
9
+ */
10
+
11
+ const { describe, it } = require("node:test");
12
+ const assert = require("node:assert/strict");
13
+ const fs = require("node:fs");
14
+ const path = require("node:path");
15
+ const { execFileSync } = require("node:child_process");
16
+
17
+ const { Initialize } = require("quantumcoin/config");
18
+ const { JsonRpcProvider, Wallet, Contract, ContractFactory, getCreateAddress } = require("quantumcoin");
19
+
20
+ const { getRpcUrl, getChainId, getSolcPath, assertSolcExists } = require("./helpers");
21
+ const {
22
+ buildAllUints,
23
+ buildAllInts,
24
+ buildAllFixedBytes,
25
+ buildAllMisc,
26
+ buildInner,
27
+ buildOuter,
28
+ canonicalize,
29
+ } = require("./all-solidity-types.fixtures");
30
+
31
+ // Hardcoded test wallet (test-only; never use for real funds)
32
+ const TEST_WALLET_ENCRYPTED_JSON =
33
+ "{\"address\":\"1a846abe71c8b989e8337c55d608be81c28ab3b2e40c83eaa2a68d516049aec6\",\"crypto\":{\"cipher\":\"aes-256-ctr\",\"ciphertext\":\"ab7e620dd66cb55ac201b9c6796de92bbb06f3681b5932eabe099871f1f7d79acabe30921a39ad13bfe74f42c515734882b6723760142aa3e26e011df514a534ae47bd15d86badd9c6f17c48d4c892711d54d441ee3a0ee0e5b060f816e79c7badd13ff4c235934b1986774223ecf6e8761388969bb239c759b54c8c70e6a2e27c93a4b70129c8159f461d271ae8f3573414c78b88e4d0abfa6365ed45456636d4ed971c7a0c6b84e6f0c2621e819268b135e2bcc169a54d1847b39e6ba2ae8ec969b69f330b7db9e785ed02204d5a1185915ae5338b0f40ef2a7f4d5aaf7563d502135e57f4eb89d5ec1efa5c77e374969d6cd85be625a2ed1225d68ecdd84067bfc69adb83ecd5c6050472eca28a5a646fcdd28077165c629975bec8a79fe1457cb53389b788b25e1f8eff8b2ca326d7dfcaba3f8839225a08057c018a458891fd2caa0d2b27632cffd80f592147ccec9a10dc8a08a48fb55047bff5cf85cda39eb089096bef63842fc3686412f298a54a9e4b0bf4ad36907ba373cbd6d32e7ac494af371da5aa9d38a3463220865114c4adc5e4ac258ba9c6af9fa2ddfd1aec2e16887e4b3977c69561df8599ac9d411c9dd2a4d57f92ea4e5c02aae3f49fb3bc83e16673e6c2dbe96bb181c8dfd0f9757ade2e4ff27215a836058c5ffeab042f6f97c7c02339f76a6284680e01b4bb733690eb3347fbfcc26614b8bf755f9dfce3fea9d4e4d15b164983201732c2e87593a86bca6da6972e128490338f76ae68135888070f4e59e90db54d23834769bdbda9769213faf5357f9167a224523975a946367b68f0cec98658575609f58bfd329e420a921c06713326e4cb20a0df1d77f37e78a320a637a96c604ca3fa89e24beb42313751b8f09b14f9c14c77e4fd13fc6382505d27c771bca0d821ec7c3765acffa99d83c50140a56b0b28101c762bd682fe55cb6f23cbeb3f421d7b36021010e45ac27160dd7ead99c864a1b550c7edb1246950fe32dcc049799f9085287f0a747a6ef7a023df46a23a22f3e833bbf8d404f84344870492658256ee1dfc40fda33bb8d48fc72d4520ba9fc820c9123104a045206809037709f2a5f6723fa77d6bac5a573823d4ec3a7f1cb786a52ee2697e622e5d75962fa554d1024a6c355e21f33a63b2b72e6c4742a8b1c373aa532b40518c38c90b5373c2eb8c9d7be2a9e16047a3ee09dc9a6849deac5183ace6cfe91a9bef2ffc0a7df6ccebfd4c858c84b0e0355650d7466971e66f1e3883013e5ad1be33199b1d110b79070ac1b745ccb14cf63a08f8cca3a21c9525e626ff5f0c34746e10750fb742ad51f11f2acae3676c2111853d7250d01b77821a6ba9e04400ba2c543ca9f2d701ae6f47bfad14ffe3039ee9e71f7b2401359ade9938750ddb9c5a8b018a7929ed8d0e717ff1861446ce17535e9b17c187711190aae3388bd9490837a636c25ed4d42d7079ad1a51e13292c683d5d012abcf46965c534b83ab53f2c1f0cf5830ef7582e06863a33c19a70511df632885d63245965047ea96b56f1af5b3b94a54999f784fb9574fdfcd7c1230e07a2aaa04acd3097b2b9f8ddba05ae9734491deb5c1a513c76ed276cb78bbf4839dae3156d76af444a5805129d5df791167a9c8576a1d7f760b2d2797c4658669608706fbd0ace1be2346f74862dfc9ef518e55632e43c043186e5d070deb34d12fb9e5aba84e5cb50213dc88efd39cc35bf42455aa82d5e3b707b3140be3b8623b34fdd81d08615c188ae8438a13881fdf6bf32f2cb9ff5fa625561040c6b71d4b8eccc90bc3b99650d28dd1ee63773e49664e3d48c484996b290943635a6f2eb1ce9796d3fa144a3f00ef82faaa32d6a413668f7b521517cb68b2b017fcf56c79326fa5e4060e643631ca3f0a0dc0ed718798b6f46b130d437c33f64039e887324b6f5e604b1669d613923794edbf04b1b3caea54793b52b44b170173a4f25c7ecef3b71e2aad76e556b1cb9f1d637ec52ececfa950dd31dbb6a60828a3ad34c1beffe09eb4785786d63bad10a0b0f66ea88c57380f38ea85f018dbd7f538cf1ee7624095b9a01ec5edd528f281168af020609e651ff316aa1320a710134ddfca600cc72174dcdb846d2aa29916488aa1b537b66da92e61af526debef4eb38c984569eaf549ff2129449269b492d030cd74d885f6f5785881cc4804b4a8a09ba4ff7aefe9074ac7d0c4f05d51fe4cc0ff7388a772092b9d02d70e5433a5cf3e02f46a6bd6b818d59a07ce3b9fbbf8b5faba74563bcc5240930c2d406c9aaee3e3ce0429bf68ac2b0a57adb09414cff50817d2a48fb9fa624ab863cb0c31a8b8dc5eaf6fa68cc1d7c6c685c5a33edd5c8933b9e8ab628ee428d0743699b2ff17f25586c7ce959280bb0b8c5342251f0a30b53dbc7bf1ee426ac9619c3560f811f2268ee37f189794e2e4b3db3a2fb2e34b649e504fb467438abfd1082619cc4a0b30d66beb831077812e418d2e2148db10cf4d4a29101ca52ec445b8d83519dd7de85a98e0beae9ee537096d3f1a55a7a80cdfa93d25f07c9f98e8af18cde19ec1f99c5dd4588b717a5039ddb7f177717caf0d0fd45420a70dbd6d3146890d9e450d5224146db4c33b779e3c3a04b976c052bad042ac57dd38be45407808c0fb0d7e2a8819e6cd53c6739e6612996ddaa6f066552590aa0343bc1e62b298ff2514a0cef8be21956c2e942816f7a3a3a0935eaf9b37251409ce444c986c3817e82835555fe18239f3ae33469d7965c2bde9991fde556bd07af01df52bbde0c35bb4ef48e3b5d0db53f8ca4ed35b83f760f0a1bc4ed9f86e85d6039a17df373c85402ef956f01db00eb39c4b74bd0660d29ee746714d9780d738e05c6cca414ce3d7b40dda8036a9eea9ab1388805f913eb19bdd3f09d9e161eaa50231bd9caba61971f194332dd28c696a60458c1c6c2cc5da8b1192611c7c553e9e12fe48ce46bbb891be8bb118721c86222e671ddd1da8f0ccb2b68e02f2014b4925e904e88369aaf7466bd7033a60c265d45955944916ecbdb84bf1b522b01b0149c632e04c568a7eb627c5bb90ece052ebcf79166c28b30d23fe52da0a5ab5dea83ca479a3e3b7a9cfbbfea04dbe6137c19d067317c2ec427a8c75a6b06bec6dcd5d5c0edc9aa80b9003b8e17c088b2f3db327d3e42630d82d20120240c3ba56232280787da4aabbf5bc95a864029f00710e195f2a76460a0317d10b552fe1bea097e41d49756c680a41d6ac186e62169b6b6cd7776ea84618b5b752328a5bacaa10aa122ff9b2698b43efe73d852a899db644863c8c9bc8068ea86ea843fd6fe36272b91cdc5d5317083ef3fd1e5462a0b0d0604dc57b3bbfceb0fca4cd349625dd7b25166af30efe5ee6a0af953a74d65f4736c59918ee55a3b0d9d9d42e04c7f8a77e479109f740e20c464d5d7e3d16805f47b61f403ff7f408c9e850d9baacd8067e544536a4953480b0f9ee9cd45f41ebd67b51f78788a6470cb1e5ca72ca346ce8a50d0ca0c921d5576a4455a1afb6d0bc688004712ee122cacdb29c51e84893324c27fa4a3f1917edf5352272b4c97579a6152e4b77663d0ab532915f2eeb6a862de8b696452321b660c3f2449673d086e95a7af28845a5259b763e0fcd09f72acf7b6c811066263060e5aa5b24658e880a01fd56bda4dad5ab604e129290f7d5489728f2a40968c6168b21cebbbcd11727cc9e9160c4e92e04387d3b0d62aab06a61f26daedd9fed11816ef2180172a47f47184ac4032b88758c98a2e0fb200f70e93ba695f5ebb7a1029610ad360d3b7fa1b4640b9dc674d3625eef786da93dff19bc7991b5d6193a3896664763fde479b5dfc04812111a80782854f2cf68ca7d82765cc9eb40fba4b44640710ed6e653abf9f07b466333f4fd22784d53cf40e17120f42caa841eaa24056b237827b0f47f7257c103c35027e9f503e5acfd023e7357b600d3084d361d5ee65ba319b45c153212a54e6fed85af7e43e0a926ebcbc2edf8de7e2ec9528f00bec262ad04d5c9dafccaea06a24748d28bf1799bae0e895543084539c50b5aaa4fb50d7431d6f0c8cee2a54aaf7ee7919b55bf40adb688632e5dbe273cea09e97b19c3d8e1f4de000deb66fa1942ad03a62d3252f51992244366c156000b49c297167a6cbdedea7ebae139d295f0ad298e0864249b905b7eb812886ec70ecdb286702274b5b8574149bf3866f9e46b997ff5ed622b169a0eb071347f18d530db1663906a28f4544ee4e004ab87b65476af30ede118052ff052b8dc986ca2c93dd5d4943266a579c7698ea014f688b3e8063a107feb162d392e2177b01bff77fb5abe5feebd0607158049a5a093325b7c9ee6b4dfa7a9f65c7c2fb628920d3603a1c2dad979eaa047cd661a268af1078c9788d720e64e4ce9d12e68de1e417ef2f293323681e1071f9220e1ee43d2e29d111b870ce3439f5100ecd4551ab65ee74aa1667e564957e9bc0ae1ea193980da2a0ec2698073388c85bec25ef447f0d5e93a5203fa44dff268e5cb799ed3b66e63d5e07b487e7534f24934c73a62a243e0151843a0fd3807711a101eaa7fc71f0ba68aebb9534d57cba41b094eebfb4c31cca8eddfa426f676aa347be8a7023a4e91ddb154b35cd4d5f7dbc2e5db491de99f33fc2cff2d57029ac950e1ccd681980af6a4e8969dfe39b3c7bfcbcf8fac92f1e6ec9fe572bfa6a7d65860eab2ed10ac01a71290b52e3148e84b7376a8605cd2bb0e8681ffc54691ce087685e33921bd44d36c78291713dce17569570f62137e6904f0d68cf53aa2ec395c389a75141f08114fb293ea63950e4ffee55ec6fc83cf44876b8e7f25cdd393ff87b9eda6eb746085b61a6900de191f0ce2cb388d61ece52e78bc47368194e8e00277e0d1631e6b9d4626ef76f8522582ccd5a40be3febc699bb510acc6271d55ff0f4cf3bb7669855a72efd9ca3e1056a2fe592a5bc877cce2b1f63b58383971da87873d2d1349cf5881242cdce4e7e2c5c514755746a0e0a7c2a6d9701cde005ae3420beb17c379a3516662253554f51f0423bb1844b0b90c54ed8177ceb0e1036a6609d836e748ca06c40ca64befadc6443ec286a0ce464678e8d11eb455f7bb305acebf6cb1f50e394a9bfeb752df1687831bac9cdd811f4f112ef6658d0f8799a866374ff96c5e2b79f30e7a74f8a2bc9ed1f88f01f30e30cb78ffb2bff10108f35e910ee3be4463e9e6f0ed910e8d598326e71dfa2277ffe5579d7fe9b6018bfe295b25219eae07b3b0270665c3fa00c3e0d180812b5cd62925585de84a7c48a9a86dba96544a251654d1966e082432dc85b6149cf21e91a46020ec32b66d28ba3b6a90c0617bc6fdd55aea819af2bcf84864ad60c28fe3c9f8339d0aee68b39d97f63b6e082835d86119cf9b9fdc8b827c847ce40aa10e1577a710132316845e825345e95bdf94d0c66ec65a6c4319fce4792313663b5f7a651a6710783e6ab71608ac6cbbf3af6911adf596ccf7c172b9bd5bceb6db379967b32b143bdd11d2ee12ddf64ecef6391e0f8570e6cddd3db95204919362b89b739fa94e7c1bfde799fd5e22aa25ca6ca42e30c08e23aae2385d99ebab441072a880dcefdab74a4c9bd39d363f6d1933d59400fca161d432aa00f23b1b1c19a154be8989699d549b66d44e39896f5523443bc6ddf4a65e91f1f3fb7b52318869a05856a4fc92f3694c81ed833c972fb918f7e5\",\"cipherparams\":{\"iv\":\"8c46d6162cd4c765759aedcbce2a5874\"},\"kdf\":\"scrypt\",\"kdfparams\":{\"dklen\":32,\"n\":262144,\"p\":1,\"r\":8,\"salt\":\"82fb6cdc6917609135277badacf15baa31899d08b71a5a0fa33167167c161537\"},\"mac\":\"9187b17f7eca48e6b8c586b0cd790dbe0feb876ac8385f93faa7d5e22a3c8fc7\"},\"id\":\"92caf6ee-2d43-48c0-859e-ffa1e0e23312\",\"version\":3}";
34
+ const TEST_WALLET_PASSPHRASE = "QuantumCoinExample123!";
35
+
36
+ function compileAllSolidityTypes({ solcPath, solPath }) {
37
+ const out = execFileSync(solcPath, ["--optimize", "--combined-json", "abi,bin", solPath], { encoding: "utf8" });
38
+ const parsed = JSON.parse(out);
39
+ const key = Object.keys(parsed.contracts || {}).find((k) => k.endsWith(":AllSolidityTypes"));
40
+ if (!key) throw new Error("AllSolidityTypes not found in solc output");
41
+ const c = parsed.contracts[key];
42
+ const abi = JSON.parse(c.abi);
43
+ const bin = String(c.bin || "");
44
+ if (!bin) throw new Error("solc produced empty bytecode");
45
+ const bytecode = bin.startsWith("0x") ? bin : `0x${bin}`;
46
+ return { abi, bytecode };
47
+ }
48
+
49
+ describe("AllSolidityTypes (dynamic Contract)", () => {
50
+ it("deploys and roundtrips all methods", async (t) => {
51
+ const rpcUrl = getRpcUrl();
52
+ if (!rpcUrl) {
53
+ t.skip("QC_RPC_URL not provided");
54
+ return;
55
+ }
56
+ const chainId = getChainId();
57
+ const solcPath = getSolcPath();
58
+ assertSolcExists(solcPath);
59
+
60
+ const repoRoot = path.resolve(__dirname, "..", "..");
61
+ const solPath = path.join(repoRoot, "examples", "AllSolidityTypes.sol");
62
+ assert.ok(fs.existsSync(solPath), "missing examples/AllSolidityTypes.sol");
63
+
64
+ await Initialize(null);
65
+ const provider = new JsonRpcProvider(rpcUrl, chainId);
66
+ const wallet = Wallet.fromEncryptedJsonSync(TEST_WALLET_ENCRYPTED_JSON, TEST_WALLET_PASSPHRASE, provider);
67
+
68
+ const { abi, bytecode } = compileAllSolidityTypes({ solcPath, solPath });
69
+
70
+ // Build expected values.
71
+ const expectedU = buildAllUints(1);
72
+ const expectedI = buildAllInts(1);
73
+ const expectedFb = buildAllFixedBytes("11");
74
+
75
+ const ctorSeedU256s = [1, 2, 3];
76
+ const ctorSeedB32 = [expectedFb.b32, buildAllFixedBytes("22").b32];
77
+
78
+ const miscParam = buildAllMisc(wallet.address);
79
+ const miscStoredExpected = {
80
+ bo: true,
81
+ addr: wallet.address,
82
+ payableAddr: wallet.address,
83
+ str: "hello",
84
+ dynBytes: "0x1234",
85
+ choice: 1,
86
+ u256s: ctorSeedU256s,
87
+ i256s: [],
88
+ b32s: [],
89
+ addrs: [],
90
+ bools: [],
91
+ strings: [],
92
+ bytesArr: [],
93
+ fixedU16: [0, 0, 0],
94
+ fixedB32: ctorSeedB32,
95
+ };
96
+
97
+ // Deploy with complex constructor args.
98
+ // NOTE: core ContractFactory.deploy does not accept overrides, so we build the tx,
99
+ // estimate gas, and send manually (like other transactional e2e tests).
100
+ const factory = new ContractFactory(abi, bytecode, wallet);
101
+
102
+ // Prefer pending nonce to avoid collisions with in-flight transactions.
103
+ let deployNonce;
104
+ try {
105
+ deployNonce = await provider.getTransactionCount(wallet.address, "pending");
106
+ } catch {
107
+ deployNonce = await provider.getTransactionCount(wallet.address, "latest");
108
+ }
109
+ const expectedAddress = getCreateAddress({ from: wallet.address, nonce: deployNonce });
110
+
111
+ const deployTxReq = factory.getDeployTransaction(
112
+ true,
113
+ wallet.address,
114
+ "hello",
115
+ "0x1234",
116
+ 1,
117
+ expectedU,
118
+ expectedI,
119
+ expectedFb,
120
+ miscParam,
121
+ ctorSeedU256s,
122
+ ctorSeedB32,
123
+ );
124
+
125
+ let gasLimit = 6_000_000;
126
+ try {
127
+ const est = await provider.estimateGas({ from: wallet.address, data: deployTxReq.data });
128
+ gasLimit = Number(est + 100_000n);
129
+ } catch {
130
+ // keep fallback
131
+ }
132
+
133
+ const deployTx = await wallet.sendTransaction({ ...deployTxReq, nonce: deployNonce, gasLimit, value: 0n });
134
+ assert.ok(deployTx && deployTx.hash);
135
+ await deployTx.wait(1, 600_000);
136
+
137
+ const code = await provider.getCode(expectedAddress, "latest");
138
+ assert.ok(typeof code === "string" && code.startsWith("0x"));
139
+ assert.ok(code.length > 2, "no code at deployed address (deployment failed or address mismatch)");
140
+
141
+ const contract = new Contract(expectedAddress, abi, wallet);
142
+ contract._deployTx = deployTx;
143
+
144
+ const unwrap1 = (v) => (Array.isArray(v) ? v[0] : v);
145
+
146
+ // 1) Echo roundtrips
147
+ assert.deepEqual(canonicalize(unwrap1(await contract.echoAllUints(expectedU))), canonicalize(expectedU));
148
+ assert.deepEqual(canonicalize(unwrap1(await contract.echoAllInts(expectedI))), canonicalize(expectedI));
149
+ assert.deepEqual(canonicalize(unwrap1(await contract.echoAllFixedBytes(expectedFb))), canonicalize(expectedFb));
150
+
151
+ const miscEcho = buildAllMisc(wallet.address);
152
+ assert.deepEqual(canonicalize(unwrap1(await contract.echoAllMisc(miscEcho))), canonicalize(miscEcho));
153
+
154
+ const inner = buildInner(wallet.address, 1);
155
+ assert.deepEqual(canonicalize(unwrap1(await contract.echoInner(inner))), canonicalize(inner));
156
+
157
+ const outer = buildOuter(wallet.address);
158
+ assert.deepEqual(canonicalize(unwrap1(await contract.echoOuter(outer))), canonicalize(outer));
159
+
160
+ const uArr = [buildAllUints(3), buildAllUints(4)];
161
+ assert.deepEqual(canonicalize(unwrap1(await contract.echoAllUintsArray(uArr))), canonicalize(uArr));
162
+
163
+ const innerArr = [buildInner(wallet.address, 3), buildInner(wallet.address, 4)];
164
+ assert.deepEqual(canonicalize(unwrap1(await contract.echoInnerArray(innerArr))), canonicalize(innerArr));
165
+
166
+ const matrix = [[1, 2], [3, 4, 5]];
167
+ assert.deepEqual(canonicalize(unwrap1(await contract.echoMatrix(matrix))), canonicalize(matrix));
168
+
169
+ // 2) multiReturn (multiple outputs)
170
+ const multi = await contract.multiReturn(true, wallet.address, expectedFb.b32, "hello", 999, expectedU);
171
+ // Some decoders return an array for multiple returns; canonicalize handles both.
172
+ const multiC = canonicalize(multi);
173
+ // Accept either array (positional) or object (named); compare by re-canonicalizing to object-like.
174
+ if (Array.isArray(multiC)) {
175
+ assert.equal(multiC.length >= 7, true);
176
+ assert.equal(multiC[0] === true || multiC[0] === "true", true);
177
+ assert.equal(multiC[1], wallet.address);
178
+ assert.equal(multiC[2], wallet.address);
179
+ assert.equal(multiC[3], canonicalize(expectedFb.b32));
180
+ assert.equal(multiC[4], "hello");
181
+ assert.equal(multiC[5], "999");
182
+ assert.deepEqual(multiC[6], canonicalize(expectedU));
183
+ } else {
184
+ assert.equal(multiC.outBo === true || multiC.outBo === "true", true);
185
+ assert.equal(multiC.outAddr, wallet.address);
186
+ assert.equal(multiC.outPayableAddr, wallet.address);
187
+ assert.equal(multiC.outB32, canonicalize(expectedFb.b32));
188
+ assert.equal(multiC.outS, "hello");
189
+ assert.equal(multiC.outX, "999");
190
+ assert.deepEqual(multiC.outU, canonicalize(expectedU));
191
+ }
192
+
193
+ // 3) Stored getters
194
+ assert.deepEqual(canonicalize(unwrap1(await contract.getStoredUints())), canonicalize(expectedU));
195
+ assert.deepEqual(canonicalize(unwrap1(await contract.getStoredInts())), canonicalize(expectedI));
196
+ assert.deepEqual(canonicalize(unwrap1(await contract.getStoredFixedBytes())), canonicalize(expectedFb));
197
+ assert.deepEqual(canonicalize(unwrap1(await contract.getStoredMisc())), canonicalize(miscStoredExpected));
198
+ }, { timeout: 1_800_000 });
199
+ });
200
+
@@ -0,0 +1,231 @@
1
+ /**
2
+ * Shared fixtures for AllSolidityTypes contract tests.
3
+ *
4
+ * We use small integers (as strings or numbers) to avoid BigInt interop issues
5
+ * in the underlying WASM ABI encoder.
6
+ */
7
+
8
+ const { normalizeHex } = require("../../src/internal/hex");
9
+
10
+ function _hexOf(byteHex, lenBytes) {
11
+ const b = byteHex.replace(/^0x/i, "");
12
+ return normalizeHex("0x" + b.repeat(lenBytes));
13
+ }
14
+
15
+ function _num(n) {
16
+ // Use number literals for ABI packing (WASM distinguishes numbers vs strings).
17
+ return Number(n);
18
+ }
19
+
20
+ function buildAllUints(seed) {
21
+ const s = seed || 1;
22
+ return {
23
+ u8: _num(8 + s),
24
+ u16: _num(16 + s),
25
+ u24: _num(24 + s),
26
+ u32: _num(32 + s),
27
+ u40: _num(40 + s),
28
+ u48: _num(48 + s),
29
+ u56: _num(56 + s),
30
+ u64: _num(64 + s),
31
+ u72: _num(72 + s),
32
+ u80: _num(80 + s),
33
+ u88: _num(88 + s),
34
+ u96: _num(96 + s),
35
+ u104: _num(104 + s),
36
+ u112: _num(112 + s),
37
+ u120: _num(120 + s),
38
+ u128: _num(128 + s),
39
+ u136: _num(136 + s),
40
+ u144: _num(144 + s),
41
+ u152: _num(152 + s),
42
+ u160: _num(160 + s),
43
+ u168: _num(168 + s),
44
+ u176: _num(176 + s),
45
+ u184: _num(184 + s),
46
+ u192: _num(192 + s),
47
+ u200: _num(200 + s),
48
+ u208: _num(208 + s),
49
+ u216: _num(216 + s),
50
+ u224: _num(224 + s),
51
+ u232: _num(232 + s),
52
+ u240: _num(240 + s),
53
+ u248: _num(248 + s),
54
+ u256: _num(256 + s),
55
+ };
56
+ }
57
+
58
+ function buildAllInts(seed) {
59
+ const s = seed || 1;
60
+ return {
61
+ i8: _num(-8 - s),
62
+ i16: _num(-16 - s),
63
+ i24: _num(-24 - s),
64
+ i32: _num(-32 - s),
65
+ i40: _num(-40 - s),
66
+ i48: _num(-48 - s),
67
+ i56: _num(-56 - s),
68
+ i64: _num(-64 - s),
69
+ i72: _num(-72 - s),
70
+ i80: _num(-80 - s),
71
+ i88: _num(-88 - s),
72
+ i96: _num(-96 - s),
73
+ i104: _num(-104 - s),
74
+ i112: _num(-112 - s),
75
+ i120: _num(-120 - s),
76
+ i128: _num(-128 - s),
77
+ i136: _num(-136 - s),
78
+ i144: _num(-144 - s),
79
+ i152: _num(-152 - s),
80
+ i160: _num(-160 - s),
81
+ i168: _num(-168 - s),
82
+ i176: _num(-176 - s),
83
+ i184: _num(-184 - s),
84
+ i192: _num(-192 - s),
85
+ i200: _num(-200 - s),
86
+ i208: _num(-208 - s),
87
+ i216: _num(-216 - s),
88
+ i224: _num(-224 - s),
89
+ i232: _num(-232 - s),
90
+ i240: _num(-240 - s),
91
+ i248: _num(-248 - s),
92
+ i256: _num(-256 - s),
93
+ };
94
+ }
95
+
96
+ function buildAllFixedBytes(seedByte) {
97
+ const b = seedByte || "11";
98
+ return {
99
+ b1: _hexOf(b, 1),
100
+ b2: _hexOf(b, 2),
101
+ b3: _hexOf(b, 3),
102
+ b4: _hexOf(b, 4),
103
+ b5: _hexOf(b, 5),
104
+ b6: _hexOf(b, 6),
105
+ b7: _hexOf(b, 7),
106
+ b8: _hexOf(b, 8),
107
+ b9: _hexOf(b, 9),
108
+ b10: _hexOf(b, 10),
109
+ b11: _hexOf(b, 11),
110
+ b12: _hexOf(b, 12),
111
+ b13: _hexOf(b, 13),
112
+ b14: _hexOf(b, 14),
113
+ b15: _hexOf(b, 15),
114
+ b16: _hexOf(b, 16),
115
+ b17: _hexOf(b, 17),
116
+ b18: _hexOf(b, 18),
117
+ b19: _hexOf(b, 19),
118
+ b20: _hexOf(b, 20),
119
+ b21: _hexOf(b, 21),
120
+ b22: _hexOf(b, 22),
121
+ b23: _hexOf(b, 23),
122
+ b24: _hexOf(b, 24),
123
+ b25: _hexOf(b, 25),
124
+ b26: _hexOf(b, 26),
125
+ b27: _hexOf(b, 27),
126
+ b28: _hexOf(b, 28),
127
+ b29: _hexOf(b, 29),
128
+ b30: _hexOf(b, 30),
129
+ b31: _hexOf(b, 31),
130
+ b32: _hexOf(b, 32),
131
+ };
132
+ }
133
+
134
+ function buildAllMisc(walletAddress) {
135
+ const addr = walletAddress;
136
+ return {
137
+ bo: true,
138
+ addr,
139
+ payableAddr: addr,
140
+ str: "hello",
141
+ dynBytes: normalizeHex("0x1234"),
142
+ choice: 1,
143
+ u256s: [1, 2, 3],
144
+ i256s: [-1, -2],
145
+ b32s: [_hexOf("aa", 32), _hexOf("bb", 32)],
146
+ addrs: [addr],
147
+ bools: [true, false, true],
148
+ strings: ["a", "b"],
149
+ // NOTE: this SDK's hex parser rejects "0x" (empty). Use "0x00" for non-empty BytesLike.
150
+ bytesArr: [normalizeHex("0x00"), normalizeHex("0x12")],
151
+ fixedU16: [1, 2, 3],
152
+ fixedB32: [_hexOf("cc", 32), _hexOf("dd", 32)],
153
+ };
154
+ }
155
+
156
+ function buildInner(walletAddress, seed) {
157
+ const u = buildAllUints(seed);
158
+ const i = buildAllInts(seed);
159
+ const fb = buildAllFixedBytes(seed % 2 ? "11" : "22");
160
+ const misc = buildAllMisc(walletAddress);
161
+ const u2 = buildAllUints((seed || 1) + 10);
162
+ const fb2 = buildAllFixedBytes(seed % 2 ? "33" : "44");
163
+
164
+ return {
165
+ u,
166
+ i,
167
+ fb,
168
+ misc,
169
+ uStructs: [u2],
170
+ fixedFb: [fb, fb2],
171
+ matrix: [[1, 2], [3]],
172
+ };
173
+ }
174
+
175
+ function buildOuter(walletAddress) {
176
+ const inner1 = buildInner(walletAddress, 1);
177
+ const inner2 = buildInner(walletAddress, 2);
178
+ return {
179
+ inner: inner1,
180
+ inners: [inner2],
181
+ fixedInners: [inner1, inner2],
182
+ b32Matrix: [[_hexOf("01", 32)], [_hexOf("02", 32), _hexOf("03", 32)]],
183
+ };
184
+ }
185
+
186
+ function canonicalize(value) {
187
+ if (value == null) return value;
188
+
189
+ if (typeof value === "bigint") return value.toString();
190
+
191
+ if (typeof value === "number") {
192
+ if (!Number.isFinite(value)) return String(value);
193
+ // Always stringify numbers to be stable across decode implementations.
194
+ return String(value);
195
+ }
196
+
197
+ if (typeof value === "string") {
198
+ // Normalize hex strings (addresses/bytes) consistently.
199
+ if (/^0x[0-9a-fA-F]*$/.test(value)) {
200
+ return normalizeHex(value);
201
+ }
202
+ // Normalize numeric strings by leaving as-is (but trim).
203
+ if (/^-?\d+$/.test(value.trim())) return value.trim();
204
+ return value;
205
+ }
206
+
207
+ if (Array.isArray(value)) return value.map(canonicalize);
208
+
209
+ if (value && typeof value === "object") {
210
+ const keys = Object.keys(value);
211
+ const hasNonNumeric = keys.some((k) => !/^\d+$/.test(k));
212
+ const picked = hasNonNumeric ? keys.filter((k) => !/^\d+$/.test(k)) : keys;
213
+ picked.sort();
214
+ const out = {};
215
+ for (const k of picked) out[k] = canonicalize(value[k]);
216
+ return out;
217
+ }
218
+
219
+ return value;
220
+ }
221
+
222
+ module.exports = {
223
+ buildAllUints,
224
+ buildAllInts,
225
+ buildAllFixedBytes,
226
+ buildAllMisc,
227
+ buildInner,
228
+ buildOuter,
229
+ canonicalize,
230
+ };
231
+