quantumcoin 7.0.2 → 7.0.4

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 (155) hide show
  1. package/.github/workflows/publish-npmjs.yaml +22 -22
  2. package/.gitignore +15 -15
  3. package/LICENSE +21 -21
  4. package/README-SDK.md +756 -756
  5. package/README.md +165 -152
  6. package/SPEC.md +3845 -3845
  7. package/config.d.ts +50 -50
  8. package/config.js +115 -115
  9. package/examples/AllSolidityTypes.sol +184 -184
  10. package/examples/SimpleIERC20.sol +74 -74
  11. package/examples/events.js +41 -35
  12. package/examples/events.ts +35 -0
  13. package/examples/example-generator-sdk-js.js +100 -95
  14. package/examples/example-generator-sdk-js.ts +77 -0
  15. package/examples/example-generator-sdk-ts.js +100 -95
  16. package/examples/example-generator-sdk-ts.ts +77 -0
  17. package/examples/example.js +72 -61
  18. package/examples/example.ts +61 -0
  19. package/examples/offline-signing.js +79 -0
  20. package/examples/offline-signing.ts +66 -0
  21. package/examples/package-lock.json +48 -57
  22. package/examples/package.json +32 -16
  23. package/examples/read-operations.js +32 -27
  24. package/examples/read-operations.ts +31 -0
  25. package/examples/sdk-generator-erc20.inline.json +251 -251
  26. package/examples/solidity-types.ts +43 -43
  27. package/examples/wallet-offline.js +35 -29
  28. package/examples/wallet-offline.ts +34 -0
  29. package/generate-sdk.js +1824 -1383
  30. package/index.js +12 -12
  31. package/package.json +95 -75
  32. package/scripts/copy-declarations.js +31 -0
  33. package/scripts/run-all-one-by-one.js +151 -0
  34. package/src/abi/fragments.d.ts +42 -42
  35. package/src/abi/fragments.js +63 -63
  36. package/src/abi/index.d.ts +13 -13
  37. package/src/abi/index.js +9 -9
  38. package/src/abi/interface.d.ts +128 -132
  39. package/src/abi/interface.js +590 -590
  40. package/src/abi/js-abi-coder.d.ts +8 -0
  41. package/src/abi/js-abi-coder.js +474 -474
  42. package/src/constants.d.ts +66 -61
  43. package/src/constants.js +101 -94
  44. package/src/contract/contract-factory.d.ts +28 -28
  45. package/src/contract/contract-factory.js +105 -105
  46. package/src/contract/contract.d.ts +113 -105
  47. package/src/contract/contract.js +354 -312
  48. package/src/contract/index.d.ts +9 -9
  49. package/src/contract/index.js +9 -9
  50. package/src/errors/index.d.ts +92 -92
  51. package/src/errors/index.js +188 -188
  52. package/src/generator/index.d.ts +74 -0
  53. package/src/generator/index.js +1404 -1201
  54. package/src/index.d.ts +125 -127
  55. package/src/index.js +41 -41
  56. package/src/internal/hex.d.ts +61 -61
  57. package/src/internal/hex.js +144 -144
  58. package/src/providers/extra-providers.d.ts +139 -128
  59. package/src/providers/extra-providers.js +600 -575
  60. package/src/providers/index.d.ts +17 -16
  61. package/src/providers/index.js +10 -10
  62. package/src/providers/json-rpc-provider.d.ts +12 -12
  63. package/src/providers/json-rpc-provider.js +79 -79
  64. package/src/providers/provider.d.ts +207 -196
  65. package/src/providers/provider.js +392 -359
  66. package/src/types/index.d.ts +214 -462
  67. package/src/types/index.js +9 -9
  68. package/src/utils/address.d.ts +72 -72
  69. package/src/utils/address.js +181 -182
  70. package/src/utils/encoding.d.ts +120 -120
  71. package/src/utils/encoding.js +306 -306
  72. package/src/utils/hashing.d.ts +82 -76
  73. package/src/utils/hashing.js +313 -298
  74. package/src/utils/index.d.ts +65 -55
  75. package/src/utils/index.js +13 -13
  76. package/src/utils/result.d.ts +57 -57
  77. package/src/utils/result.js +128 -128
  78. package/src/utils/rlp.d.ts +12 -12
  79. package/src/utils/rlp.js +200 -200
  80. package/src/utils/units.d.ts +29 -29
  81. package/src/utils/units.js +107 -107
  82. package/src/wallet/index.d.ts +10 -10
  83. package/src/wallet/index.js +8 -8
  84. package/src/wallet/wallet.d.ts +160 -160
  85. package/src/wallet/wallet.js +483 -489
  86. package/test/e2e/all-solidity-types.dynamic.test.js +207 -200
  87. package/test/e2e/all-solidity-types.dynamic.test.ts +191 -0
  88. package/test/e2e/all-solidity-types.fixtures.js +231 -231
  89. package/test/e2e/all-solidity-types.generated-sdks.e2e.test.js +387 -361
  90. package/test/e2e/all-solidity-types.generated-sdks.e2e.test.ts +350 -0
  91. package/test/e2e/helpers.js +59 -47
  92. package/test/e2e/signing-context-and-fee.e2e.test.js +137 -0
  93. package/test/e2e/signing-context-and-fee.e2e.test.ts +128 -0
  94. package/test/e2e/simple-erc20.generated-sdks.e2e.test.js +168 -144
  95. package/test/e2e/simple-erc20.generated-sdks.e2e.test.ts +141 -0
  96. package/test/e2e/transactional.test.js +245 -191
  97. package/test/e2e/transactional.test.ts +208 -0
  98. package/test/e2e/typed-generator.e2e.test.js +407 -402
  99. package/test/e2e/typed-generator.e2e.test.ts +337 -0
  100. package/test/fixtures/ConstructorParam.sol +23 -23
  101. package/test/fixtures/MultiContracts.sol +37 -37
  102. package/test/fixtures/SimpleStorage.sol +18 -18
  103. package/test/fixtures/StakingContract.abi.json +1 -1
  104. package/test/integration/ipc-provider.test.js +49 -44
  105. package/test/integration/ipc-provider.test.ts +44 -0
  106. package/test/integration/provider.test.js +88 -72
  107. package/test/integration/provider.test.ts +85 -0
  108. package/test/integration/ws-provider.test.js +41 -33
  109. package/test/integration/ws-provider.test.ts +38 -0
  110. package/test/security/malformed-input.test.js +37 -31
  111. package/test/security/malformed-input.test.ts +35 -0
  112. package/test/unit/_encrypted-output.txt +6 -0
  113. package/test/unit/_log-encrypted-jsons.js +45 -0
  114. package/test/unit/_write-keystore-fixture.js +16 -0
  115. package/test/unit/abi-interface.test.js +103 -98
  116. package/test/unit/abi-interface.test.ts +102 -0
  117. package/test/unit/address-wallet.test.js +355 -257
  118. package/test/unit/address-wallet.test.ts +342 -0
  119. package/test/unit/browser-provider.test.js +85 -82
  120. package/test/unit/browser-provider.test.ts +79 -0
  121. package/test/unit/contract.test.js +85 -82
  122. package/test/unit/contract.test.ts +83 -0
  123. package/test/unit/encoding-units-rlp.test.js +92 -89
  124. package/test/unit/encoding-units-rlp.test.ts +91 -0
  125. package/test/unit/errors.test.js +77 -74
  126. package/test/unit/errors.test.ts +76 -0
  127. package/test/unit/filter-by-blockhash.test.js +55 -52
  128. package/test/unit/filter-by-blockhash.test.ts +54 -0
  129. package/test/unit/fixtures/encrypted-keystores-48-32-36.js +9 -0
  130. package/test/unit/generate-contract-cli.test.js +42 -39
  131. package/test/unit/generate-contract-cli.test.ts +41 -0
  132. package/test/unit/generate-sdk-artifacts-json.test.js +113 -110
  133. package/test/unit/generate-sdk-artifacts-json.test.ts +110 -0
  134. package/test/unit/generator.test.js +102 -98
  135. package/test/unit/generator.test.ts +101 -0
  136. package/test/unit/hashing.test.js +68 -54
  137. package/test/unit/hashing.test.ts +67 -0
  138. package/test/unit/init.test.js +39 -36
  139. package/test/unit/init.test.ts +38 -0
  140. package/test/unit/interface.test.js +56 -53
  141. package/test/unit/interface.test.ts +54 -0
  142. package/test/unit/internal-hex.test.js +50 -47
  143. package/test/unit/internal-hex.test.ts +49 -0
  144. package/test/unit/populate-transaction.test.js +65 -0
  145. package/test/unit/populate-transaction.test.ts +64 -0
  146. package/test/unit/providers.test.js +200 -144
  147. package/test/unit/providers.test.ts +196 -0
  148. package/test/unit/result.test.js +80 -77
  149. package/test/unit/result.test.ts +79 -0
  150. package/test/unit/solidity-types.test.js +49 -46
  151. package/test/unit/solidity-types.test.ts +39 -0
  152. package/test/unit/utils.test.js +57 -54
  153. package/test/unit/utils.test.ts +56 -0
  154. package/test/verbose-logger.js +74 -0
  155. package/tsconfig.build.json +14 -0
