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,61 +1,72 @@
1
- /**
2
- * QuantumCoin.js Examples (non-transactional).
3
- *
4
- * Run:
5
- * - `node examples/example.js`
6
- *
7
- * These examples only perform:
8
- * - SDK initialization
9
- * - Read-only JSON-RPC queries
10
- * - Offline wallet operations (sign/encrypt)
11
- * - Read-only contract call against the system staking contract
12
- */
13
-
14
- const { Initialize } = require("../config");
15
- const qc = require("..");
16
-
17
- const RPC = "https://public.rpc.quantumcoinapi.com";
18
- const CHAIN_ID = 123123;
19
- const STAKING_CONTRACT = "0x" + "00".repeat(30) + "10" + "00"; // 0x...1000
20
-
21
- async function main() {
22
- const ok = await Initialize(null);
23
- if (!ok) throw new Error("Initialize failed");
24
-
25
- console.log("SDK initialized:", qc.isInitialized());
26
-
27
- // Provider: basic read-only calls
28
- const provider = new qc.JsonRpcProvider(RPC, CHAIN_ID);
29
- const blockNumber = await provider.getBlockNumber();
30
- console.log("Latest block:", blockNumber);
31
-
32
- const block = await provider.getBlock("latest");
33
- console.log("Latest block hash:", block.hash);
34
-
35
- const balance = await provider.getBalance(STAKING_CONTRACT);
36
- console.log("Staking contract balance (wei):", balance.toString());
37
-
38
- // Wallet: offline operations
39
- const wallet = qc.Wallet.createRandom();
40
- console.log("Random wallet address:", wallet.address);
41
- const sig = wallet.signMessageSync("Hello, QuantumCoin!");
42
- console.log("Signed message signature:", sig.slice(0, 18) + "...");
43
- const recovered = qc.verifyMessage("Hello, QuantumCoin!", sig);
44
- console.log("Recovered address:", recovered);
45
-
46
- const encrypted = wallet.encryptSync("mySecurePassword123");
47
- console.log("Encrypted wallet JSON length:", encrypted.length);
48
-
49
- // Contract: read-only call
50
- const abi = require("../test/fixtures/StakingContract.abi.json");
51
- const staking = new qc.Contract(STAKING_CONTRACT, abi, provider);
52
- const depositorCount = await staking.getDepositorCount();
53
- console.log("typeof depositorCount:", typeof depositorCount);
54
- console.log("Depositor count:", depositorCount.toString());
55
- }
56
-
57
- main().catch((e) => {
58
- console.error(e);
59
- process.exitCode = 1;
60
- });
61
-
1
+ /**
2
+ * QuantumCoin.js Examples (non-transactional).
3
+ *
4
+ * Run:
5
+ * - `node examples/example.js`
6
+ * - `VERBOSE=1 node examples/example.js` for verbose logging (addresses, block hashes, etc.)
7
+ *
8
+ * These examples only perform:
9
+ * - SDK initialization
10
+ * - Read-only JSON-RPC queries
11
+ * - Offline wallet operations (sign/encrypt)
12
+ * - Read-only contract call against the system staking contract
13
+ */
14
+
15
+ const { Initialize } = require("../config");
16
+ const qc = require("..");
17
+ const { logExample, logAddress } = require("../test/verbose-logger");
18
+
19
+ const RPC = "https://public.rpc.quantumcoinapi.com";
20
+ const CHAIN_ID = 123123;
21
+ const STAKING_CONTRACT = "0x" + "00".repeat(30) + "10" + "00"; // 0x...1000
22
+
23
+ async function main() {
24
+ logExample("example.js", "starting", { rpc: RPC, chainId: CHAIN_ID });
25
+
26
+ const ok = await Initialize(null);
27
+ if (!ok) throw new Error("Initialize failed");
28
+
29
+ console.log("SDK initialized:", qc.isInitialized());
30
+
31
+ // Provider: basic read-only calls (getProvider detects http/ws/ipc from endpoint)
32
+ const provider = qc.getProvider(RPC, CHAIN_ID);
33
+ const blockNumber = await provider.getBlockNumber();
34
+ console.log("Latest block:", blockNumber);
35
+ logExample("example.js", "getBlockNumber", { blockNumber });
36
+
37
+ const block = await provider.getBlock("latest");
38
+ console.log("Latest block hash:", block.hash);
39
+ logExample("example.js", "getBlock(latest)", { blockNumber: block.number, blockHash: block.hash });
40
+
41
+ const balance = await provider.getBalance(STAKING_CONTRACT);
42
+ console.log("Staking contract balance (wei):", balance.toString());
43
+ logAddress("staking_contract", STAKING_CONTRACT);
44
+ logExample("example.js", "getBalance", { balance: balance.toString() });
45
+
46
+ // Wallet: offline operations
47
+ const wallet = qc.Wallet.createRandom();
48
+ console.log("Random wallet address:", wallet.address);
49
+ logAddress("random_wallet", wallet.address);
50
+ const sig = wallet.signMessageSync("Hello, QuantumCoin!");
51
+ console.log("Signed message signature:", sig.slice(0, 18) + "...");
52
+ const recovered = qc.verifyMessage("Hello, QuantumCoin!", sig);
53
+ console.log("Recovered address:", recovered);
54
+ logAddress("recovered", recovered);
55
+
56
+ const encrypted = wallet.encryptSync("mySecurePassword123");
57
+ console.log("Encrypted wallet JSON length:", encrypted.length);
58
+
59
+ // Contract: read-only call
60
+ const abi = require("../test/fixtures/StakingContract.abi.json");
61
+ const staking = new qc.Contract(STAKING_CONTRACT, abi, provider);
62
+ const depositorCount = await staking.getDepositorCount();
63
+ console.log("typeof depositorCount:", typeof depositorCount);
64
+ console.log("Depositor count:", depositorCount.toString());
65
+ logExample("example.js", "getDepositorCount", { depositorCount: depositorCount.toString() });
66
+ }
67
+
68
+ main().catch((e) => {
69
+ console.error(e);
70
+ process.exitCode = 1;
71
+ });
72
+
@@ -0,0 +1,61 @@
1
+ /**
2
+ * QuantumCoin.js Examples (non-transactional) - TypeScript.
3
+ *
4
+ * Run: npx tsx examples/example.ts
5
+ * VERBOSE=1 npx tsx examples/example.ts for verbose logging.
6
+ */
7
+
8
+ import { Initialize } from "../config";
9
+ import qc from "..";
10
+ import { logExample, logAddress } from "../test/verbose-logger";
11
+
12
+ const RPC = "https://public.rpc.quantumcoinapi.com";
13
+ const CHAIN_ID = 123123;
14
+ const STAKING_CONTRACT = "0x" + "00".repeat(30) + "10" + "00";
15
+
16
+ async function main(): Promise<void> {
17
+ logExample("example.ts", "starting", { rpc: RPC, chainId: CHAIN_ID });
18
+
19
+ const ok = await Initialize(null);
20
+ if (!ok) throw new Error("Initialize failed");
21
+
22
+ console.log("SDK initialized:", qc.isInitialized());
23
+
24
+ const provider = qc.getProvider(RPC, CHAIN_ID);
25
+ const blockNumber = await provider.getBlockNumber();
26
+ console.log("Latest block:", blockNumber);
27
+ logExample("example.ts", "getBlockNumber", { blockNumber });
28
+
29
+ const block = await provider.getBlock("latest");
30
+ console.log("Latest block hash:", block.hash);
31
+ logExample("example.ts", "getBlock(latest)", { blockNumber: block.number, blockHash: block.hash });
32
+
33
+ const balance = await provider.getBalance(STAKING_CONTRACT);
34
+ console.log("Staking contract balance (wei):", balance.toString());
35
+ logAddress("staking_contract", STAKING_CONTRACT);
36
+ logExample("example.ts", "getBalance", { balance: balance.toString() });
37
+
38
+ const wallet = qc.Wallet.createRandom();
39
+ console.log("Random wallet address:", wallet.address);
40
+ logAddress("random_wallet", wallet.address);
41
+ const sig = wallet.signMessageSync("Hello, QuantumCoin!");
42
+ console.log("Signed message signature:", sig.slice(0, 18) + "...");
43
+ const recovered = qc.verifyMessage("Hello, QuantumCoin!", sig);
44
+ console.log("Recovered address:", recovered);
45
+ logAddress("recovered", recovered);
46
+
47
+ const encrypted = wallet.encryptSync("mySecurePassword123");
48
+ console.log("Encrypted wallet JSON length:", encrypted.length);
49
+
50
+ const abi = require("../test/fixtures/StakingContract.abi.json");
51
+ const staking = new qc.Contract(STAKING_CONTRACT, abi, provider);
52
+ const depositorCount = await staking.getDepositorCount();
53
+ console.log("typeof depositorCount:", typeof depositorCount);
54
+ console.log("Depositor count:", depositorCount.toString());
55
+ logExample("example.ts", "getDepositorCount", { depositorCount: depositorCount.toString() });
56
+ }
57
+
58
+ main().catch((e) => {
59
+ console.error(e);
60
+ process.exitCode = 1;
61
+ });
@@ -0,0 +1,79 @@
1
+ /**
2
+ * Example: Offline signing with populateTransaction + sendRawTransaction.
3
+ * Run with VERBOSE=1 for wallet address, tx hash, receipt block/status.
4
+ *
5
+ * This demonstrates how to:
6
+ * - build unsigned tx data via `contract.populateTransaction.<method>()`
7
+ * - sign it via `wallet.signTransaction(txReq)`
8
+ * - broadcast via `provider.sendRawTransaction(rawTx)`
9
+ *
10
+ * NOTE:
11
+ * - This example WILL broadcast a real transaction and change chain state.
12
+ * - It uses a HARDCODED test wallet (funded) for convenience. Never use for real funds.
13
+ * - Requires `QC_RPC_URL` env var.
14
+ */
15
+
16
+ const { Initialize } = require("../config");
17
+ const qc = require("..");
18
+ const { logExample, logAddress, logTxn } = require("../test/verbose-logger");
19
+
20
+ // Hardcoded test wallet (test-only; never use for real funds)
21
+ const TEST_WALLET_ENCRYPTED_JSON =
22
+ "{\"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}";
23
+ const TEST_WALLET_PASSPHRASE = "QuantumCoinExample123!";
24
+
25
+ async function main() {
26
+ const rpcUrl = process.env.QC_RPC_URL;
27
+ if (!rpcUrl) throw new Error("QC_RPC_URL is required");
28
+ const chainId = process.env.QC_CHAIN_ID ? Number(process.env.QC_CHAIN_ID) : 123123;
29
+ logExample("offline-signing.js", "starting", { chainId });
30
+
31
+ await Initialize(null);
32
+
33
+ const provider = qc.getProvider(rpcUrl, chainId);
34
+ const offlineWallet = qc.Wallet.fromEncryptedJsonSync(TEST_WALLET_ENCRYPTED_JSON, TEST_WALLET_PASSPHRASE);
35
+ logAddress("wallet", offlineWallet.address);
36
+
37
+ // Minimal ABI for demonstration
38
+ const abi = [
39
+ {
40
+ type: "function",
41
+ name: "setValue",
42
+ stateMutability: "nonpayable",
43
+ inputs: [{ name: "value", type: "uint256" }],
44
+ outputs: [],
45
+ },
46
+ ];
47
+
48
+ const dummyContractAddress = "0x" + "11".repeat(32);
49
+ const contract = new qc.Contract(dummyContractAddress, abi);
50
+
51
+ const txReq = await contract.populateTransaction.setValue(123, { gasLimit: 200000 });
52
+
53
+ // True offline signing: explicitly set nonce + chainId (+ gasPrice if desired)
54
+ const nonce = await provider.getTransactionCount(offlineWallet.address, "pending");
55
+ const rawTx = await offlineWallet.signTransaction({
56
+ ...txReq,
57
+ nonce,
58
+ chainId,
59
+ gasPrice: 1n,
60
+ });
61
+
62
+ console.log("Unsigned txReq:", txReq);
63
+ console.log("Signed rawTx:", rawTx);
64
+ console.log("Submitting via sendRawTransaction...");
65
+
66
+ const sent = await provider.sendRawTransaction(rawTx);
67
+ console.log("tx hash:", sent.hash);
68
+ logTxn(sent.hash, { from: offlineWallet.address });
69
+ const receipt = await sent.wait(1, 600_000);
70
+ console.log("receipt block:", receipt.blockNumber);
71
+ console.log("receipt status:", receipt.status);
72
+ logTxn(sent.hash, { blockNumber: receipt.blockNumber, status: receipt.status });
73
+ }
74
+
75
+ main().catch((e) => {
76
+ console.error(e);
77
+ process.exitCode = 1;
78
+ });
79
+
@@ -0,0 +1,66 @@
1
+ /**
2
+ * Example: Offline signing with populateTransaction + sendRawTransaction - TypeScript.
3
+ * Run: npx tsx examples/offline-signing.ts
4
+ * Requires QC_RPC_URL. Uses hardcoded test wallet (never use for real funds).
5
+ */
6
+
7
+ import { Initialize } from "../config";
8
+ import qc from "..";
9
+ import { logExample, logAddress, logTxn } from "../test/verbose-logger";
10
+
11
+ const TEST_WALLET_ENCRYPTED_JSON =
12
+ "{\"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}";
13
+ const TEST_WALLET_PASSPHRASE = "QuantumCoinExample123!";
14
+
15
+ async function main(): Promise<void> {
16
+ const rpcUrl = process.env.QC_RPC_URL;
17
+ if (!rpcUrl) throw new Error("QC_RPC_URL is required");
18
+ const chainId = process.env.QC_CHAIN_ID ? Number(process.env.QC_CHAIN_ID) : 123123;
19
+ logExample("offline-signing.ts", "starting", { chainId });
20
+
21
+ await Initialize(null);
22
+
23
+ const provider = qc.getProvider(rpcUrl, chainId);
24
+ const offlineWallet = qc.Wallet.fromEncryptedJsonSync(TEST_WALLET_ENCRYPTED_JSON, TEST_WALLET_PASSPHRASE);
25
+ logAddress("wallet", offlineWallet.address);
26
+
27
+ const abi = [
28
+ {
29
+ type: "function",
30
+ name: "setValue",
31
+ stateMutability: "nonpayable",
32
+ inputs: [{ name: "value", type: "uint256" }],
33
+ outputs: [],
34
+ },
35
+ ];
36
+
37
+ const dummyContractAddress = "0x" + "11".repeat(32);
38
+ const contract = new qc.Contract(dummyContractAddress, abi);
39
+
40
+ const txReq = await contract.populateTransaction.setValue(123, { gasLimit: 200000 });
41
+
42
+ const nonce = await provider.getTransactionCount(offlineWallet.address, "pending");
43
+ const rawTx = await offlineWallet.signTransaction({
44
+ ...txReq,
45
+ nonce,
46
+ chainId,
47
+ gasPrice: 1n,
48
+ });
49
+
50
+ console.log("Unsigned txReq:", txReq);
51
+ console.log("Signed rawTx:", rawTx);
52
+ console.log("Submitting via sendRawTransaction...");
53
+
54
+ const sent = await provider.sendRawTransaction(rawTx);
55
+ console.log("tx hash:", sent.hash);
56
+ logTxn(sent.hash, { from: offlineWallet.address });
57
+ const receipt = await sent.wait(1, 600_000);
58
+ console.log("receipt block:", receipt.blockNumber);
59
+ console.log("receipt status:", receipt.status);
60
+ logTxn(sent.hash, { blockNumber: receipt.blockNumber, status: receipt.status });
61
+ }
62
+
63
+ main().catch((e) => {
64
+ console.error(e);
65
+ process.exitCode = 1;
66
+ });
@@ -1,57 +1,48 @@
1
- {
2
- "name": "quantumcoinjs-sdk-examples",
3
- "version": "1.0.0",
4
- "lockfileVersion": 3,
5
- "requires": true,
6
- "packages": {
7
- "": {
8
- "name": "quantumcoinjs-sdk-examples",
9
- "version": "1.0.0",
10
- "license": "MIT",
11
- "dependencies": {
12
- "quantum-coin-js-sdk": "^1.0.28",
13
- "quantum-coin-pqc-js-sdk": "^1.0.5",
14
- "quantumcoin": "file:.."
15
- }
16
- },
17
- "..": {
18
- "name": "quantumcoin",
19
- "version": "7.0.2",
20
- "license": "MIT",
21
- "dependencies": {
22
- "quantum-coin-js-sdk": "^1.0.28",
23
- "quantum-coin-pqc-js-sdk": "^1.0.5",
24
- "seed-words": "^1.0.2"
25
- },
26
- "bin": {
27
- "quantumcoin-sdk-generator": "generate-sdk.js"
28
- }
29
- },
30
- "node_modules/quantum-coin-js-sdk": {
31
- "version": "1.0.28",
32
- "resolved": "https://registry.npmjs.org/quantum-coin-js-sdk/-/quantum-coin-js-sdk-1.0.28.tgz",
33
- "integrity": "sha512-ok3kCoEecSWRLsFw285IDzz8Xvtc772BvBujeCasQ7G2f+Jwly6UFtjivyrh0gDPuZFwYU/hbaKvpb8OtkuM/Q==",
34
- "license": "MIT",
35
- "dependencies": {
36
- "quantum-coin-pqc-js-sdk": "1.0.5",
37
- "seed-words": "1.0.2"
38
- }
39
- },
40
- "node_modules/quantum-coin-pqc-js-sdk": {
41
- "version": "1.0.5",
42
- "resolved": "https://registry.npmjs.org/quantum-coin-pqc-js-sdk/-/quantum-coin-pqc-js-sdk-1.0.5.tgz",
43
- "integrity": "sha512-9P1YDkca5CqrO7++TnGJZzPvYlBjYARopHgxYbDcEHgPicP7+tA8zD+DE1BooHp7sLEWF9ObklMj6AdaDUjDVw==",
44
- "license": "MIT"
45
- },
46
- "node_modules/quantumcoin": {
47
- "resolved": "..",
48
- "link": true
49
- },
50
- "node_modules/seed-words": {
51
- "version": "1.0.2",
52
- "resolved": "https://registry.npmjs.org/seed-words/-/seed-words-1.0.2.tgz",
53
- "integrity": "sha512-ia58deuPjcR8DpJ8uIgZ7gqDnIWD8vnjb4jX/sEsIDcuQaH/AJj9J8L3DXkUHfUGqAq9Y6pVMuG90t3XUJH90g==",
54
- "license": "MIT"
55
- }
56
- }
57
- }
1
+ {
2
+ "name": "quantumcoinjs-sdk-examples",
3
+ "version": "1.0.0",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "quantumcoinjs-sdk-examples",
9
+ "version": "1.0.0",
10
+ "license": "MIT",
11
+ "dependencies": {
12
+ "quantum-coin-js-sdk": "1.0.31",
13
+ "quantumcoin": "file:.."
14
+ }
15
+ },
16
+ "..": {
17
+ "name": "quantumcoin",
18
+ "version": "7.0.4",
19
+ "license": "MIT",
20
+ "dependencies": {
21
+ "quantum-coin-js-sdk": "1.0.31",
22
+ "seed-words": "^1.0.2"
23
+ },
24
+ "bin": {
25
+ "sdkgen": "generate-sdk.js"
26
+ }
27
+ },
28
+ "node_modules/quantum-coin-js-sdk": {
29
+ "version": "1.0.31",
30
+ "resolved": "https://registry.npmjs.org/quantum-coin-js-sdk/-/quantum-coin-js-sdk-1.0.31.tgz",
31
+ "integrity": "sha512-enXV7IrUSi8Rjxd4xkoTMmQ9C49ev75FPzu+UbJuzo7D4og/FpD7Ycu1xkLau9cQl/YJFzN9MV+F0/GIoPEG4Q==",
32
+ "license": "MIT",
33
+ "dependencies": {
34
+ "seed-words": "1.0.2"
35
+ }
36
+ },
37
+ "node_modules/quantumcoin": {
38
+ "resolved": "..",
39
+ "link": true
40
+ },
41
+ "node_modules/seed-words": {
42
+ "version": "1.0.2",
43
+ "resolved": "https://registry.npmjs.org/seed-words/-/seed-words-1.0.2.tgz",
44
+ "integrity": "sha512-ia58deuPjcR8DpJ8uIgZ7gqDnIWD8vnjb4jX/sEsIDcuQaH/AJj9J8L3DXkUHfUGqAq9Y6pVMuG90t3XUJH90g==",
45
+ "license": "MIT"
46
+ }
47
+ }
48
+ }
@@ -1,16 +1,32 @@
1
- {
2
- "name": "quantumcoinjs-sdk-examples",
3
- "version": "1.0.0",
4
- "description": "QuantumCoin.js SDK Examples",
5
- "main": "example.js",
6
- "scripts": {
7
- "test": "echo \"Error: no test specified\" && exit 1"
8
- },
9
- "author": "QuantumCoin Community",
10
- "license": "MIT",
11
- "dependencies": {
12
- "quantum-coin-js-sdk": "^1.0.28",
13
- "quantum-coin-pqc-js-sdk": "^1.0.5",
14
- "quantumcoin": "file:.."
15
- }
16
- }
1
+ {
2
+ "name": "quantumcoinjs-sdk-examples",
3
+ "version": "1.0.0",
4
+ "description": "QuantumCoin.js SDK Examples",
5
+ "main": "example.js",
6
+ "scripts": {
7
+ "test": "echo \"Error: no test specified\" && exit 1",
8
+ "example": "node example.js",
9
+ "example:ts": "npx tsx example.ts",
10
+ "example:wallet": "node wallet-offline.js",
11
+ "example:wallet:ts": "npx tsx wallet-offline.ts",
12
+ "example:read": "node read-operations.js",
13
+ "example:read:ts": "npx tsx read-operations.ts",
14
+ "example:events": "node events.js",
15
+ "example:events:ts": "npx tsx events.ts",
16
+ "example:offline-signing": "node offline-signing.js",
17
+ "example:offline-signing:ts": "npx tsx offline-signing.ts",
18
+ "example:generator-js": "node example-generator-sdk-js.js",
19
+ "example:generator-js:ts": "npx tsx example-generator-sdk-js.ts",
20
+ "example:generator-ts": "node example-generator-sdk-ts.js",
21
+ "example:generator-ts:ts": "npx tsx example-generator-sdk-ts.ts"
22
+ },
23
+ "author": "QuantumCoin Community",
24
+ "license": "MIT",
25
+ "devDependencies": {
26
+ "tsx": "^4.19.0"
27
+ },
28
+ "dependencies": {
29
+ "quantum-coin-js-sdk": "1.0.31",
30
+ "quantumcoin": "file:.."
31
+ }
32
+ }
@@ -1,27 +1,32 @@
1
- /**
2
- * Example: Read-only operations using JsonRpcProvider + Contract.
3
- */
4
-
5
- const { Initialize } = require("../config");
6
- const { JsonRpcProvider, Contract } = require("..");
7
-
8
- const RPC = "https://public.rpc.quantumcoinapi.com";
9
- const CHAIN_ID = 123123;
10
- const STAKING_CONTRACT = "0x" + "00".repeat(30) + "10" + "00"; // 0x...1000
11
-
12
- async function readOperations() {
13
- await Initialize(null);
14
-
15
- const provider = new JsonRpcProvider(RPC, CHAIN_ID);
16
- const abi = require("../test/fixtures/StakingContract.abi.json");
17
- const contract = new Contract(STAKING_CONTRACT, abi, provider);
18
-
19
- const total = await contract.getTotalDepositedBalance();
20
- console.log("Total deposited balance:", total.toString());
21
- }
22
-
23
- readOperations().catch((e) => {
24
- console.error(e);
25
- process.exitCode = 1;
26
- });
27
-
1
+ /**
2
+ * Example: Read-only operations using JsonRpcProvider + Contract.
3
+ * Run with VERBOSE=1 for addresses and operation details.
4
+ */
5
+
6
+ const { Initialize } = require("../config");
7
+ const { getProvider, Contract } = require("..");
8
+ const { logExample, logAddress } = require("../test/verbose-logger");
9
+
10
+ const RPC = "https://public.rpc.quantumcoinapi.com";
11
+ const CHAIN_ID = 123123;
12
+ const STAKING_CONTRACT = "0x" + "00".repeat(30) + "10" + "00"; // 0x...1000
13
+
14
+ async function readOperations() {
15
+ logExample("read-operations.js", "starting", { rpc: RPC, chainId: CHAIN_ID });
16
+ await Initialize(null);
17
+
18
+ const provider = getProvider(RPC, CHAIN_ID);
19
+ logAddress("staking_contract", STAKING_CONTRACT);
20
+ const abi = require("../test/fixtures/StakingContract.abi.json");
21
+ const contract = new Contract(STAKING_CONTRACT, abi, provider);
22
+
23
+ const total = await contract.getTotalDepositedBalance();
24
+ console.log("Total deposited balance:", total.toString());
25
+ logExample("read-operations.js", "getTotalDepositedBalance", { total: total.toString() });
26
+ }
27
+
28
+ readOperations().catch((e) => {
29
+ console.error(e);
30
+ process.exitCode = 1;
31
+ });
32
+
@@ -0,0 +1,31 @@
1
+ /**
2
+ * Example: Read-only operations (JsonRpcProvider + Contract) - TypeScript.
3
+ * Run: npx tsx examples/read-operations.ts
4
+ */
5
+
6
+ import { Initialize } from "../config";
7
+ import { getProvider, Contract } from "..";
8
+ import { logExample, logAddress } from "../test/verbose-logger";
9
+
10
+ const RPC = "https://public.rpc.quantumcoinapi.com";
11
+ const CHAIN_ID = 123123;
12
+ const STAKING_CONTRACT = "0x" + "00".repeat(30) + "10" + "00";
13
+
14
+ async function readOperations(): Promise<void> {
15
+ logExample("read-operations.ts", "starting", { rpc: RPC, chainId: CHAIN_ID });
16
+ await Initialize(null);
17
+
18
+ const provider = getProvider(RPC, CHAIN_ID);
19
+ logAddress("staking_contract", STAKING_CONTRACT);
20
+ const abi = require("../test/fixtures/StakingContract.abi.json");
21
+ const contract = new Contract(STAKING_CONTRACT, abi, provider);
22
+
23
+ const total = await contract.getTotalDepositedBalance();
24
+ console.log("Total deposited balance:", total.toString());
25
+ logExample("read-operations.ts", "getTotalDepositedBalance", { total: total.toString() });
26
+ }
27
+
28
+ readOperations().catch((e) => {
29
+ console.error(e);
30
+ process.exitCode = 1;
31
+ });