quantumcoin 7.0.3 → 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.
- package/.github/workflows/publish-npmjs.yaml +22 -22
- package/.gitignore +15 -15
- package/LICENSE +21 -21
- package/README-SDK.md +756 -754
- package/README.md +165 -150
- package/SPEC.md +3845 -3843
- package/config.d.ts +50 -50
- package/config.js +115 -115
- package/examples/AllSolidityTypes.sol +184 -184
- package/examples/SimpleIERC20.sol +74 -74
- package/examples/events.js +41 -35
- package/examples/events.ts +35 -0
- package/examples/example-generator-sdk-js.js +100 -95
- package/examples/example-generator-sdk-js.ts +77 -0
- package/examples/example-generator-sdk-ts.js +100 -95
- package/examples/example-generator-sdk-ts.ts +77 -0
- package/examples/example.js +72 -61
- package/examples/example.ts +61 -0
- package/examples/offline-signing.js +79 -73
- package/examples/offline-signing.ts +66 -0
- package/examples/package-lock.json +48 -57
- package/examples/package.json +32 -16
- package/examples/read-operations.js +32 -27
- package/examples/read-operations.ts +31 -0
- package/examples/sdk-generator-erc20.inline.json +251 -251
- package/examples/solidity-types.ts +43 -43
- package/examples/wallet-offline.js +35 -29
- package/examples/wallet-offline.ts +34 -0
- package/generate-sdk.js +1824 -1490
- package/index.js +12 -12
- package/package.json +95 -75
- package/scripts/copy-declarations.js +31 -0
- package/scripts/run-all-one-by-one.js +151 -0
- package/src/abi/fragments.d.ts +42 -42
- package/src/abi/fragments.js +63 -63
- package/src/abi/index.d.ts +13 -13
- package/src/abi/index.js +9 -9
- package/src/abi/interface.d.ts +128 -132
- package/src/abi/interface.js +590 -590
- package/src/abi/js-abi-coder.d.ts +8 -0
- package/src/abi/js-abi-coder.js +474 -474
- package/src/constants.d.ts +66 -61
- package/src/constants.js +101 -94
- package/src/contract/contract-factory.d.ts +28 -28
- package/src/contract/contract-factory.js +105 -105
- package/src/contract/contract.d.ts +113 -114
- package/src/contract/contract.js +354 -354
- package/src/contract/index.d.ts +9 -9
- package/src/contract/index.js +9 -9
- package/src/errors/index.d.ts +92 -92
- package/src/errors/index.js +188 -188
- package/src/generator/index.d.ts +74 -0
- package/src/generator/index.js +1404 -1404
- package/src/index.d.ts +125 -127
- package/src/index.js +41 -41
- package/src/internal/hex.d.ts +61 -61
- package/src/internal/hex.js +144 -144
- package/src/providers/extra-providers.d.ts +139 -128
- package/src/providers/extra-providers.js +600 -575
- package/src/providers/index.d.ts +17 -16
- package/src/providers/index.js +10 -10
- package/src/providers/json-rpc-provider.d.ts +12 -12
- package/src/providers/json-rpc-provider.js +79 -79
- package/src/providers/provider.d.ts +207 -203
- package/src/providers/provider.js +392 -371
- package/src/types/index.d.ts +214 -462
- package/src/types/index.js +9 -9
- package/src/utils/address.d.ts +72 -72
- package/src/utils/address.js +181 -182
- package/src/utils/encoding.d.ts +120 -120
- package/src/utils/encoding.js +306 -306
- package/src/utils/hashing.d.ts +82 -76
- package/src/utils/hashing.js +313 -298
- package/src/utils/index.d.ts +65 -55
- package/src/utils/index.js +13 -13
- package/src/utils/result.d.ts +57 -57
- package/src/utils/result.js +128 -128
- package/src/utils/rlp.d.ts +12 -12
- package/src/utils/rlp.js +200 -200
- package/src/utils/units.d.ts +29 -29
- package/src/utils/units.js +107 -107
- package/src/wallet/index.d.ts +10 -10
- package/src/wallet/index.js +8 -8
- package/src/wallet/wallet.d.ts +160 -160
- package/src/wallet/wallet.js +483 -489
- package/test/e2e/all-solidity-types.dynamic.test.js +207 -200
- package/test/e2e/all-solidity-types.dynamic.test.ts +191 -0
- package/test/e2e/all-solidity-types.fixtures.js +231 -231
- package/test/e2e/all-solidity-types.generated-sdks.e2e.test.js +387 -368
- package/test/e2e/all-solidity-types.generated-sdks.e2e.test.ts +350 -0
- package/test/e2e/helpers.js +59 -47
- package/test/e2e/signing-context-and-fee.e2e.test.js +137 -0
- package/test/e2e/signing-context-and-fee.e2e.test.ts +128 -0
- package/test/e2e/simple-erc20.generated-sdks.e2e.test.js +168 -151
- package/test/e2e/simple-erc20.generated-sdks.e2e.test.ts +141 -0
- package/test/e2e/transactional.test.js +245 -191
- package/test/e2e/transactional.test.ts +208 -0
- package/test/e2e/typed-generator.e2e.test.js +407 -404
- package/test/e2e/typed-generator.e2e.test.ts +337 -0
- package/test/fixtures/ConstructorParam.sol +23 -23
- package/test/fixtures/MultiContracts.sol +37 -37
- package/test/fixtures/SimpleStorage.sol +18 -18
- package/test/fixtures/StakingContract.abi.json +1 -1
- package/test/integration/ipc-provider.test.js +49 -44
- package/test/integration/ipc-provider.test.ts +44 -0
- package/test/integration/provider.test.js +88 -72
- package/test/integration/provider.test.ts +85 -0
- package/test/integration/ws-provider.test.js +41 -33
- package/test/integration/ws-provider.test.ts +38 -0
- package/test/security/malformed-input.test.js +37 -31
- package/test/security/malformed-input.test.ts +35 -0
- package/test/unit/_encrypted-output.txt +6 -0
- package/test/unit/_log-encrypted-jsons.js +45 -0
- package/test/unit/_write-keystore-fixture.js +16 -0
- package/test/unit/abi-interface.test.js +103 -98
- package/test/unit/abi-interface.test.ts +102 -0
- package/test/unit/address-wallet.test.js +355 -257
- package/test/unit/address-wallet.test.ts +342 -0
- package/test/unit/browser-provider.test.js +85 -82
- package/test/unit/browser-provider.test.ts +79 -0
- package/test/unit/contract.test.js +85 -82
- package/test/unit/contract.test.ts +83 -0
- package/test/unit/encoding-units-rlp.test.js +92 -89
- package/test/unit/encoding-units-rlp.test.ts +91 -0
- package/test/unit/errors.test.js +77 -74
- package/test/unit/errors.test.ts +76 -0
- package/test/unit/filter-by-blockhash.test.js +55 -52
- package/test/unit/filter-by-blockhash.test.ts +54 -0
- package/test/unit/fixtures/encrypted-keystores-48-32-36.js +9 -0
- package/test/unit/generate-contract-cli.test.js +42 -39
- package/test/unit/generate-contract-cli.test.ts +41 -0
- package/test/unit/generate-sdk-artifacts-json.test.js +113 -110
- package/test/unit/generate-sdk-artifacts-json.test.ts +110 -0
- package/test/unit/generator.test.js +102 -99
- package/test/unit/generator.test.ts +101 -0
- package/test/unit/hashing.test.js +68 -54
- package/test/unit/hashing.test.ts +67 -0
- package/test/unit/init.test.js +39 -36
- package/test/unit/init.test.ts +38 -0
- package/test/unit/interface.test.js +56 -53
- package/test/unit/interface.test.ts +54 -0
- package/test/unit/internal-hex.test.js +50 -47
- package/test/unit/internal-hex.test.ts +49 -0
- package/test/unit/populate-transaction.test.js +65 -62
- package/test/unit/populate-transaction.test.ts +64 -0
- package/test/unit/providers.test.js +200 -144
- package/test/unit/providers.test.ts +196 -0
- package/test/unit/result.test.js +80 -77
- package/test/unit/result.test.ts +79 -0
- package/test/unit/solidity-types.test.js +49 -46
- package/test/unit/solidity-types.test.ts +39 -0
- package/test/unit/utils.test.js +57 -54
- package/test/unit/utils.test.ts +56 -0
- package/test/verbose-logger.js +74 -0
- package/tsconfig.build.json +14 -0
|
@@ -1,82 +1,85 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @testCategory unit
|
|
3
|
-
* @blockchainRequired false
|
|
4
|
-
* @transactional false
|
|
5
|
-
* @description Contract behavior that can be tested offline (overrides, listeners)
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
const { describe, it } = require("node:test");
|
|
9
|
-
const assert = require("node:assert/strict");
|
|
10
|
-
|
|
11
|
-
const { Initialize } = require("../../config");
|
|
12
|
-
const qc = require("../../index");
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
const
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
const
|
|
54
|
-
const
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
c.
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
c.
|
|
64
|
-
assert.equal(c._listeners.
|
|
65
|
-
|
|
66
|
-
c.
|
|
67
|
-
c.
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
1
|
+
/**
|
|
2
|
+
* @testCategory unit
|
|
3
|
+
* @blockchainRequired false
|
|
4
|
+
* @transactional false
|
|
5
|
+
* @description Contract behavior that can be tested offline (overrides, listeners)
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
const { describe, it } = require("node:test");
|
|
9
|
+
const assert = require("node:assert/strict");
|
|
10
|
+
|
|
11
|
+
const { Initialize } = require("../../config");
|
|
12
|
+
const qc = require("../../index");
|
|
13
|
+
const { logSuite, logTest } = require("../verbose-logger");
|
|
14
|
+
|
|
15
|
+
describe("Contract", () => {
|
|
16
|
+
logSuite("Contract");
|
|
17
|
+
it("_invoke treats last arg as overrides only when object-like (null is NOT overrides)", async () => {
|
|
18
|
+
logTest("_invoke treats last arg as overrides only when object-like (null is NOT overrides)", {});
|
|
19
|
+
await Initialize(null);
|
|
20
|
+
const addr = qc.Wallet.createRandom().address;
|
|
21
|
+
const abi = [
|
|
22
|
+
{ type: "function", name: "viewFn", stateMutability: "view", inputs: [{ name: "x", type: "uint256" }], outputs: [] },
|
|
23
|
+
{ type: "function", name: "writeFn", stateMutability: "nonpayable", inputs: [{ name: "x", type: "uint256" }], outputs: [] },
|
|
24
|
+
];
|
|
25
|
+
|
|
26
|
+
const c = new qc.Contract(addr, abi, null);
|
|
27
|
+
|
|
28
|
+
/** @type {any[]} */
|
|
29
|
+
const seen = [];
|
|
30
|
+
c.call = async (methodName, args, overrides) => {
|
|
31
|
+
seen.push({ kind: "call", methodName, args, overrides });
|
|
32
|
+
return "ok";
|
|
33
|
+
};
|
|
34
|
+
c.send = async (methodName, args, overrides) => {
|
|
35
|
+
seen.push({ kind: "send", methodName, args, overrides });
|
|
36
|
+
return { hash: "0x" + "22".repeat(32) };
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
// Overrides object extracted
|
|
40
|
+
await c._invoke("viewFn", [1n, { gasLimit: 123 }]);
|
|
41
|
+
// Passing null is NOT treated as overrides (it becomes a second argument)
|
|
42
|
+
await c._invoke("viewFn", [1n, null]);
|
|
43
|
+
// State-changing path chooses send()
|
|
44
|
+
await c._invoke("writeFn", [1n, { gasLimit: 456 }]);
|
|
45
|
+
|
|
46
|
+
assert.deepEqual(seen[0], { kind: "call", methodName: "viewFn", args: [1n], overrides: { gasLimit: 123 } });
|
|
47
|
+
assert.deepEqual(seen[1], { kind: "call", methodName: "viewFn", args: [1n, null], overrides: undefined });
|
|
48
|
+
assert.deepEqual(seen[2], { kind: "send", methodName: "writeFn", args: [1n], overrides: { gasLimit: 456 } });
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
it("removeAllListeners clears all when event omitted vs null, and clears one event when provided", async () => {
|
|
52
|
+
await Initialize(null);
|
|
53
|
+
const addr = qc.Wallet.createRandom().address;
|
|
54
|
+
const c = new qc.Contract(addr, [], null);
|
|
55
|
+
|
|
56
|
+
const cb1 = () => {};
|
|
57
|
+
const cb2 = () => {};
|
|
58
|
+
c.on("A", cb1);
|
|
59
|
+
c.on("B", cb2);
|
|
60
|
+
|
|
61
|
+
c.removeAllListeners("A");
|
|
62
|
+
// B remains
|
|
63
|
+
assert.equal(c._listeners.get("A")?.length || 0, 0);
|
|
64
|
+
assert.equal(c._listeners.get("B")?.length || 0, 1);
|
|
65
|
+
|
|
66
|
+
c.removeAllListeners(null); // treated as "no event"
|
|
67
|
+
assert.equal(c._listeners.size, 0);
|
|
68
|
+
|
|
69
|
+
c.on("C", cb1);
|
|
70
|
+
c.removeAllListeners(); // omitted => clear all
|
|
71
|
+
assert.equal(c._listeners.size, 0);
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
describe("ContractTransactionReceipt", () => {
|
|
76
|
+
it("getEvent/getEvents filter logs by eventName", () => {
|
|
77
|
+
const receipt = new qc.ContractTransactionReceipt({
|
|
78
|
+
logs: [{ eventName: "A", data: 1 }, { eventName: "B", data: 2 }, { eventName: "A", data: 3 }],
|
|
79
|
+
});
|
|
80
|
+
assert.deepEqual(receipt.getEvents("A").map((l) => l.data), [1, 3]);
|
|
81
|
+
assert.equal(receipt.getEvent("B").data, 2);
|
|
82
|
+
assert.equal(receipt.getEvent("Missing"), null);
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @testCategory unit
|
|
3
|
+
* @blockchainRequired false
|
|
4
|
+
* @transactional false
|
|
5
|
+
* @description Contract behavior that can be tested offline (overrides, listeners)
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { describe, it } from "node:test";
|
|
9
|
+
import assert from "node:assert/strict";
|
|
10
|
+
|
|
11
|
+
import { Initialize } from "../../config";
|
|
12
|
+
import qc from "../../index";
|
|
13
|
+
import { logSuite, logTest } from "../verbose-logger";
|
|
14
|
+
|
|
15
|
+
describe("Contract", () => {
|
|
16
|
+
logSuite("Contract");
|
|
17
|
+
it("_invoke treats last arg as overrides only when object-like (null is NOT overrides)", async () => {
|
|
18
|
+
logTest("_invoke treats last arg as overrides only when object-like (null is NOT overrides)", {});
|
|
19
|
+
await Initialize(null);
|
|
20
|
+
const addr = qc.Wallet.createRandom().address;
|
|
21
|
+
const abi = [
|
|
22
|
+
{ type: "function", name: "viewFn", stateMutability: "view", inputs: [{ name: "x", type: "uint256" }], outputs: [] },
|
|
23
|
+
{ type: "function", name: "writeFn", stateMutability: "nonpayable", inputs: [{ name: "x", type: "uint256" }], outputs: [] },
|
|
24
|
+
];
|
|
25
|
+
|
|
26
|
+
const c = new qc.Contract(addr, abi, null);
|
|
27
|
+
|
|
28
|
+
const seen: { kind: string; methodName: string; args: unknown[]; overrides?: unknown }[] = [];
|
|
29
|
+
c.call = async (methodName: string, args: unknown[], overrides?: unknown) => {
|
|
30
|
+
seen.push({ kind: "call", methodName, args, overrides });
|
|
31
|
+
return "ok";
|
|
32
|
+
};
|
|
33
|
+
c.send = async (methodName: string, args: unknown[], overrides?: unknown) => {
|
|
34
|
+
seen.push({ kind: "send", methodName, args, overrides });
|
|
35
|
+
return { hash: "0x" + "22".repeat(32) };
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
await c._invoke("viewFn", [1n, { gasLimit: 123 }]);
|
|
39
|
+
await c._invoke("viewFn", [1n, null]);
|
|
40
|
+
await c._invoke("writeFn", [1n, { gasLimit: 456 }]);
|
|
41
|
+
|
|
42
|
+
assert.deepEqual(seen[0], { kind: "call", methodName: "viewFn", args: [1n], overrides: { gasLimit: 123 } });
|
|
43
|
+
assert.deepEqual(seen[1], { kind: "call", methodName: "viewFn", args: [1n, null], overrides: undefined });
|
|
44
|
+
assert.deepEqual(seen[2], { kind: "send", methodName: "writeFn", args: [1n], overrides: { gasLimit: 456 } });
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
it("removeAllListeners clears all when event omitted vs null, and clears one event when provided", async () => {
|
|
48
|
+
await Initialize(null);
|
|
49
|
+
const addr = qc.Wallet.createRandom().address;
|
|
50
|
+
const c = new qc.Contract(addr, [], null);
|
|
51
|
+
|
|
52
|
+
const cb1 = () => {};
|
|
53
|
+
const cb2 = () => {};
|
|
54
|
+
c.on("A", cb1);
|
|
55
|
+
c.on("B", cb2);
|
|
56
|
+
|
|
57
|
+
c.removeAllListeners("A");
|
|
58
|
+
assert.equal(c._listeners.get("A")?.length || 0, 0);
|
|
59
|
+
assert.equal(c._listeners.get("B")?.length || 0, 1);
|
|
60
|
+
|
|
61
|
+
c.removeAllListeners(null);
|
|
62
|
+
assert.equal(c._listeners.size, 0);
|
|
63
|
+
|
|
64
|
+
c.on("C", cb1);
|
|
65
|
+
c.removeAllListeners();
|
|
66
|
+
assert.equal(c._listeners.size, 0);
|
|
67
|
+
});
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
describe("ContractTransactionReceipt", () => {
|
|
71
|
+
it("getEvent/getEvents filter logs by eventName", () => {
|
|
72
|
+
const receipt = new qc.ContractTransactionReceipt({
|
|
73
|
+
logs: [
|
|
74
|
+
{ eventName: "A", data: 1 },
|
|
75
|
+
{ eventName: "B", data: 2 },
|
|
76
|
+
{ eventName: "A", data: 3 },
|
|
77
|
+
],
|
|
78
|
+
});
|
|
79
|
+
assert.deepEqual(receipt.getEvents("A").map((l) => l.data), [1, 3]);
|
|
80
|
+
assert.equal(receipt.getEvent("B").data, 2);
|
|
81
|
+
assert.equal(receipt.getEvent("Missing"), null);
|
|
82
|
+
});
|
|
83
|
+
});
|
|
@@ -1,89 +1,92 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @testCategory unit
|
|
3
|
-
* @blockchainRequired false
|
|
4
|
-
* @transactional false
|
|
5
|
-
* @description Encoding + units + RLP edge cases (optional params, negative paths)
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
const { describe, it } = require("node:test");
|
|
9
|
-
const assert = require("node:assert/strict");
|
|
10
|
-
|
|
11
|
-
const qc = require("../../index");
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
1
|
+
/**
|
|
2
|
+
* @testCategory unit
|
|
3
|
+
* @blockchainRequired false
|
|
4
|
+
* @transactional false
|
|
5
|
+
* @description Encoding + units + RLP edge cases (optional params, negative paths)
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
const { describe, it } = require("node:test");
|
|
9
|
+
const assert = require("node:assert/strict");
|
|
10
|
+
|
|
11
|
+
const qc = require("../../index");
|
|
12
|
+
const { logSuite, logTest } = require("../verbose-logger");
|
|
13
|
+
|
|
14
|
+
describe("Encoding", () => {
|
|
15
|
+
logSuite("Encoding");
|
|
16
|
+
it("toHex/hexlify normalize strings and bytes", () => {
|
|
17
|
+
logTest("toHex/hexlify normalize strings and bytes", {});
|
|
18
|
+
assert.equal(qc.toHex("0x0A"), "0x0a");
|
|
19
|
+
assert.equal(qc.hexlify(new Uint8Array([0x0a])), "0x0a");
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
it("concat concatenates and rejects non-array", () => {
|
|
23
|
+
const out = qc.concat([new Uint8Array([1, 2]), "0x03"]);
|
|
24
|
+
assert.equal(out, "0x010203");
|
|
25
|
+
assert.throws(() => qc.concat("nope"));
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
it("stripZerosLeft removes leading zeros", () => {
|
|
29
|
+
assert.equal(qc.stripZerosLeft("0x0000"), "0x");
|
|
30
|
+
assert.equal(qc.stripZerosLeft("0x000001"), "0x01");
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
it("zeroPad throws when value exceeds length", () => {
|
|
34
|
+
assert.throws(() => qc.zeroPad("0x1234", 1));
|
|
35
|
+
assert.equal(qc.zeroPad("0x12", 2), "0x0012");
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
it("base64 encode/decode roundtrip", () => {
|
|
39
|
+
const b = qc.toUtf8Bytes("hello");
|
|
40
|
+
const b64 = qc.encodeBase64(b);
|
|
41
|
+
const out = qc.decodeBase64(b64);
|
|
42
|
+
assert.equal(qc.toUtf8String(out), "hello");
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it("isBytesLike returns true for hex and Uint8Array", () => {
|
|
46
|
+
assert.equal(qc.isBytesLike("0x1234"), true);
|
|
47
|
+
assert.equal(qc.isBytesLike(new Uint8Array([1, 2, 3])), true);
|
|
48
|
+
assert.equal(qc.isBytesLike("not-hex"), false);
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
it("solidityPacked helpers throw (not implemented)", () => {
|
|
52
|
+
assert.throws(() => qc.solidityPacked(), /not implemented/i);
|
|
53
|
+
assert.throws(() => qc.solidityPackedKeccak256(), /not implemented/i);
|
|
54
|
+
assert.throws(() => qc.solidityPackedSha256(), /not implemented/i);
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
describe("Units", () => {
|
|
59
|
+
it("parseUnits defaults decimals when omitted vs null", () => {
|
|
60
|
+
const a = qc.parseUnits("1.5");
|
|
61
|
+
const b = qc.parseUnits("1.5", null);
|
|
62
|
+
assert.equal(a, 1500000000000000000n);
|
|
63
|
+
assert.equal(a, b);
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
it("formatUnits defaults decimals when omitted vs null", () => {
|
|
67
|
+
const a = qc.formatUnits(1500000000000000000n);
|
|
68
|
+
const b = qc.formatUnits(1500000000000000000n, null);
|
|
69
|
+
assert.equal(a, "1.5");
|
|
70
|
+
assert.equal(a, b);
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
describe("RLP", () => {
|
|
75
|
+
it("decodeRlp rejects non-string input", () => {
|
|
76
|
+
assert.throws(() => qc.decodeRlp(123));
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
it("decodeRlp rejects trailing data", () => {
|
|
80
|
+
const encoded = qc.encodeRlp("0x01");
|
|
81
|
+
const withTrailing = "0x" + encoded.slice(2) + "00";
|
|
82
|
+
assert.throws(() => qc.decodeRlp(withTrailing), /trailing data/i);
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
it("encodeRlp supports null as empty bytes", () => {
|
|
86
|
+
const encoded = qc.encodeRlp(null);
|
|
87
|
+
assert.ok(typeof encoded === "string" && encoded.startsWith("0x"));
|
|
88
|
+
const decoded = qc.decodeRlp(encoded);
|
|
89
|
+
assert.equal(decoded, "0x");
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @testCategory unit
|
|
3
|
+
* @blockchainRequired false
|
|
4
|
+
* @transactional false
|
|
5
|
+
* @description Encoding + units + RLP edge cases (optional params, negative paths)
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { describe, it } from "node:test";
|
|
9
|
+
import assert from "node:assert/strict";
|
|
10
|
+
|
|
11
|
+
import qc from "../../index";
|
|
12
|
+
import { logSuite, logTest } from "../verbose-logger";
|
|
13
|
+
|
|
14
|
+
describe("Encoding", () => {
|
|
15
|
+
logSuite("Encoding");
|
|
16
|
+
it("toHex/hexlify normalize strings and bytes", () => {
|
|
17
|
+
logTest("toHex/hexlify normalize strings and bytes", {});
|
|
18
|
+
assert.equal(qc.toHex("0x0A"), "0x0a");
|
|
19
|
+
assert.equal(qc.hexlify(new Uint8Array([0x0a])), "0x0a");
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
it("concat concatenates and rejects non-array", () => {
|
|
23
|
+
const out = qc.concat([new Uint8Array([1, 2]), "0x03"]);
|
|
24
|
+
assert.equal(out, "0x010203");
|
|
25
|
+
assert.throws(() => qc.concat("nope" as unknown as (string | Uint8Array)[]));
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
it("stripZerosLeft removes leading zeros", () => {
|
|
29
|
+
assert.equal(qc.stripZerosLeft("0x0000"), "0x");
|
|
30
|
+
assert.equal(qc.stripZerosLeft("0x000001"), "0x01");
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
it("zeroPad throws when value exceeds length", () => {
|
|
34
|
+
assert.throws(() => qc.zeroPad("0x1234", 1));
|
|
35
|
+
assert.equal(qc.zeroPad("0x12", 2), "0x0012");
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
it("base64 encode/decode roundtrip", () => {
|
|
39
|
+
const b = qc.toUtf8Bytes("hello");
|
|
40
|
+
const b64 = qc.encodeBase64(b);
|
|
41
|
+
const out = qc.decodeBase64(b64);
|
|
42
|
+
assert.equal(qc.toUtf8String(out), "hello");
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it("isBytesLike returns true for hex and Uint8Array", () => {
|
|
46
|
+
assert.equal(qc.isBytesLike("0x1234"), true);
|
|
47
|
+
assert.equal(qc.isBytesLike(new Uint8Array([1, 2, 3])), true);
|
|
48
|
+
assert.equal(qc.isBytesLike("not-hex"), false);
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
it("solidityPacked helpers throw (not implemented)", () => {
|
|
52
|
+
assert.throws(() => qc.solidityPacked(), /not implemented/i);
|
|
53
|
+
assert.throws(() => qc.solidityPackedKeccak256(), /not implemented/i);
|
|
54
|
+
assert.throws(() => qc.solidityPackedSha256(), /not implemented/i);
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
describe("Units", () => {
|
|
59
|
+
it("parseUnits defaults decimals when omitted vs null", () => {
|
|
60
|
+
const a = qc.parseUnits("1.5");
|
|
61
|
+
const b = qc.parseUnits("1.5", null);
|
|
62
|
+
assert.equal(a, 1500000000000000000n);
|
|
63
|
+
assert.equal(a, b);
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
it("formatUnits defaults decimals when omitted vs null", () => {
|
|
67
|
+
const a = qc.formatUnits(1500000000000000000n);
|
|
68
|
+
const b = qc.formatUnits(1500000000000000000n, null);
|
|
69
|
+
assert.equal(a, "1.5");
|
|
70
|
+
assert.equal(a, b);
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
describe("RLP", () => {
|
|
75
|
+
it("decodeRlp rejects non-string input", () => {
|
|
76
|
+
assert.throws(() => qc.decodeRlp(123 as unknown as string));
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
it("decodeRlp rejects trailing data", () => {
|
|
80
|
+
const encoded = qc.encodeRlp("0x01");
|
|
81
|
+
const withTrailing = "0x" + encoded.slice(2) + "00";
|
|
82
|
+
assert.throws(() => qc.decodeRlp(withTrailing), /trailing data/i);
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
it("encodeRlp supports null as empty bytes", () => {
|
|
86
|
+
const encoded = qc.encodeRlp(null);
|
|
87
|
+
assert.ok(typeof encoded === "string" && encoded.startsWith("0x"));
|
|
88
|
+
const decoded = qc.decodeRlp(encoded);
|
|
89
|
+
assert.equal(decoded, "0x");
|
|
90
|
+
});
|
|
91
|
+
});
|