@@ -1,361 +1,387 @@
1
- /**
2
- * @testCategory e2e
3
- * @blockchainRequired write
4
- * @transactional true
5
- *
6
- * Generates typed SDK packages (TS and JS) for AllSolidityTypes.sol,
7
- * then injects additional tests that call all methods and validate outputs.
8
- */
9
-
10
- const { describe, it } = require("node:test");
11
- const assert = require("node:assert/strict");
12
- const fs = require("node:fs");
13
- const path = require("node:path");
14
- const { spawnSync } = require("node:child_process");
15
-
16
- const { getRpcUrl, getChainId } = require("./helpers");
17
-
18
- function getNpmCmd() {
19
- return process.platform === "win32" ? "npm.cmd" : "npm";
20
- }
21
-
22
- function run(cmd, args, cwd, env) {
23
- const res = spawnSync(cmd, args, {
24
- cwd,
25
- env,
26
- encoding: "utf8",
27
- stdio: "pipe",
28
- shell: false,
29
- windowsHide: true,
30
- });
31
- if (res.error) throw res.error;
32
- return res;
33
- }
34
-
35
- function _quoteIfNeeded(s) {
36
- if (typeof s !== "string") return s;
37
- return /[ \t"]/g.test(s) ? `"${s.replace(/"/g, '\\"')}"` : s;
38
- }
39
-
40
- function runNpm(args, cwd, env) {
41
- if (process.platform === "win32") {
42
- const cmd = `${getNpmCmd()} ${args.map(_quoteIfNeeded).join(" ")}`;
43
- return run("cmd.exe", ["/d", "/s", "/c", cmd], cwd, env);
44
- }
45
- return run(getNpmCmd(), args, cwd, env);
46
- }
47
-
48
- function writeExtraTest(pkgRoot) {
49
- const testDir = path.join(pkgRoot, "test", "e2e");
50
- fs.mkdirSync(testDir, { recursive: true });
51
-
52
- const content = `/**
53
- * @testCategory e2e
54
- * @blockchainRequired write
55
- * @transactional true
56
- * @description Extra exhaustive checks for AllSolidityTypes
57
- */
58
-
59
- const { describe, it } = require("node:test");
60
- const assert = require("node:assert/strict");
61
-
62
- const { Initialize } = require("quantumcoin/config");
63
- const { JsonRpcProvider, Wallet } = require("quantumcoin");
64
-
65
- // Require the generated package root (works for both TS and JS packages)
66
- const { AllSolidityTypes__factory } = require("../..");
67
-
68
- // Hardcoded test wallet (test-only; never use for real funds)
69
- const TEST_WALLET_ENCRYPTED_JSON =
70
- ${JSON.stringify(
71
- "{\"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}",
72
- )};
73
- const TEST_WALLET_PASSPHRASE = "QuantumCoinExample123!";
74
-
75
- function normalizeHex(s) {
76
- const t = String(s || "").trim();
77
- const x = t.startsWith("0x") ? t : "0x" + t;
78
- return "0x" + x.slice(2).toLowerCase();
79
- }
80
-
81
- function canon(v) {
82
- if (v == null) return v;
83
- if (typeof v === "bigint") return v.toString();
84
- if (typeof v === "number") return String(v);
85
- if (typeof v === "string") {
86
- if (/^0x[0-9a-fA-F]*$/.test(v)) return normalizeHex(v);
87
- if (/^-?\\d+$/.test(v.trim())) return v.trim();
88
- return v;
89
- }
90
- if (Array.isArray(v)) return v.map(canon);
91
- if (typeof v === "object") {
92
- const keys = Object.keys(v);
93
- const hasNonNumeric = keys.some((k) => !/^\\d+$/.test(k));
94
- const picked = hasNonNumeric ? keys.filter((k) => !/^\\d+$/.test(k)) : keys;
95
- picked.sort();
96
- const out = {};
97
- for (const k of picked) out[k] = canon(v[k]);
98
- return out;
99
- }
100
- return v;
101
- }
102
-
103
- function hexOf(byteHex, lenBytes) {
104
- const b = byteHex.replace(/^0x/i, "");
105
- return normalizeHex("0x" + b.repeat(lenBytes));
106
- }
107
-
108
- function buildAllUints(seed) {
109
- const s = seed || 1;
110
- const num = (n) => Number(n);
111
- return {
112
- u8: num(8 + s), u16: num(16 + s), u24: num(24 + s), u32: num(32 + s), u40: num(40 + s), u48: num(48 + s), u56: num(56 + s), u64: num(64 + s),
113
- u72: num(72 + s), u80: num(80 + s), u88: num(88 + s), u96: num(96 + s), u104: num(104 + s), u112: num(112 + s), u120: num(120 + s), u128: num(128 + s),
114
- u136: num(136 + s), u144: num(144 + s), u152: num(152 + s), u160: num(160 + s), u168: num(168 + s), u176: num(176 + s), u184: num(184 + s), u192: num(192 + s),
115
- u200: num(200 + s), u208: num(208 + s), u216: num(216 + s), u224: num(224 + s), u232: num(232 + s), u240: num(240 + s), u248: num(248 + s), u256: num(256 + s),
116
- };
117
- }
118
-
119
- function buildAllInts(seed) {
120
- const s = seed || 1;
121
- const num = (n) => Number(n);
122
- return {
123
- i8: num(-8 - s), i16: num(-16 - s), i24: num(-24 - s), i32: num(-32 - s), i40: num(-40 - s), i48: num(-48 - s), i56: num(-56 - s), i64: num(-64 - s),
124
- i72: num(-72 - s), i80: num(-80 - s), i88: num(-88 - s), i96: num(-96 - s), i104: num(-104 - s), i112: num(-112 - s), i120: num(-120 - s), i128: num(-128 - s),
125
- i136: num(-136 - s), i144: num(-144 - s), i152: num(-152 - s), i160: num(-160 - s), i168: num(-168 - s), i176: num(-176 - s), i184: num(-184 - s), i192: num(-192 - s),
126
- i200: num(-200 - s), i208: num(-208 - s), i216: num(-216 - s), i224: num(-224 - s), i232: num(-232 - s), i240: num(-240 - s), i248: num(-248 - s), i256: num(-256 - s),
127
- };
128
- }
129
-
130
- function buildAllFixedBytes(seedByte) {
131
- const b = seedByte || "11";
132
- const o = {};
133
- for (let i = 1; i <= 32; i++) o["b" + i] = hexOf(b, i);
134
- return o;
135
- }
136
-
137
- function buildAllMisc(addr) {
138
- return {
139
- bo: true,
140
- addr,
141
- payableAddr: addr,
142
- str: "hello",
143
- dynBytes: normalizeHex("0x1234"),
144
- choice: 1,
145
- u256s: [1, 2, 3],
146
- i256s: [-1, -2],
147
- b32s: [hexOf("aa", 32), hexOf("bb", 32)],
148
- addrs: [addr],
149
- bools: [true, false, true],
150
- strings: ["a", "b"],
151
- bytesArr: [normalizeHex("0x00"), normalizeHex("0x12")],
152
- fixedU16: [1, 2, 3],
153
- fixedB32: [hexOf("cc", 32), hexOf("dd", 32)],
154
- };
155
- }
156
-
157
- function buildInner(addr, seed) {
158
- const u = buildAllUints(seed);
159
- const i = buildAllInts(seed);
160
- const fb = buildAllFixedBytes(seed % 2 ? "11" : "22");
161
- const misc = buildAllMisc(addr);
162
- const u2 = buildAllUints((seed || 1) + 10);
163
- const fb2 = buildAllFixedBytes(seed % 2 ? "33" : "44");
164
- return { u, i, fb, misc, uStructs: [u2], fixedFb: [fb, fb2], matrix: [[1,2],[3]] };
165
- }
166
-
167
- function buildOuter(addr) {
168
- const inner1 = buildInner(addr, 1);
169
- const inner2 = buildInner(addr, 2);
170
- return { inner: inner1, inners: [inner2], fixedInners: [inner1, inner2], b32Matrix: [[hexOf("01",32)],[hexOf("02",32),hexOf("03",32)]] };
171
- }
172
-
173
- describe("AllSolidityTypes (extra)", () => {
174
- it("roundtrips all methods", async (t) => {
175
- const rpcUrl = process.env.QC_RPC_URL;
176
- if (!rpcUrl) { t.skip("QC_RPC_URL not provided"); return; }
177
- const chainId = process.env.QC_CHAIN_ID ? Number(process.env.QC_CHAIN_ID) : 123123;
178
- await Initialize(null);
179
-
180
- const provider = new JsonRpcProvider(rpcUrl, chainId);
181
- const wallet = Wallet.fromEncryptedJsonSync(TEST_WALLET_ENCRYPTED_JSON, TEST_WALLET_PASSPHRASE, provider);
182
-
183
- const expectedU = buildAllUints(1);
184
- const expectedI = buildAllInts(1);
185
- const expectedFb = buildAllFixedBytes("11");
186
-
187
- const ctorSeedU256s = [1,2,3];
188
- const ctorSeedB32 = [expectedFb.b32, buildAllFixedBytes("22").b32];
189
- const miscParam = buildAllMisc(wallet.address);
190
-
191
- const factory = new AllSolidityTypes__factory(wallet);
192
- const contract = await factory.deploy(
193
- true,
194
- wallet.address,
195
- "hello",
196
- "0x1234",
197
- 1,
198
- expectedU,
199
- expectedI,
200
- expectedFb,
201
- miscParam,
202
- ctorSeedU256s,
203
- ctorSeedB32,
204
- { gasLimit: 6_000_000 },
205
- );
206
-
207
- const tx = contract.deployTransaction();
208
- if (tx) await tx.wait(1, 600_000);
209
-
210
- assert.deepEqual(canon(await contract.echoAllUints(expectedU)), canon(expectedU));
211
- assert.deepEqual(canon(await contract.echoAllInts(expectedI)), canon(expectedI));
212
- assert.deepEqual(canon(await contract.echoAllFixedBytes(expectedFb)), canon(expectedFb));
213
- assert.deepEqual(canon(await contract.echoAllMisc(miscParam)), canon(miscParam));
214
-
215
- const inner = buildInner(wallet.address, 1);
216
- assert.deepEqual(canon(await contract.echoInner(inner)), canon(inner));
217
-
218
- const outer = buildOuter(wallet.address);
219
- assert.deepEqual(canon(await contract.echoOuter(outer)), canon(outer));
220
-
221
- const uArr = [buildAllUints(3), buildAllUints(4)];
222
- assert.deepEqual(canon(await contract.echoAllUintsArray(uArr)), canon(uArr));
223
-
224
- const innerArr = [buildInner(wallet.address, 3), buildInner(wallet.address, 4)];
225
- assert.deepEqual(canon(await contract.echoInnerArray(innerArr)), canon(innerArr));
226
-
227
- const matrix = [[1,2],[3,4,5]];
228
- assert.deepEqual(canon(await contract.echoMatrix(matrix)), canon(matrix));
229
-
230
- const multi = await contract.multiReturn(true, wallet.address, expectedFb.b32, "hello", 999, expectedU);
231
- const multiC = canon(multi);
232
- if (Array.isArray(multiC)) {
233
- assert.equal(multiC[0] === true || multiC[0] === "true", true);
234
- assert.equal(multiC[1], wallet.address);
235
- assert.equal(multiC[2], wallet.address);
236
- assert.equal(multiC[3], canon(expectedFb.b32));
237
- assert.equal(multiC[4], "hello");
238
- assert.equal(multiC[5], "999");
239
- assert.deepEqual(multiC[6], canon(expectedU));
240
- } else {
241
- assert.equal(multiC.outBo === true || multiC.outBo === "true", true);
242
- assert.equal(multiC.outAddr, wallet.address);
243
- assert.equal(multiC.outPayableAddr, wallet.address);
244
- assert.equal(multiC.outB32, canon(expectedFb.b32));
245
- assert.equal(multiC.outS, "hello");
246
- assert.equal(multiC.outX, "999");
247
- assert.deepEqual(multiC.outU, canon(expectedU));
248
- }
249
- }, { timeout: 1_800_000 });
250
- });
251
- `;
252
-
253
- fs.writeFileSync(path.join(testDir, "AllSolidityTypes.extra.test.js"), content, "utf8");
254
- }
255
-
256
- function assertNoLegacyGenericTypes(pkgRoot, contractName, lang) {
257
- const srcDir = path.join(pkgRoot, "src");
258
- const files =
259
- lang === "ts"
260
- ? [path.join(srcDir, `${contractName}.ts`)]
261
- : [path.join(srcDir, `${contractName}.js`), path.join(srcDir, `${contractName}.d.ts`)];
262
- for (const file of files) {
263
- const text = fs.readFileSync(file, "utf8");
264
- assert.equal(/SolidityInputValue\s*</.test(text), false, `${path.basename(file)} still contains SolidityInputValue<>`);
265
- assert.equal(/SolidityOutputValue\s*</.test(text), false, `${path.basename(file)} still contains SolidityOutputValue<>`);
266
- assert.equal(/Promise<any>/.test(text), false, `${path.basename(file)} still contains Promise<any>`);
267
- }
268
- }
269
-
270
- describe("AllSolidityTypes generated SDKs (extra tests)", () => {
271
- it("generates TS and JS packages and runs extra tests", async (t) => {
272
- const rpcUrl = getRpcUrl();
273
- if (!rpcUrl) {
274
- t.skip("QC_RPC_URL not provided");
275
- return;
276
- }
277
-
278
- const chainId = getChainId();
279
- const repoRoot = path.resolve(__dirname, "..", "..");
280
- const solPath = path.join(repoRoot, "examples", "AllSolidityTypes.sol");
281
- assert.ok(fs.existsSync(solPath), "missing examples/AllSolidityTypes.sol");
282
-
283
- // Store generated SDK packages under test/e2e so they're easy to inspect.
284
- // NOTE: This folder is .gitignored (see root .gitignore).
285
- const outBase = path.join(repoRoot, "test", "e2e", "generated-sdks", "all-solidity-types");
286
- fs.mkdirSync(outBase, { recursive: true });
287
-
288
- const mkPkg = (lang) => {
289
- const pkgName = `all-solidity-types-${lang}`;
290
- const pkgRoot = path.join(outBase, pkgName);
291
- // Ensure deterministic output by clearing any previous run.
292
- fs.rmSync(pkgRoot, { recursive: true, force: true });
293
- const genCli = path.join(repoRoot, "generate-sdk.js");
294
- const res = run(
295
- process.execPath,
296
- [
297
- genCli,
298
- "--lang",
299
- lang,
300
- "--sol",
301
- solPath,
302
- "--name",
303
- "AllSolidityTypes",
304
- "--create-package",
305
- "--package-dir",
306
- outBase,
307
- "--package-name",
308
- pkgName,
309
- "--package-description",
310
- `${lang.toUpperCase()} typed package generated from AllSolidityTypes.sol (e2e)`,
311
- "--package-author",
312
- "quantumcoin.js test",
313
- "--package-license",
314
- "MIT",
315
- "--package-version",
316
- "0.0.1",
317
- "--non-interactive",
318
- ],
319
- repoRoot,
320
- process.env,
321
- );
322
- assert.equal(res.status, 0, `generator failed:\n${res.stdout}\n${res.stderr}`);
323
- return pkgRoot;
324
- };
325
-
326
- let succeeded = false;
327
- try {
328
- const tsPkg = mkPkg("ts");
329
- const jsPkg = mkPkg("js");
330
-
331
- // Inject extra tests
332
- writeExtraTest(tsPkg);
333
- writeExtraTest(jsPkg);
334
-
335
- // Verify we generated concrete (non-generic) Solidity types
336
- assertNoLegacyGenericTypes(tsPkg, "AllSolidityTypes", "ts");
337
- assertNoLegacyGenericTypes(jsPkg, "AllSolidityTypes", "js");
338
-
339
- const env = { ...process.env, QC_RPC_URL: rpcUrl, QC_CHAIN_ID: String(chainId) };
340
-
341
- const tsRun = runNpm(["test"], tsPkg, env);
342
- assert.equal(tsRun.status, 0, `TS package tests failed:\n${tsRun.stdout}\n${tsRun.stderr}`);
343
-
344
- const jsRun = runNpm(["test"], jsPkg, env);
345
- assert.equal(jsRun.status, 0, `JS package tests failed:\n${jsRun.stdout}\n${jsRun.stderr}`);
346
-
347
- succeeded = true;
348
- } finally {
349
- // We intentionally keep the generated SDK packages under test/e2e.
350
- // Cleanup heavy folders to keep the repo light while preserving sources.
351
- if (succeeded) {
352
- for (const lang of ["ts", "js"]) {
353
- const pkgRoot = path.join(outBase, `all-solidity-types-${lang}`);
354
- fs.rmSync(path.join(pkgRoot, "node_modules"), { recursive: true, force: true });
355
- fs.rmSync(path.join(pkgRoot, "dist"), { recursive: true, force: true });
356
- }
357
- }
358
- }
359
- }, { timeout: 3_600_000 });
360
- });
361
-
1
+ /**
2
+ * @testCategory e2e
3
+ * @blockchainRequired write
4
+ * @transactional true
5
+ *
6
+ * Generates typed SDK packages (TS and JS) for AllSolidityTypes.sol,
7
+ * then injects additional tests that call all methods and validate outputs.
8
+ */
9
+
10
+ const { describe, it } = require("node:test");
11
+ const assert = require("node:assert/strict");
12
+ const fs = require("node:fs");
13
+ const path = require("node:path");
14
+ const { spawnSync } = require("node:child_process");
15
+
16
+ const { getRpcUrl, getChainId, logE2eConfig } = require("./helpers");
17
+ const { logSuite, logTest } = require("../verbose-logger");
18
+
19
+ function getNpmCmd() {
20
+ return process.platform === "win32" ? "npm.cmd" : "npm";
21
+ }
22
+
23
+ function run(cmd, args, cwd, env) {
24
+ const res = spawnSync(cmd, args, {
25
+ cwd,
26
+ env,
27
+ encoding: "utf8",
28
+ stdio: "pipe",
29
+ shell: false,
30
+ windowsHide: true,
31
+ });
32
+ if (res.error) throw res.error;
33
+ return res;
34
+ }
35
+
36
+ function _quoteIfNeeded(s) {
37
+ if (typeof s !== "string") return s;
38
+ return /[ \t"]/g.test(s) ? `"${s.replace(/"/g, '\\"')}"` : s;
39
+ }
40
+
41
+ function runNpm(args, cwd, env) {
42
+ if (process.platform === "win32") {
43
+ const cmd = `${getNpmCmd()} ${args.map(_quoteIfNeeded).join(" ")}`;
44
+ return run("cmd.exe", ["/d", "/s", "/c", cmd], cwd, env);
45
+ }
46
+ return run(getNpmCmd(), args, cwd, env);
47
+ }
48
+
49
+ function runNpx(args, cwd, env) {
50
+ if (process.platform === "win32") {
51
+ const cmd = `npx ${args.map(_quoteIfNeeded).join(" ")}`;
52
+ return run("cmd.exe", ["/d", "/s", "/c", cmd], cwd, env);
53
+ }
54
+ return run("npx", args, cwd, env);
55
+ }
56
+
57
+ function writeExtraTest(pkgRoot) {
58
+ const testDir = path.join(pkgRoot, "test", "e2e");
59
+ fs.mkdirSync(testDir, { recursive: true });
60
+
61
+ const content = `/**
62
+ * @testCategory e2e
63
+ * @blockchainRequired write
64
+ * @transactional true
65
+ * @description Extra exhaustive checks for AllSolidityTypes
66
+ */
67
+
68
+ const { describe, it } = require("node:test");
69
+ const assert = require("node:assert/strict");
70
+
71
+ const { Initialize } = require("quantumcoin/config");
72
+ const { getProvider, Wallet } = require("quantumcoin");
73
+
74
+ // Require the generated package root (works for both TS and JS packages)
75
+ const { AllSolidityTypes__factory } = require("../..");
76
+
77
+ // Hardcoded test wallet (test-only; never use for real funds)
78
+ const TEST_WALLET_ENCRYPTED_JSON =
79
+ ${JSON.stringify(
80
+ "{\"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}",
81
+ )};
82
+ const TEST_WALLET_PASSPHRASE = "QuantumCoinExample123!";
83
+
84
+ function normalizeHex(s) {
85
+ const t = String(s || "").trim();
86
+ const x = t.startsWith("0x") ? t : "0x" + t;
87
+ return "0x" + x.slice(2).toLowerCase();
88
+ }
89
+
90
+ function canon(v) {
91
+ if (v == null) return v;
92
+ if (typeof v === "bigint") return v.toString();
93
+ if (typeof v === "number") return String(v);
94
+ if (typeof v === "string") {
95
+ if (/^0x[0-9a-fA-F]*$/.test(v)) return normalizeHex(v);
96
+ if (/^-?\\d+$/.test(v.trim())) return v.trim();
97
+ return v;
98
+ }
99
+ if (Array.isArray(v)) return v.map(canon);
100
+ if (typeof v === "object") {
101
+ const keys = Object.keys(v);
102
+ const hasNonNumeric = keys.some((k) => !/^\\d+$/.test(k));
103
+ const picked = hasNonNumeric ? keys.filter((k) => !/^\\d+$/.test(k)) : keys;
104
+ picked.sort();
105
+ const out = {};
106
+ for (const k of picked) out[k] = canon(v[k]);
107
+ return out;
108
+ }
109
+ return v;
110
+ }
111
+
112
+ function hexOf(byteHex, lenBytes) {
113
+ const b = byteHex.replace(/^0x/i, "");
114
+ return normalizeHex("0x" + b.repeat(lenBytes));
115
+ }
116
+
117
+ function buildAllUints(seed) {
118
+ const s = seed || 1;
119
+ const num = (n) => Number(n);
120
+ return {
121
+ u8: num(8 + s), u16: num(16 + s), u24: num(24 + s), u32: num(32 + s), u40: num(40 + s), u48: num(48 + s), u56: num(56 + s), u64: num(64 + s),
122
+ u72: num(72 + s), u80: num(80 + s), u88: num(88 + s), u96: num(96 + s), u104: num(104 + s), u112: num(112 + s), u120: num(120 + s), u128: num(128 + s),
123
+ u136: num(136 + s), u144: num(144 + s), u152: num(152 + s), u160: num(160 + s), u168: num(168 + s), u176: num(176 + s), u184: num(184 + s), u192: num(192 + s),
124
+ u200: num(200 + s), u208: num(208 + s), u216: num(216 + s), u224: num(224 + s), u232: num(232 + s), u240: num(240 + s), u248: num(248 + s), u256: num(256 + s),
125
+ };
126
+ }
127
+
128
+ function buildAllInts(seed) {
129
+ const s = seed || 1;
130
+ const num = (n) => Number(n);
131
+ return {
132
+ i8: num(-8 - s), i16: num(-16 - s), i24: num(-24 - s), i32: num(-32 - s), i40: num(-40 - s), i48: num(-48 - s), i56: num(-56 - s), i64: num(-64 - s),
133
+ i72: num(-72 - s), i80: num(-80 - s), i88: num(-88 - s), i96: num(-96 - s), i104: num(-104 - s), i112: num(-112 - s), i120: num(-120 - s), i128: num(-128 - s),
134
+ i136: num(-136 - s), i144: num(-144 - s), i152: num(-152 - s), i160: num(-160 - s), i168: num(-168 - s), i176: num(-176 - s), i184: num(-184 - s), i192: num(-192 - s),
135
+ i200: num(-200 - s), i208: num(-208 - s), i216: num(-216 - s), i224: num(-224 - s), i232: num(-232 - s), i240: num(-240 - s), i248: num(-248 - s), i256: num(-256 - s),
136
+ };
137
+ }
138
+
139
+ function buildAllFixedBytes(seedByte) {
140
+ const b = seedByte || "11";
141
+ const o = {};
142
+ for (let i = 1; i <= 32; i++) o["b" + i] = hexOf(b, i);
143
+ return o;
144
+ }
145
+
146
+ function buildAllMisc(addr) {
147
+ return {
148
+ bo: true,
149
+ addr,
150
+ payableAddr: addr,
151
+ str: "hello",
152
+ dynBytes: normalizeHex("0x1234"),
153
+ choice: 1,
154
+ u256s: [1, 2, 3],
155
+ i256s: [-1, -2],
156
+ b32s: [hexOf("aa", 32), hexOf("bb", 32)],
157
+ addrs: [addr],
158
+ bools: [true, false, true],
159
+ strings: ["a", "b"],
160
+ bytesArr: [normalizeHex("0x00"), normalizeHex("0x12")],
161
+ fixedU16: [1, 2, 3],
162
+ fixedB32: [hexOf("cc", 32), hexOf("dd", 32)],
163
+ };
164
+ }
165
+
166
+ function buildInner(addr, seed) {
167
+ const u = buildAllUints(seed);
168
+ const i = buildAllInts(seed);
169
+ const fb = buildAllFixedBytes(seed % 2 ? "11" : "22");
170
+ const misc = buildAllMisc(addr);
171
+ const u2 = buildAllUints((seed || 1) + 10);
172
+ const fb2 = buildAllFixedBytes(seed % 2 ? "33" : "44");
173
+ return { u, i, fb, misc, uStructs: [u2], fixedFb: [fb, fb2], matrix: [[1,2],[3]] };
174
+ }
175
+
176
+ function buildOuter(addr) {
177
+ const inner1 = buildInner(addr, 1);
178
+ const inner2 = buildInner(addr, 2);
179
+ return { inner: inner1, inners: [inner2], fixedInners: [inner1, inner2], b32Matrix: [[hexOf("01",32)],[hexOf("02",32),hexOf("03",32)]] };
180
+ }
181
+
182
+ describe("AllSolidityTypes (extra)", () => {
183
+ it("roundtrips all methods", async (t) => {
184
+ logSuite("AllSolidityTypes (extra)");
185
+ logTest("roundtrips all methods", {});
186
+ const rpcUrl = getRpcUrl();
187
+ if (!rpcUrl) { t.skip("QC_RPC_URL/QC_ENDPOINT not provided"); return; }
188
+ const chainId = getChainId();
189
+ await Initialize(null);
190
+
191
+ const provider = getProvider(rpcUrl, chainId);
192
+ const wallet = Wallet.fromEncryptedJsonSync(TEST_WALLET_ENCRYPTED_JSON, TEST_WALLET_PASSPHRASE, provider);
193
+
194
+ const expectedU = buildAllUints(1);
195
+ const expectedI = buildAllInts(1);
196
+ const expectedFb = buildAllFixedBytes("11");
197
+
198
+ const ctorSeedU256s = [1,2,3];
199
+ const ctorSeedB32 = [expectedFb.b32, buildAllFixedBytes("22").b32];
200
+ const miscParam = buildAllMisc(wallet.address);
201
+
202
+ const factory = new AllSolidityTypes__factory(wallet);
203
+ const contract = await factory.deploy(
204
+ true,
205
+ wallet.address,
206
+ "hello",
207
+ "0x1234",
208
+ 1,
209
+ expectedU,
210
+ expectedI,
211
+ expectedFb,
212
+ miscParam,
213
+ ctorSeedU256s,
214
+ ctorSeedB32,
215
+ { gasLimit: 6_000_000 },
216
+ );
217
+
218
+ const tx = contract.deployTransaction();
219
+ if (tx) await tx.wait(1, 600_000);
220
+
221
+ assert.deepEqual(canon(await contract.echoAllUints(expectedU)), canon(expectedU));
222
+ assert.deepEqual(canon(await contract.echoAllInts(expectedI)), canon(expectedI));
223
+ assert.deepEqual(canon(await contract.echoAllFixedBytes(expectedFb)), canon(expectedFb));
224
+ assert.deepEqual(canon(await contract.echoAllMisc(miscParam)), canon(miscParam));
225
+
226
+ const inner = buildInner(wallet.address, 1);
227
+ assert.deepEqual(canon(await contract.echoInner(inner)), canon(inner));
228
+
229
+ const outer = buildOuter(wallet.address);
230
+ assert.deepEqual(canon(await contract.echoOuter(outer)), canon(outer));
231
+
232
+ const uArr = [buildAllUints(3), buildAllUints(4)];
233
+ assert.deepEqual(canon(await contract.echoAllUintsArray(uArr)), canon(uArr));
234
+
235
+ const innerArr = [buildInner(wallet.address, 3), buildInner(wallet.address, 4)];
236
+ assert.deepEqual(canon(await contract.echoInnerArray(innerArr)), canon(innerArr));
237
+
238
+ const matrix = [[1,2],[3,4,5]];
239
+ assert.deepEqual(canon(await contract.echoMatrix(matrix)), canon(matrix));
240
+
241
+ const multi = await contract.multiReturn(true, wallet.address, expectedFb.b32, "hello", 999, expectedU);
242
+ const multiC = canon(multi);
243
+ if (Array.isArray(multiC)) {
244
+ assert.equal(multiC[0] === true || multiC[0] === "true", true);
245
+ assert.equal(multiC[1], wallet.address);
246
+ assert.equal(multiC[2], wallet.address);
247
+ assert.equal(multiC[3], canon(expectedFb.b32));
248
+ assert.equal(multiC[4], "hello");
249
+ assert.equal(multiC[5], "999");
250
+ assert.deepEqual(multiC[6], canon(expectedU));
251
+ } else {
252
+ assert.equal(multiC.outBo === true || multiC.outBo === "true", true);
253
+ assert.equal(multiC.outAddr, wallet.address);
254
+ assert.equal(multiC.outPayableAddr, wallet.address);
255
+ assert.equal(multiC.outB32, canon(expectedFb.b32));
256
+ assert.equal(multiC.outS, "hello");
257
+ assert.equal(multiC.outX, "999");
258
+ assert.deepEqual(multiC.outU, canon(expectedU));
259
+ }
260
+ }, { timeout: 1_800_000 });
261
+ });
262
+ `;
263
+
264
+ fs.writeFileSync(path.join(testDir, "AllSolidityTypes.extra.test.js"), content, "utf8");
265
+ }
266
+
267
+ function assertNoLegacyGenericTypes(pkgRoot, contractName, lang) {
268
+ const srcDir = path.join(pkgRoot, "src");
269
+ const files =
270
+ lang === "ts"
271
+ ? [path.join(srcDir, `${contractName}.ts`)]
272
+ : [path.join(srcDir, `${contractName}.js`), path.join(srcDir, `${contractName}.d.ts`)];
273
+ for (const file of files) {
274
+ const text = fs.readFileSync(file, "utf8");
275
+ assert.equal(/SolidityInputValue\s*</.test(text), false, `${path.basename(file)} still contains SolidityInputValue<>`);
276
+ assert.equal(/SolidityOutputValue\s*</.test(text), false, `${path.basename(file)} still contains SolidityOutputValue<>`);
277
+ assert.equal(/Promise<any>/.test(text), false, `${path.basename(file)} still contains Promise<any>`);
278
+ }
279
+ }
280
+
281
+ describe("AllSolidityTypes generated SDKs (extra tests)", () => {
282
+ it("generates TS and JS packages and runs extra tests", async (t) => {
283
+ logSuite("AllSolidityTypes generated SDKs (extra tests)");
284
+ logTest("generates TS and JS packages and runs extra tests", {});
285
+ const rpcUrl = getRpcUrl();
286
+ if (!rpcUrl) {
287
+ t.skip("QC_RPC_URL not provided");
288
+ return;
289
+ }
290
+ logE2eConfig();
291
+ const chainId = getChainId();
292
+ const repoRoot = path.resolve(__dirname, "..", "..");
293
+ const solPath = path.join(repoRoot, "examples", "AllSolidityTypes.sol");
294
+ assert.ok(fs.existsSync(solPath), "missing examples/AllSolidityTypes.sol");
295
+
296
+ // Store generated SDK packages under test/e2e so they're easy to inspect.
297
+ // NOTE: This folder is .gitignored (see root .gitignore).
298
+ const outBase = path.join(repoRoot, "test", "e2e", "generated-sdks", "all-solidity-types");
299
+ fs.mkdirSync(outBase, { recursive: true });
300
+
301
+ const mkPkg = (lang) => {
302
+ const pkgName = `all-solidity-types-${lang}`;
303
+ const pkgRoot = path.join(outBase, pkgName);
304
+ // Ensure deterministic output by clearing any previous run.
305
+ fs.rmSync(pkgRoot, { recursive: true, force: true });
306
+ const genCli = path.join(repoRoot, "generate-sdk.js");
307
+ const res = run(
308
+ process.execPath,
309
+ [
310
+ genCli,
311
+ "--lang",
312
+ lang,
313
+ "--sol",
314
+ solPath,
315
+ "--name",
316
+ "AllSolidityTypes",
317
+ "--create-package",
318
+ "--package-dir",
319
+ outBase,
320
+ "--package-name",
321
+ pkgName,
322
+ "--package-description",
323
+ `${lang.toUpperCase()} typed package generated from AllSolidityTypes.sol (e2e)`,
324
+ "--package-author",
325
+ "quantumcoin.js test",
326
+ "--package-license",
327
+ "MIT",
328
+ "--package-version",
329
+ "0.0.1",
330
+ "--non-interactive",
331
+ ],
332
+ repoRoot,
333
+ process.env,
334
+ );
335
+ assert.equal(res.status, 0, `generator failed:\n${res.stdout}\n${res.stderr}`);
336
+ return pkgRoot;
337
+ };
338
+
339
+ let succeeded = false;
340
+ try {
341
+ const tsPkg = mkPkg("ts");
342
+ const jsPkg = mkPkg("js");
343
+
344
+ // Inject extra tests
345
+ writeExtraTest(tsPkg);
346
+ writeExtraTest(jsPkg);
347
+
348
+ // Verify we generated concrete (non-generic) Solidity types
349
+ assertNoLegacyGenericTypes(tsPkg, "AllSolidityTypes", "ts");
350
+ assertNoLegacyGenericTypes(jsPkg, "AllSolidityTypes", "js");
351
+
352
+ const env = { ...process.env, QC_RPC_URL: rpcUrl, QC_CHAIN_ID: String(chainId) };
353
+
354
+ const tsRun = runNpm(["test"], tsPkg, env);
355
+ assert.equal(tsRun.status, 0, `TS package tests failed:\n${tsRun.stdout}\n${tsRun.stderr}`);
356
+
357
+ const jsRun = runNpm(["test"], jsPkg, env);
358
+ assert.equal(jsRun.status, 0, `JS package tests failed:\n${jsRun.stdout}\n${jsRun.stderr}`);
359
+
360
+ // Run the generated offline signing example (offline deploy via sendRawTransaction) - both JS and TS.
361
+ const tsExampleJs = run(process.execPath, [path.join(tsPkg, "examples", "offline-signing.js")], tsPkg, env);
362
+ assert.equal(tsExampleJs.status, 0, `TS package offline-signing.js failed:\n${tsExampleJs.stdout}\n${tsExampleJs.stderr}`);
363
+
364
+ const tsExampleTs = runNpx(["tsx", path.join("examples", "offline-signing.ts")], tsPkg, env);
365
+ assert.equal(tsExampleTs.status, 0, `TS package offline-signing.ts failed:\n${tsExampleTs.stdout}\n${tsExampleTs.stderr}`);
366
+
367
+ const jsExampleJs = run(process.execPath, [path.join(jsPkg, "examples", "offline-signing.js")], jsPkg, env);
368
+ assert.equal(jsExampleJs.status, 0, `JS package offline-signing.js failed:\n${jsExampleJs.stdout}\n${jsExampleJs.stderr}`);
369
+
370
+ const jsExampleTs = runNpx(["tsx", path.join("examples", "offline-signing.ts")], jsPkg, env);
371
+ assert.equal(jsExampleTs.status, 0, `JS package offline-signing.ts failed:\n${jsExampleTs.stdout}\n${jsExampleTs.stderr}`);
372
+
373
+ succeeded = true;
374
+ } finally {
375
+ // We intentionally keep the generated SDK packages under test/e2e.
376
+ // Cleanup heavy folders to keep the repo light while preserving sources.
377
+ if (succeeded) {
378
+ for (const lang of ["ts", "js"]) {
379
+ const pkgRoot = path.join(outBase, `all-solidity-types-${lang}`);
380
+ fs.rmSync(path.join(pkgRoot, "node_modules"), { recursive: true, force: true });
381
+ fs.rmSync(path.join(pkgRoot, "dist"), { recursive: true, force: true });
382
+ }
383
+ }
384
+ }
385
+ }, { timeout: 3_600_000 });
386
+ });
387
+