@bitgo-beta/sdk-coin-dot 2.2.8-beta.167 → 2.2.8-beta.1670
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/dist/{src → cjs/src}/dot.d.ts +8 -19
- package/dist/cjs/src/dot.d.ts.map +1 -0
- package/dist/cjs/src/dot.js +623 -0
- package/dist/cjs/src/index.d.ts.map +1 -0
- package/dist/{src → cjs/src}/index.js +6 -2
- package/dist/cjs/src/lib/addressInitializationBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/addressInitializationBuilder.js +182 -0
- package/dist/{src → cjs/src}/lib/batchTransactionBuilder.d.ts +1 -0
- package/dist/cjs/src/lib/batchTransactionBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/batchTransactionBuilder.js +229 -0
- package/dist/cjs/src/lib/claimBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/claimBuilder.js +117 -0
- package/dist/cjs/src/lib/errors.d.ts.map +1 -0
- package/dist/cjs/src/lib/errors.js +19 -0
- package/dist/{src → cjs/src}/lib/iface.d.ts +16 -13
- package/dist/cjs/src/lib/iface.d.ts.map +1 -0
- package/dist/cjs/src/lib/iface.js +128 -0
- package/dist/cjs/src/lib/iface_utils.d.ts.map +1 -0
- package/dist/cjs/src/lib/iface_utils.js +92 -0
- package/dist/{src → cjs/src}/lib/index.d.ts +2 -0
- package/dist/cjs/src/lib/index.d.ts.map +1 -0
- package/dist/cjs/src/lib/index.js +73 -0
- package/dist/{src → cjs/src}/lib/keyPair.d.ts +7 -0
- package/dist/cjs/src/lib/keyPair.d.ts.map +1 -0
- package/dist/cjs/src/lib/keyPair.js +117 -0
- package/dist/cjs/src/lib/nativeTransferBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/nativeTransferBuilder.js +219 -0
- package/dist/cjs/src/lib/proxyBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/proxyBuilder.js +116 -0
- package/dist/cjs/src/lib/singletonRegistry.d.ts.map +1 -0
- package/dist/cjs/src/lib/singletonRegistry.js +20 -0
- package/dist/cjs/src/lib/stakingBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/stakingBuilder.js +167 -0
- package/dist/{src → cjs/src}/lib/transaction.d.ts +0 -1
- package/dist/cjs/src/lib/transaction.d.ts.map +1 -0
- package/dist/cjs/src/lib/transaction.js +624 -0
- package/dist/{src → cjs/src}/lib/transactionBuilder.d.ts +0 -1
- package/dist/cjs/src/lib/transactionBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/transactionBuilder.js +351 -0
- package/dist/cjs/src/lib/transactionBuilderFactory.d.ts.map +1 -0
- package/dist/cjs/src/lib/transactionBuilderFactory.js +104 -0
- package/dist/cjs/src/lib/transferBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/transferBuilder.js +11 -0
- package/dist/cjs/src/lib/txnSchema.d.ts.map +1 -0
- package/dist/cjs/src/lib/txnSchema.js +117 -0
- package/dist/cjs/src/lib/unnominateBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/unnominateBuilder.js +48 -0
- package/dist/cjs/src/lib/unstakeBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/unstakeBuilder.js +87 -0
- package/dist/{src → cjs/src}/lib/utils.d.ts +10 -2
- package/dist/cjs/src/lib/utils.d.ts.map +1 -0
- package/dist/cjs/src/lib/utils.js +491 -0
- package/dist/cjs/src/lib/wasmParser.d.ts +57 -0
- package/dist/cjs/src/lib/wasmParser.d.ts.map +1 -0
- package/dist/cjs/src/lib/wasmParser.js +345 -0
- package/dist/cjs/src/lib/withdrawUnstakedBuilder.d.ts.map +1 -0
- package/dist/cjs/src/lib/withdrawUnstakedBuilder.js +88 -0
- package/dist/cjs/src/register.d.ts.map +1 -0
- package/dist/cjs/src/register.js +11 -0
- package/dist/cjs/src/resources/index.d.ts +5 -0
- package/dist/cjs/src/resources/index.d.ts.map +1 -0
- package/dist/cjs/src/resources/index.js +21 -0
- package/dist/cjs/src/resources/mainnet.d.ts.map +1 -0
- package/dist/cjs/src/resources/mainnet.js +5 -0
- package/dist/cjs/src/resources/polkadotAssetHub.d.ts +2 -0
- package/dist/cjs/src/resources/polkadotAssetHub.d.ts.map +1 -0
- package/dist/cjs/src/resources/polkadotAssetHub.js +5 -0
- package/dist/cjs/src/resources/westend.d.ts.map +1 -0
- package/dist/cjs/src/resources/westend.js +5 -0
- package/dist/cjs/src/resources/westendAssetHub.d.ts +2 -0
- package/dist/cjs/src/resources/westendAssetHub.d.ts.map +1 -0
- package/dist/cjs/src/resources/westendAssetHub.js +5 -0
- package/dist/cjs/src/tdot.d.ts.map +1 -0
- package/dist/cjs/src/tdot.js +27 -0
- package/dist/cjs/test/fixtures.d.ts +283 -0
- package/dist/cjs/test/fixtures.d.ts.map +1 -0
- package/dist/cjs/test/fixtures.js +439 -0
- package/dist/cjs/test/resources/index.d.ts +141 -0
- package/dist/cjs/test/resources/index.d.ts.map +1 -0
- package/dist/cjs/test/resources/index.js +157 -0
- package/dist/cjs/test/resources/materialData.json +8 -0
- package/dist/cjs/test/resources/materialDataModified.json +8 -0
- package/dist/cjs/test/resources/testnet.d.ts +2 -0
- package/dist/cjs/test/resources/testnet.d.ts.map +1 -0
- package/dist/{src/resources/westend.js → cjs/test/resources/testnet.js} +3 -3
- package/dist/cjs/test/unit/address.d.ts +2 -0
- package/dist/cjs/test/unit/address.d.ts.map +1 -0
- package/dist/cjs/test/unit/address.js +45 -0
- package/dist/cjs/test/unit/dot.d.ts +2 -0
- package/dist/cjs/test/unit/dot.d.ts.map +1 -0
- package/dist/cjs/test/unit/dot.js +944 -0
- package/dist/cjs/test/unit/keypair.d.ts +2 -0
- package/dist/cjs/test/unit/keypair.d.ts.map +1 -0
- package/dist/cjs/test/unit/keypair.js +111 -0
- package/dist/cjs/test/unit/transaction.d.ts +2 -0
- package/dist/cjs/test/unit/transaction.d.ts.map +1 -0
- package/dist/cjs/test/unit/transaction.js +269 -0
- package/dist/cjs/test/unit/transactionBuilder/addressInitializationBuilder.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/addressInitializationBuilder.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/addressInitializationBuilder.js +265 -0
- package/dist/cjs/test/unit/transactionBuilder/base.d.ts +8 -0
- package/dist/cjs/test/unit/transactionBuilder/base.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/base.js +233 -0
- package/dist/cjs/test/unit/transactionBuilder/batchTransactionBuilder.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/batchTransactionBuilder.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/batchTransactionBuilder.js +521 -0
- package/dist/cjs/test/unit/transactionBuilder/claimBuilder.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/claimBuilder.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/claimBuilder.js +129 -0
- package/dist/cjs/test/unit/transactionBuilder/proxyBuilder.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/proxyBuilder.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/proxyBuilder.js +88 -0
- package/dist/cjs/test/unit/transactionBuilder/singeltonRegistry.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/singeltonRegistry.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/singeltonRegistry.js +59 -0
- package/dist/cjs/test/unit/transactionBuilder/stakingBuilder.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/stakingBuilder.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/stakingBuilder.js +235 -0
- package/dist/cjs/test/unit/transactionBuilder/transactionBuilderFactory.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/transactionBuilderFactory.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/transactionBuilderFactory.js +107 -0
- package/dist/cjs/test/unit/transactionBuilder/transferBuilder.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/transferBuilder.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/transferBuilder.js +437 -0
- package/dist/cjs/test/unit/transactionBuilder/unnominateBuilder.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/unnominateBuilder.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/unnominateBuilder.js +97 -0
- package/dist/cjs/test/unit/transactionBuilder/unstakeBuilder.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/unstakeBuilder.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/unstakeBuilder.js +116 -0
- package/dist/cjs/test/unit/transactionBuilder/withdrawUnstakedBuilder.d.ts +2 -0
- package/dist/cjs/test/unit/transactionBuilder/withdrawUnstakedBuilder.d.ts.map +1 -0
- package/dist/cjs/test/unit/transactionBuilder/withdrawUnstakedBuilder.js +115 -0
- package/dist/cjs/test/unit/utils.d.ts +2 -0
- package/dist/cjs/test/unit/utils.d.ts.map +1 -0
- package/dist/cjs/test/unit/utils.js +117 -0
- package/dist/cjs/test/unit/wasmBuilderByteComparison.d.ts +22 -0
- package/dist/cjs/test/unit/wasmBuilderByteComparison.d.ts.map +1 -0
- package/dist/cjs/test/unit/wasmBuilderByteComparison.js +197 -0
- package/dist/cjs/test/unit/wasmParserExplanation.d.ts +15 -0
- package/dist/cjs/test/unit/wasmParserExplanation.d.ts.map +1 -0
- package/dist/cjs/test/unit/wasmParserExplanation.js +147 -0
- package/dist/cjs/tsconfig.tsbuildinfo +1 -0
- package/dist/esm/dot.d.ts +147 -0
- package/dist/esm/dot.d.ts.map +1 -0
- package/dist/esm/dot.js +583 -0
- package/dist/esm/index.d.ts +5 -0
- package/dist/esm/index.js +5 -0
- package/dist/esm/lib/addressInitializationBuilder.d.ts +79 -0
- package/dist/esm/lib/addressInitializationBuilder.js +175 -0
- package/dist/esm/lib/batchTransactionBuilder.d.ts +62 -0
- package/dist/{src → esm}/lib/batchTransactionBuilder.d.ts.map +1 -1
- package/dist/esm/lib/batchTransactionBuilder.js +222 -0
- package/dist/esm/lib/claimBuilder.d.ts +58 -0
- package/dist/esm/lib/claimBuilder.js +110 -0
- package/dist/esm/lib/errors.d.ts +8 -0
- package/dist/esm/lib/errors.js +14 -0
- package/dist/esm/lib/iface.d.ts +337 -0
- package/dist/esm/lib/iface.d.ts.map +1 -0
- package/dist/esm/lib/iface.js +125 -0
- package/dist/esm/lib/iface_utils.d.ts +59 -0
- package/dist/esm/lib/iface_utils.js +83 -0
- package/dist/esm/lib/index.d.ts +21 -0
- package/dist/{src → esm}/lib/index.d.ts.map +1 -1
- package/dist/esm/lib/index.js +20 -0
- package/dist/esm/lib/keyPair.d.ts +38 -0
- package/dist/esm/lib/keyPair.d.ts.map +1 -0
- package/dist/esm/lib/keyPair.js +77 -0
- package/dist/esm/lib/nativeTransferBuilder.d.ts +84 -0
- package/dist/esm/lib/nativeTransferBuilder.js +212 -0
- package/dist/esm/lib/proxyBuilder.d.ts +54 -0
- package/dist/esm/lib/proxyBuilder.js +109 -0
- package/dist/esm/lib/singletonRegistry.d.ts +7 -0
- package/dist/esm/lib/singletonRegistry.js +16 -0
- package/dist/esm/lib/stakingBuilder.d.ts +66 -0
- package/dist/esm/lib/stakingBuilder.js +160 -0
- package/dist/esm/lib/transaction.d.ts +83 -0
- package/dist/esm/lib/transaction.d.ts.map +1 -0
- package/dist/esm/lib/transaction.js +584 -0
- package/dist/esm/lib/transactionBuilder.d.ts +131 -0
- package/dist/esm/lib/transactionBuilder.d.ts.map +1 -0
- package/dist/esm/lib/transactionBuilder.js +311 -0
- package/dist/esm/lib/transactionBuilderFactory.d.ts +28 -0
- package/dist/esm/lib/transactionBuilderFactory.js +97 -0
- package/dist/esm/lib/transferBuilder.d.ts +11 -0
- package/dist/esm/lib/transferBuilder.js +7 -0
- package/dist/esm/lib/txnSchema.d.ts +15 -0
- package/dist/esm/lib/txnSchema.js +111 -0
- package/dist/esm/lib/unnominateBuilder.d.ts +24 -0
- package/dist/esm/lib/unnominateBuilder.js +44 -0
- package/dist/esm/lib/unstakeBuilder.d.ts +38 -0
- package/dist/esm/lib/unstakeBuilder.js +80 -0
- package/dist/esm/lib/utils.d.ts +261 -0
- package/dist/esm/lib/utils.d.ts.map +1 -0
- package/dist/esm/lib/utils.js +451 -0
- package/dist/esm/lib/wasmParser.d.ts +57 -0
- package/dist/esm/lib/wasmParser.d.ts.map +1 -0
- package/dist/esm/lib/wasmParser.js +341 -0
- package/dist/esm/lib/withdrawUnstakedBuilder.d.ts +38 -0
- package/dist/esm/lib/withdrawUnstakedBuilder.js +81 -0
- package/dist/esm/register.d.ts +3 -0
- package/dist/esm/register.js +7 -0
- package/dist/esm/resources/index.d.ts +5 -0
- package/dist/{src → esm}/resources/index.d.ts.map +1 -1
- package/dist/esm/resources/index.js +5 -0
- package/dist/esm/resources/mainnet.d.ts +2 -0
- package/dist/esm/resources/mainnet.js +2 -0
- package/dist/esm/resources/polkadotAssetHub.d.ts +2 -0
- package/dist/esm/resources/polkadotAssetHub.d.ts.map +1 -0
- package/dist/esm/resources/polkadotAssetHub.js +2 -0
- package/dist/esm/resources/westend.d.ts +2 -0
- package/dist/esm/resources/westend.js +2 -0
- package/dist/esm/resources/westendAssetHub.d.ts +2 -0
- package/dist/esm/resources/westendAssetHub.d.ts.map +1 -0
- package/dist/esm/resources/westendAssetHub.js +2 -0
- package/dist/esm/tdot.d.ts +12 -0
- package/dist/esm/tdot.js +23 -0
- package/package.json +43 -21
- package/.eslintignore +0 -5
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -450
- package/dist/src/dot.d.ts.map +0 -1
- package/dist/src/dot.js +0 -562
- package/dist/src/lib/addressInitializationBuilder.js +0 -184
- package/dist/src/lib/batchTransactionBuilder.js +0 -220
- package/dist/src/lib/claimBuilder.js +0 -119
- package/dist/src/lib/errors.js +0 -19
- package/dist/src/lib/iface.d.ts.map +0 -1
- package/dist/src/lib/iface.js +0 -128
- package/dist/src/lib/iface_utils.js +0 -93
- package/dist/src/lib/index.js +0 -57
- package/dist/src/lib/keyPair.d.ts.map +0 -1
- package/dist/src/lib/keyPair.js +0 -70
- package/dist/src/lib/nativeTransferBuilder.js +0 -221
- package/dist/src/lib/proxyBuilder.js +0 -118
- package/dist/src/lib/singletonRegistry.js +0 -20
- package/dist/src/lib/stakingBuilder.js +0 -169
- package/dist/src/lib/transaction.d.ts.map +0 -1
- package/dist/src/lib/transaction.js +0 -590
- package/dist/src/lib/transactionBuilder.d.ts.map +0 -1
- package/dist/src/lib/transactionBuilder.js +0 -337
- package/dist/src/lib/transactionBuilderFactory.js +0 -105
- package/dist/src/lib/transferBuilder.js +0 -11
- package/dist/src/lib/txnSchema.js +0 -117
- package/dist/src/lib/unnominateBuilder.js +0 -50
- package/dist/src/lib/unstakeBuilder.js +0 -89
- package/dist/src/lib/utils.d.ts.map +0 -1
- package/dist/src/lib/utils.js +0 -458
- package/dist/src/lib/withdrawUnstakedBuilder.js +0 -90
- package/dist/src/register.js +0 -11
- package/dist/src/resources/index.d.ts +0 -3
- package/dist/src/resources/index.js +0 -15
- package/dist/src/resources/mainnet.js +0 -5
- package/dist/src/tdot.js +0 -27
- /package/dist/{src → cjs/src}/index.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/addressInitializationBuilder.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/claimBuilder.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/errors.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/iface_utils.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/nativeTransferBuilder.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/proxyBuilder.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/singletonRegistry.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/stakingBuilder.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/transactionBuilderFactory.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/transferBuilder.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/txnSchema.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/unnominateBuilder.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/unstakeBuilder.d.ts +0 -0
- /package/dist/{src → cjs/src}/lib/withdrawUnstakedBuilder.d.ts +0 -0
- /package/dist/{src → cjs/src}/register.d.ts +0 -0
- /package/dist/{src → cjs/src}/resources/mainnet.d.ts +0 -0
- /package/dist/{src → cjs/src}/resources/westend.d.ts +0 -0
- /package/dist/{src → cjs/src}/tdot.d.ts +0 -0
- /package/dist/{src → esm}/index.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/addressInitializationBuilder.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/claimBuilder.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/errors.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/iface_utils.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/nativeTransferBuilder.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/proxyBuilder.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/singletonRegistry.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/stakingBuilder.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/transactionBuilderFactory.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/transferBuilder.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/txnSchema.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/unnominateBuilder.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/unstakeBuilder.d.ts.map +0 -0
- /package/dist/{src → esm}/lib/withdrawUnstakedBuilder.d.ts.map +0 -0
- /package/dist/{src → esm}/register.d.ts.map +0 -0
- /package/dist/{src → esm}/resources/mainnet.d.ts.map +0 -0
- /package/dist/{src → esm}/resources/westend.d.ts.map +0 -0
- /package/dist/{src → esm}/tdot.d.ts.map +0 -0
|
@@ -0,0 +1,197 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* WASM Builder Byte Comparison Tests
|
|
4
|
+
*
|
|
5
|
+
* Compare serialized output between:
|
|
6
|
+
* 1. Legacy approach (using @substrate/txwrapper-polkadot)
|
|
7
|
+
* 2. WASM approach (using @bitgo/wasm-dot buildTransaction)
|
|
8
|
+
*
|
|
9
|
+
* For unsigned transactions, legacy toBroadcastFormat() returns the signing payload
|
|
10
|
+
* (via construct.signingPayload). We compare WASM signablePayload() against it.
|
|
11
|
+
*
|
|
12
|
+
* Format difference: txwrapper encodes the call as `Bytes` (Vec<u8>) which includes
|
|
13
|
+
* a SCALE compact-length prefix. subxt encodes it as raw `Call` (no prefix).
|
|
14
|
+
* We strip this prefix from the legacy side before comparing, since the actual
|
|
15
|
+
* call data, era, nonce, tip, and chain context are identical.
|
|
16
|
+
*
|
|
17
|
+
* Note: wasm-dot@1.3.0 uses high-level business intents (payment, stake, unstake,
|
|
18
|
+
* claim, consolidate) instead of low-level call types. Batch and proxy operations
|
|
19
|
+
* are composed automatically from the intent (e.g., stake with proxyAddress
|
|
20
|
+
* produces batchAll(bond, addProxy)).
|
|
21
|
+
*/
|
|
22
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
23
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
24
|
+
};
|
|
25
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
26
|
+
const assert_1 = __importDefault(require("assert"));
|
|
27
|
+
const statics_1 = require("@bitgo-beta/statics");
|
|
28
|
+
const transactionBuilderFactory_1 = require("../../src/lib/transactionBuilderFactory");
|
|
29
|
+
const fixtures_1 = require("../fixtures");
|
|
30
|
+
const utils_1 = __importDefault(require("../../src/lib/utils"));
|
|
31
|
+
// Import WASM builder
|
|
32
|
+
const wasm_dot_1 = require("@bitgo/wasm-dot");
|
|
33
|
+
describe('WASM vs Legacy Builder Byte Comparison', function () {
|
|
34
|
+
const coin = statics_1.coins.get('tdot');
|
|
35
|
+
// Get material from utils to ensure same metadata as legacy builder
|
|
36
|
+
const material = utils_1.default.getMaterial(coin);
|
|
37
|
+
function createWasmContext(overrides = {}) {
|
|
38
|
+
return {
|
|
39
|
+
sender: fixtures_1.accounts.account1.address,
|
|
40
|
+
nonce: 0,
|
|
41
|
+
tip: 0n,
|
|
42
|
+
material: material,
|
|
43
|
+
validity: {
|
|
44
|
+
firstValid: fixtures_1.westendBlock.blockNumber,
|
|
45
|
+
maxDuration: 2400,
|
|
46
|
+
},
|
|
47
|
+
referenceBlock: fixtures_1.westendBlock.hash,
|
|
48
|
+
...overrides,
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* Strip SCALE compact-length prefix from the legacy signing payload.
|
|
53
|
+
*
|
|
54
|
+
* Legacy (txwrapper) encodes the call as `Bytes` type, which adds a compact-length
|
|
55
|
+
* prefix. subxt encodes it as raw `Call` (no prefix). Both produce identical
|
|
56
|
+
* call data + era + nonce + tip + chain context after this prefix.
|
|
57
|
+
*/
|
|
58
|
+
function stripCompactPrefix(hex) {
|
|
59
|
+
const data = hex.startsWith('0x') ? hex.slice(2) : hex;
|
|
60
|
+
const bytes = Buffer.from(data, 'hex');
|
|
61
|
+
const mode = bytes[0] & 0b11;
|
|
62
|
+
let offset;
|
|
63
|
+
if (mode === 0b00)
|
|
64
|
+
offset = 1;
|
|
65
|
+
else if (mode === 0b01)
|
|
66
|
+
offset = 2;
|
|
67
|
+
else if (mode === 0b10)
|
|
68
|
+
offset = 4;
|
|
69
|
+
else
|
|
70
|
+
throw new Error('Big compact not supported');
|
|
71
|
+
return '0x' + bytes.slice(offset).toString('hex');
|
|
72
|
+
}
|
|
73
|
+
// ===========================================================================
|
|
74
|
+
// Transfer Transaction Tests
|
|
75
|
+
// ===========================================================================
|
|
76
|
+
describe('Transfer Transactions', function () {
|
|
77
|
+
it('should produce identical signing payload for transfer', async function () {
|
|
78
|
+
const to = fixtures_1.accounts.account2.address;
|
|
79
|
+
const amount = '1000000000000'; // 1 DOT
|
|
80
|
+
const factory = new transactionBuilderFactory_1.TransactionBuilderFactory(coin);
|
|
81
|
+
const legacyBuilder = factory.getTransferBuilder();
|
|
82
|
+
legacyBuilder
|
|
83
|
+
.sender({ address: fixtures_1.accounts.account1.address })
|
|
84
|
+
.to({ address: to })
|
|
85
|
+
.amount(amount)
|
|
86
|
+
.validity({ firstValid: fixtures_1.westendBlock.blockNumber, maxDuration: 2400 })
|
|
87
|
+
.referenceBlock(fixtures_1.westendBlock.hash)
|
|
88
|
+
.sequenceId({ name: 'Nonce', keyword: 'nonce', value: 0 });
|
|
89
|
+
const legacyTx = await legacyBuilder.build();
|
|
90
|
+
const legacyHex = legacyTx.toBroadcastFormat();
|
|
91
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'payment', to, amount: BigInt(amount), keepAlive: true }, createWasmContext());
|
|
92
|
+
const wasmHex = '0x' + Buffer.from(wasmTx.signablePayload()).toString('hex');
|
|
93
|
+
assert_1.default.strictEqual(wasmHex, stripCompactPrefix(legacyHex), 'Signing payload should match');
|
|
94
|
+
});
|
|
95
|
+
it('should produce identical signing payload for transferKeepAlive with different nonce', async function () {
|
|
96
|
+
const to = fixtures_1.accounts.account2.address;
|
|
97
|
+
const amount = '5000000000000'; // 5 DOT
|
|
98
|
+
const factory = new transactionBuilderFactory_1.TransactionBuilderFactory(coin);
|
|
99
|
+
const legacyBuilder = factory.getTransferBuilder();
|
|
100
|
+
legacyBuilder
|
|
101
|
+
.sender({ address: fixtures_1.accounts.account1.address })
|
|
102
|
+
.to({ address: to })
|
|
103
|
+
.amount(amount)
|
|
104
|
+
.validity({ firstValid: fixtures_1.westendBlock.blockNumber, maxDuration: 2400 })
|
|
105
|
+
.referenceBlock(fixtures_1.westendBlock.hash)
|
|
106
|
+
.sequenceId({ name: 'Nonce', keyword: 'nonce', value: 5 });
|
|
107
|
+
const legacyTx = await legacyBuilder.build();
|
|
108
|
+
const legacyHex = legacyTx.toBroadcastFormat();
|
|
109
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'payment', to, amount: BigInt(amount), keepAlive: true }, createWasmContext({ nonce: 5 }));
|
|
110
|
+
const wasmHex = '0x' + Buffer.from(wasmTx.signablePayload()).toString('hex');
|
|
111
|
+
assert_1.default.strictEqual(wasmHex, stripCompactPrefix(legacyHex), 'Signing payload should match');
|
|
112
|
+
});
|
|
113
|
+
});
|
|
114
|
+
// ===========================================================================
|
|
115
|
+
// Staking Transaction Tests
|
|
116
|
+
// ===========================================================================
|
|
117
|
+
describe('Staking Transactions', function () {
|
|
118
|
+
// Note: wasm-dot@1.3.0 `stake` without proxyAddress produces bondExtra (top-up),
|
|
119
|
+
// while the legacy getStakingBuilder() produces bond (initial stake). These are
|
|
120
|
+
// different extrinsic calls so byte comparison is not applicable. The bond call
|
|
121
|
+
// is now only produced via `stake` with proxyAddress (which creates a batch).
|
|
122
|
+
// See Intent-based Transaction Building tests below for stake sanity checks.
|
|
123
|
+
it('should produce identical signing payload for unstake (unbond)', async function () {
|
|
124
|
+
const amount = '5000000000000'; // 5 DOT
|
|
125
|
+
const factory = new transactionBuilderFactory_1.TransactionBuilderFactory(coin);
|
|
126
|
+
const legacyBuilder = factory.getUnstakeBuilder();
|
|
127
|
+
legacyBuilder
|
|
128
|
+
.sender({ address: fixtures_1.accounts.account1.address })
|
|
129
|
+
.amount(amount)
|
|
130
|
+
.validity({ firstValid: fixtures_1.westendBlock.blockNumber, maxDuration: 2400 })
|
|
131
|
+
.referenceBlock(fixtures_1.westendBlock.hash)
|
|
132
|
+
.sequenceId({ name: 'Nonce', keyword: 'nonce', value: 0 });
|
|
133
|
+
const legacyTx = await legacyBuilder.build();
|
|
134
|
+
const legacyHex = legacyTx.toBroadcastFormat();
|
|
135
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'unstake', amount: BigInt(amount) }, createWasmContext());
|
|
136
|
+
const wasmHex = '0x' + Buffer.from(wasmTx.signablePayload()).toString('hex');
|
|
137
|
+
assert_1.default.strictEqual(wasmHex, stripCompactPrefix(legacyHex), 'Signing payload should match');
|
|
138
|
+
});
|
|
139
|
+
it('should produce identical signing payload for withdrawUnbonded (claim)', async function () {
|
|
140
|
+
const slashingSpans = 0;
|
|
141
|
+
const factory = new transactionBuilderFactory_1.TransactionBuilderFactory(coin);
|
|
142
|
+
const legacyBuilder = factory.getWithdrawUnstakedBuilder();
|
|
143
|
+
legacyBuilder
|
|
144
|
+
.sender({ address: fixtures_1.accounts.account1.address })
|
|
145
|
+
.slashingSpans(slashingSpans)
|
|
146
|
+
.validity({ firstValid: fixtures_1.westendBlock.blockNumber, maxDuration: 2400 })
|
|
147
|
+
.referenceBlock(fixtures_1.westendBlock.hash)
|
|
148
|
+
.sequenceId({ name: 'Nonce', keyword: 'nonce', value: 0 });
|
|
149
|
+
const legacyTx = await legacyBuilder.build();
|
|
150
|
+
const legacyHex = legacyTx.toBroadcastFormat();
|
|
151
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'claim', slashingSpans }, createWasmContext());
|
|
152
|
+
const wasmHex = '0x' + Buffer.from(wasmTx.signablePayload()).toString('hex');
|
|
153
|
+
assert_1.default.strictEqual(wasmHex, stripCompactPrefix(legacyHex), 'Signing payload should match');
|
|
154
|
+
});
|
|
155
|
+
});
|
|
156
|
+
// ===========================================================================
|
|
157
|
+
// Intent-based Transaction Building (sanity checks)
|
|
158
|
+
// ===========================================================================
|
|
159
|
+
describe('Intent-based Transaction Building', function () {
|
|
160
|
+
it('should build payment from intent', async function () {
|
|
161
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'payment', to: fixtures_1.accounts.account2.address, amount: 1000000000000n, keepAlive: true }, createWasmContext());
|
|
162
|
+
const serialized = wasmTx.toBroadcastFormat();
|
|
163
|
+
(0, assert_1.default)(serialized.startsWith('0x'), 'Should be hex encoded');
|
|
164
|
+
(0, assert_1.default)(serialized.length > 10, 'Should have content');
|
|
165
|
+
});
|
|
166
|
+
it('should build stake (top-up) from intent', async function () {
|
|
167
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'stake', amount: 5000000000000n, payee: { type: 'staked' } }, createWasmContext());
|
|
168
|
+
(0, assert_1.default)(wasmTx.toBroadcastFormat().startsWith('0x'));
|
|
169
|
+
});
|
|
170
|
+
it('should build stake with proxy (new stake / batchAll(bond, addProxy)) from intent', async function () {
|
|
171
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'stake', amount: 10000000000000n, proxyAddress: fixtures_1.accounts.account2.address, payee: { type: 'staked' } }, createWasmContext());
|
|
172
|
+
(0, assert_1.default)(wasmTx.toBroadcastFormat().startsWith('0x'));
|
|
173
|
+
});
|
|
174
|
+
it('should build claim (withdrawUnbonded) from intent', async function () {
|
|
175
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'claim', slashingSpans: 0 }, createWasmContext());
|
|
176
|
+
(0, assert_1.default)(wasmTx.toBroadcastFormat().startsWith('0x'));
|
|
177
|
+
});
|
|
178
|
+
it('should build unstake (partial / unbond) from intent', async function () {
|
|
179
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'unstake', amount: 5000000000000n }, createWasmContext());
|
|
180
|
+
(0, assert_1.default)(wasmTx.toBroadcastFormat().startsWith('0x'));
|
|
181
|
+
});
|
|
182
|
+
it('should build unstake with stopStaking (full / batchAll(removeProxy, chill, unbond)) from intent', async function () {
|
|
183
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({
|
|
184
|
+
type: 'unstake',
|
|
185
|
+
amount: 5000000000000n,
|
|
186
|
+
stopStaking: true,
|
|
187
|
+
proxyAddress: fixtures_1.accounts.account2.address,
|
|
188
|
+
}, createWasmContext());
|
|
189
|
+
(0, assert_1.default)(wasmTx.toBroadcastFormat().startsWith('0x'));
|
|
190
|
+
});
|
|
191
|
+
it('should build consolidate (transferAll) from intent', async function () {
|
|
192
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'consolidate', to: fixtures_1.accounts.account2.address }, createWasmContext());
|
|
193
|
+
(0, assert_1.default)(wasmTx.toBroadcastFormat().startsWith('0x'));
|
|
194
|
+
});
|
|
195
|
+
});
|
|
196
|
+
});
|
|
197
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"wasmBuilderByteComparison.js","sourceRoot":"","sources":["../../../../test/unit/wasmBuilderByteComparison.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * WASM Builder Byte Comparison Tests\n *\n * Compare serialized output between:\n * 1. Legacy approach (using @substrate/txwrapper-polkadot)\n * 2. WASM approach (using @bitgo/wasm-dot buildTransaction)\n *\n * For unsigned transactions, legacy toBroadcastFormat() returns the signing payload\n * (via construct.signingPayload). We compare WASM signablePayload() against it.\n *\n * Format difference: txwrapper encodes the call as `Bytes` (Vec<u8>) which includes\n * a SCALE compact-length prefix. subxt encodes it as raw `Call` (no prefix).\n * We strip this prefix from the legacy side before comparing, since the actual\n * call data, era, nonce, tip, and chain context are identical.\n *\n * Note: wasm-dot@1.3.0 uses high-level business intents (payment, stake, unstake,\n * claim, consolidate) instead of low-level call types. Batch and proxy operations\n * are composed automatically from the intent (e.g., stake with proxyAddress\n * produces batchAll(bond, addProxy)).\n */\n\nimport assert from 'assert';\nimport { coins } from '@bitgo-beta/statics';\nimport { TransactionBuilderFactory } from '../../src/lib/transactionBuilderFactory';\nimport { TransferBuilder } from '../../src/lib/transferBuilder';\nimport { accounts, westendBlock } from '../fixtures';\nimport utils from '../../src/lib/utils';\n\n// Import WASM builder\nimport { buildTransaction, type BuildContext, type Material } from '@bitgo/wasm-dot';\n\ndescribe('WASM vs Legacy Builder Byte Comparison', function () {\n  const coin = coins.get('tdot');\n\n  // Get material from utils to ensure same metadata as legacy builder\n  const material = utils.getMaterial(coin);\n\n  function createWasmContext(overrides: Partial<BuildContext> = {}): BuildContext {\n    return {\n      sender: accounts.account1.address,\n      nonce: 0,\n      tip: 0n,\n      material: material as Material,\n      validity: {\n        firstValid: westendBlock.blockNumber,\n        maxDuration: 2400,\n      },\n      referenceBlock: westendBlock.hash,\n      ...overrides,\n    };\n  }\n\n  /**\n   * Strip SCALE compact-length prefix from the legacy signing payload.\n   *\n   * Legacy (txwrapper) encodes the call as `Bytes` type, which adds a compact-length\n   * prefix. subxt encodes it as raw `Call` (no prefix). Both produce identical\n   * call data + era + nonce + tip + chain context after this prefix.\n   */\n  function stripCompactPrefix(hex: string): string {\n    const data = hex.startsWith('0x') ? hex.slice(2) : hex;\n    const bytes = Buffer.from(data, 'hex');\n    const mode = bytes[0] & 0b11;\n    let offset: number;\n    if (mode === 0b00) offset = 1;\n    else if (mode === 0b01) offset = 2;\n    else if (mode === 0b10) offset = 4;\n    else throw new Error('Big compact not supported');\n    return '0x' + bytes.slice(offset).toString('hex');\n  }\n\n  // ===========================================================================\n  // Transfer Transaction Tests\n  // ===========================================================================\n  describe('Transfer Transactions', function () {\n    it('should produce identical signing payload for transfer', async function () {\n      const to = accounts.account2.address;\n      const amount = '1000000000000'; // 1 DOT\n\n      const factory = new TransactionBuilderFactory(coin);\n      const legacyBuilder = factory.getTransferBuilder() as TransferBuilder;\n\n      legacyBuilder\n        .sender({ address: accounts.account1.address })\n        .to({ address: to })\n        .amount(amount)\n        .validity({ firstValid: westendBlock.blockNumber, maxDuration: 2400 })\n        .referenceBlock(westendBlock.hash)\n        .sequenceId({ name: 'Nonce', keyword: 'nonce', value: 0 });\n\n      const legacyTx = await legacyBuilder.build();\n      const legacyHex = legacyTx.toBroadcastFormat();\n\n      const wasmTx = buildTransaction(\n        { type: 'payment', to, amount: BigInt(amount), keepAlive: true },\n        createWasmContext()\n      );\n      const wasmHex = '0x' + Buffer.from(wasmTx.signablePayload()).toString('hex');\n\n      assert.strictEqual(wasmHex, stripCompactPrefix(legacyHex), 'Signing payload should match');\n    });\n\n    it('should produce identical signing payload for transferKeepAlive with different nonce', async function () {\n      const to = accounts.account2.address;\n      const amount = '5000000000000'; // 5 DOT\n\n      const factory = new TransactionBuilderFactory(coin);\n      const legacyBuilder = factory.getTransferBuilder() as TransferBuilder;\n\n      legacyBuilder\n        .sender({ address: accounts.account1.address })\n        .to({ address: to })\n        .amount(amount)\n        .validity({ firstValid: westendBlock.blockNumber, maxDuration: 2400 })\n        .referenceBlock(westendBlock.hash)\n        .sequenceId({ name: 'Nonce', keyword: 'nonce', value: 5 });\n\n      const legacyTx = await legacyBuilder.build();\n      const legacyHex = legacyTx.toBroadcastFormat();\n\n      const wasmTx = buildTransaction(\n        { type: 'payment', to, amount: BigInt(amount), keepAlive: true },\n        createWasmContext({ nonce: 5 })\n      );\n      const wasmHex = '0x' + Buffer.from(wasmTx.signablePayload()).toString('hex');\n\n      assert.strictEqual(wasmHex, stripCompactPrefix(legacyHex), 'Signing payload should match');\n    });\n  });\n\n  // ===========================================================================\n  // Staking Transaction Tests\n  // ===========================================================================\n  describe('Staking Transactions', function () {\n    // Note: wasm-dot@1.3.0 `stake` without proxyAddress produces bondExtra (top-up),\n    // while the legacy getStakingBuilder() produces bond (initial stake). These are\n    // different extrinsic calls so byte comparison is not applicable. The bond call\n    // is now only produced via `stake` with proxyAddress (which creates a batch).\n    // See Intent-based Transaction Building tests below for stake sanity checks.\n\n    it('should produce identical signing payload for unstake (unbond)', async function () {\n      const amount = '5000000000000'; // 5 DOT\n\n      const factory = new TransactionBuilderFactory(coin);\n      const legacyBuilder = factory.getUnstakeBuilder();\n\n      legacyBuilder\n        .sender({ address: accounts.account1.address })\n        .amount(amount)\n        .validity({ firstValid: westendBlock.blockNumber, maxDuration: 2400 })\n        .referenceBlock(westendBlock.hash)\n        .sequenceId({ name: 'Nonce', keyword: 'nonce', value: 0 });\n\n      const legacyTx = await legacyBuilder.build();\n      const legacyHex = legacyTx.toBroadcastFormat();\n\n      const wasmTx = buildTransaction({ type: 'unstake', amount: BigInt(amount) }, createWasmContext());\n      const wasmHex = '0x' + Buffer.from(wasmTx.signablePayload()).toString('hex');\n\n      assert.strictEqual(wasmHex, stripCompactPrefix(legacyHex), 'Signing payload should match');\n    });\n\n    it('should produce identical signing payload for withdrawUnbonded (claim)', async function () {\n      const slashingSpans = 0;\n\n      const factory = new TransactionBuilderFactory(coin);\n      const legacyBuilder = factory.getWithdrawUnstakedBuilder();\n\n      legacyBuilder\n        .sender({ address: accounts.account1.address })\n        .slashingSpans(slashingSpans)\n        .validity({ firstValid: westendBlock.blockNumber, maxDuration: 2400 })\n        .referenceBlock(westendBlock.hash)\n        .sequenceId({ name: 'Nonce', keyword: 'nonce', value: 0 });\n\n      const legacyTx = await legacyBuilder.build();\n      const legacyHex = legacyTx.toBroadcastFormat();\n\n      const wasmTx = buildTransaction({ type: 'claim', slashingSpans }, createWasmContext());\n      const wasmHex = '0x' + Buffer.from(wasmTx.signablePayload()).toString('hex');\n\n      assert.strictEqual(wasmHex, stripCompactPrefix(legacyHex), 'Signing payload should match');\n    });\n  });\n\n  // ===========================================================================\n  // Intent-based Transaction Building (sanity checks)\n  // ===========================================================================\n  describe('Intent-based Transaction Building', function () {\n    it('should build payment from intent', async function () {\n      const wasmTx = buildTransaction(\n        { type: 'payment', to: accounts.account2.address, amount: 1000000000000n, keepAlive: true },\n        createWasmContext()\n      );\n      const serialized = wasmTx.toBroadcastFormat();\n      assert(serialized.startsWith('0x'), 'Should be hex encoded');\n      assert(serialized.length > 10, 'Should have content');\n    });\n\n    it('should build stake (top-up) from intent', async function () {\n      const wasmTx = buildTransaction(\n        { type: 'stake', amount: 5000000000000n, payee: { type: 'staked' } },\n        createWasmContext()\n      );\n      assert(wasmTx.toBroadcastFormat().startsWith('0x'));\n    });\n\n    it('should build stake with proxy (new stake / batchAll(bond, addProxy)) from intent', async function () {\n      const wasmTx = buildTransaction(\n        { type: 'stake', amount: 10000000000000n, proxyAddress: accounts.account2.address, payee: { type: 'staked' } },\n        createWasmContext()\n      );\n      assert(wasmTx.toBroadcastFormat().startsWith('0x'));\n    });\n\n    it('should build claim (withdrawUnbonded) from intent', async function () {\n      const wasmTx = buildTransaction({ type: 'claim', slashingSpans: 0 }, createWasmContext());\n      assert(wasmTx.toBroadcastFormat().startsWith('0x'));\n    });\n\n    it('should build unstake (partial / unbond) from intent', async function () {\n      const wasmTx = buildTransaction({ type: 'unstake', amount: 5000000000000n }, createWasmContext());\n      assert(wasmTx.toBroadcastFormat().startsWith('0x'));\n    });\n\n    it('should build unstake with stopStaking (full / batchAll(removeProxy, chill, unbond)) from intent', async function () {\n      const wasmTx = buildTransaction(\n        {\n          type: 'unstake',\n          amount: 5000000000000n,\n          stopStaking: true,\n          proxyAddress: accounts.account2.address,\n        },\n        createWasmContext()\n      );\n      assert(wasmTx.toBroadcastFormat().startsWith('0x'));\n    });\n\n    it('should build consolidate (transferAll) from intent', async function () {\n      const wasmTx = buildTransaction({ type: 'consolidate', to: accounts.account2.address }, createWasmContext());\n      assert(wasmTx.toBroadcastFormat().startsWith('0x'));\n    });\n  });\n});\n"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* WASM Parser Explanation Tests
|
|
3
|
+
*
|
|
4
|
+
* Tests for explainDotTransaction, specifically verifying batch transaction
|
|
5
|
+
* handling with proxy deposit costs matches legacy account-lib behavior.
|
|
6
|
+
*
|
|
7
|
+
* Uses WASM-built transactions (not legacy rawTx fixtures) since the WASM
|
|
8
|
+
* parser requires metadata-compatible signed extension encoding.
|
|
9
|
+
*
|
|
10
|
+
* Note: wasm-dot@1.3.0 uses high-level business intents. Batch transactions
|
|
11
|
+
* are produced automatically from the intent (e.g., unstake with stopStaking
|
|
12
|
+
* produces batchAll(removeProxy, chill, unbond)).
|
|
13
|
+
*/
|
|
14
|
+
export {};
|
|
15
|
+
//# sourceMappingURL=wasmParserExplanation.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"wasmParserExplanation.d.ts","sourceRoot":"","sources":["../../../../test/unit/wasmParserExplanation.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG"}
|
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* WASM Parser Explanation Tests
|
|
4
|
+
*
|
|
5
|
+
* Tests for explainDotTransaction, specifically verifying batch transaction
|
|
6
|
+
* handling with proxy deposit costs matches legacy account-lib behavior.
|
|
7
|
+
*
|
|
8
|
+
* Uses WASM-built transactions (not legacy rawTx fixtures) since the WASM
|
|
9
|
+
* parser requires metadata-compatible signed extension encoding.
|
|
10
|
+
*
|
|
11
|
+
* Note: wasm-dot@1.3.0 uses high-level business intents. Batch transactions
|
|
12
|
+
* are produced automatically from the intent (e.g., unstake with stopStaking
|
|
13
|
+
* produces batchAll(removeProxy, chill, unbond)).
|
|
14
|
+
*/
|
|
15
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
16
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
17
|
+
};
|
|
18
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
19
|
+
const assert_1 = __importDefault(require("assert"));
|
|
20
|
+
const statics_1 = require("@bitgo-beta/statics");
|
|
21
|
+
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
22
|
+
const wasmParser_1 = require("../../src/lib/wasmParser");
|
|
23
|
+
const wasm_dot_1 = require("@bitgo/wasm-dot");
|
|
24
|
+
const fixtures_1 = require("../fixtures");
|
|
25
|
+
const utils_1 = __importDefault(require("../../src/lib/utils"));
|
|
26
|
+
describe('WASM Parser Explanation', function () {
|
|
27
|
+
const coin = statics_1.coins.get('tdot');
|
|
28
|
+
// utils.getMaterial returns the iface Material shape; cast to WasmMaterial for buildTransaction
|
|
29
|
+
const material = utils_1.default.getMaterial(coin);
|
|
30
|
+
function createWasmContext(overrides = {}) {
|
|
31
|
+
return {
|
|
32
|
+
sender: fixtures_1.accounts.account1.address,
|
|
33
|
+
nonce: 0,
|
|
34
|
+
tip: 0n,
|
|
35
|
+
material,
|
|
36
|
+
validity: {
|
|
37
|
+
firstValid: fixtures_1.westendBlock.blockNumber,
|
|
38
|
+
maxDuration: 2400,
|
|
39
|
+
},
|
|
40
|
+
referenceBlock: fixtures_1.westendBlock.hash,
|
|
41
|
+
...overrides,
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
describe('Batch unstake (removeProxy + chill + unbond)', function () {
|
|
45
|
+
it('should explain batch unstake with proxy deposit cost', function () {
|
|
46
|
+
const unbondAmount = 5000000000000n; // 5 DOT
|
|
47
|
+
const proxyDelegate = fixtures_1.accounts.account2.address;
|
|
48
|
+
// Build a full unstake: produces batchAll(removeProxy, chill, unbond)
|
|
49
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({
|
|
50
|
+
type: 'unstake',
|
|
51
|
+
amount: unbondAmount,
|
|
52
|
+
stopStaking: true,
|
|
53
|
+
proxyAddress: proxyDelegate,
|
|
54
|
+
}, createWasmContext());
|
|
55
|
+
const txHex = wasmTx.toBroadcastFormat();
|
|
56
|
+
const explanation = (0, wasmParser_1.explainDotTransaction)({
|
|
57
|
+
txHex,
|
|
58
|
+
material,
|
|
59
|
+
senderAddress: fixtures_1.accounts.account1.address,
|
|
60
|
+
});
|
|
61
|
+
// Should be Batch type
|
|
62
|
+
assert_1.default.strictEqual(explanation.type, sdk_core_1.TransactionType.Batch);
|
|
63
|
+
assert_1.default.ok(explanation.methodName.includes('batchAll'), `Expected batchAll, got ${explanation.methodName}`);
|
|
64
|
+
// Outputs should contain proxy deposit cost, NOT the unbond amount
|
|
65
|
+
assert_1.default.strictEqual(explanation.outputs.length, 1, 'Should have exactly one output (proxy deposit cost)');
|
|
66
|
+
const output = explanation.outputs[0];
|
|
67
|
+
assert_1.default.strictEqual(output.address, fixtures_1.accounts.account1.address, 'Output should go to sender (deposit refund)');
|
|
68
|
+
const proxyDepositCost = BigInt(output.amount);
|
|
69
|
+
assert_1.default.ok(proxyDepositCost > 0n, 'Proxy deposit cost should be positive');
|
|
70
|
+
// The proxy deposit cost should NOT equal the unbond amount
|
|
71
|
+
assert_1.default.notStrictEqual(proxyDepositCost, unbondAmount, 'Should use proxy deposit cost, not unbond amount');
|
|
72
|
+
// Input should come from the proxy delegate address
|
|
73
|
+
assert_1.default.strictEqual(explanation.inputs.length, 1, 'Should have exactly one input');
|
|
74
|
+
assert_1.default.strictEqual(explanation.inputs[0].address, proxyDelegate, 'Input should come from proxy delegate');
|
|
75
|
+
assert_1.default.strictEqual(explanation.inputs[0].valueString, output.amount, 'Input value should equal proxy deposit cost');
|
|
76
|
+
});
|
|
77
|
+
it('proxy deposit cost should be consistent across calls', function () {
|
|
78
|
+
const proxyDelegate = fixtures_1.accounts.account2.address;
|
|
79
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({
|
|
80
|
+
type: 'unstake',
|
|
81
|
+
amount: 1000000000000n,
|
|
82
|
+
stopStaking: true,
|
|
83
|
+
proxyAddress: proxyDelegate,
|
|
84
|
+
}, createWasmContext());
|
|
85
|
+
const txHex = wasmTx.toBroadcastFormat();
|
|
86
|
+
const explanation1 = (0, wasmParser_1.explainDotTransaction)({ txHex, material, senderAddress: fixtures_1.accounts.account1.address });
|
|
87
|
+
const explanation2 = (0, wasmParser_1.explainDotTransaction)({ txHex, material, senderAddress: fixtures_1.accounts.account1.address });
|
|
88
|
+
assert_1.default.strictEqual(explanation1.outputs[0].amount, explanation2.outputs[0].amount);
|
|
89
|
+
});
|
|
90
|
+
});
|
|
91
|
+
describe('Batch stake (bond + addProxy)', function () {
|
|
92
|
+
it('should explain batch stake with bond amount and proxy deposit cost', function () {
|
|
93
|
+
const bondAmount = 10000000000000n; // 10 DOT
|
|
94
|
+
const proxyDelegate = fixtures_1.accounts.account2.address;
|
|
95
|
+
// Build a new stake with proxy: produces batchAll(bond, addProxy)
|
|
96
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({
|
|
97
|
+
type: 'stake',
|
|
98
|
+
amount: bondAmount,
|
|
99
|
+
proxyAddress: proxyDelegate,
|
|
100
|
+
payee: { type: 'staked' },
|
|
101
|
+
}, createWasmContext());
|
|
102
|
+
const txHex = wasmTx.toBroadcastFormat();
|
|
103
|
+
const explanation = (0, wasmParser_1.explainDotTransaction)({ txHex, material, senderAddress: fixtures_1.accounts.account1.address });
|
|
104
|
+
assert_1.default.strictEqual(explanation.type, sdk_core_1.TransactionType.Batch);
|
|
105
|
+
// Should have two outputs: bond amount (to STAKING_DESTINATION sentinel) + proxy deposit cost (to proxy delegate)
|
|
106
|
+
assert_1.default.strictEqual(explanation.outputs.length, 2, 'Should have bond + proxy deposit outputs');
|
|
107
|
+
const stakingOutput = explanation.outputs.find((o) => o.address === '5C4hrfjw9DjXZTzV3MwzrrAr9P1MJhSrvWGWqi1eSuyUpnhM');
|
|
108
|
+
assert_1.default.ok(stakingOutput, 'Should have STAKING_DESTINATION sentinel output for bond amount');
|
|
109
|
+
assert_1.default.strictEqual(BigInt(stakingOutput.amount), bondAmount, 'Bond amount should match');
|
|
110
|
+
const proxyOutput = explanation.outputs.find((o) => o.address !== '5C4hrfjw9DjXZTzV3MwzrrAr9P1MJhSrvWGWqi1eSuyUpnhM');
|
|
111
|
+
assert_1.default.ok(proxyOutput, 'Should have proxy deposit output');
|
|
112
|
+
assert_1.default.strictEqual(proxyOutput.address, proxyDelegate);
|
|
113
|
+
assert_1.default.ok(BigInt(proxyOutput.amount) > 0n, 'Proxy deposit cost should be positive');
|
|
114
|
+
// All inputs should come from sender
|
|
115
|
+
assert_1.default.strictEqual(explanation.inputs.length, 2);
|
|
116
|
+
for (const input of explanation.inputs) {
|
|
117
|
+
assert_1.default.strictEqual(input.address, fixtures_1.accounts.account1.address);
|
|
118
|
+
}
|
|
119
|
+
});
|
|
120
|
+
});
|
|
121
|
+
describe('Non-batch transactions (should not be affected)', function () {
|
|
122
|
+
it('should explain transfer normally', function () {
|
|
123
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'payment', to: fixtures_1.accounts.account2.address, amount: 1000000000000n, keepAlive: true }, createWasmContext());
|
|
124
|
+
const explanation = (0, wasmParser_1.explainDotTransaction)({
|
|
125
|
+
txHex: wasmTx.toBroadcastFormat(),
|
|
126
|
+
material,
|
|
127
|
+
senderAddress: fixtures_1.accounts.account1.address,
|
|
128
|
+
});
|
|
129
|
+
assert_1.default.strictEqual(explanation.type, sdk_core_1.TransactionType.Send);
|
|
130
|
+
assert_1.default.strictEqual(explanation.outputs.length, 1);
|
|
131
|
+
assert_1.default.strictEqual(explanation.outputs[0].address, fixtures_1.accounts.account2.address);
|
|
132
|
+
assert_1.default.strictEqual(explanation.outputs[0].amount, '1000000000000');
|
|
133
|
+
});
|
|
134
|
+
it('should explain single unstake (unbond) normally', function () {
|
|
135
|
+
const wasmTx = (0, wasm_dot_1.buildTransaction)({ type: 'unstake', amount: 5000000000000n }, createWasmContext());
|
|
136
|
+
const explanation = (0, wasmParser_1.explainDotTransaction)({
|
|
137
|
+
txHex: wasmTx.toBroadcastFormat(),
|
|
138
|
+
material,
|
|
139
|
+
senderAddress: fixtures_1.accounts.account1.address,
|
|
140
|
+
});
|
|
141
|
+
assert_1.default.strictEqual(explanation.type, sdk_core_1.TransactionType.StakingUnlock);
|
|
142
|
+
// Unbond has no outputs (balance stays in account as unlocking, matching legacy behavior)
|
|
143
|
+
assert_1.default.strictEqual(explanation.outputs.length, 0);
|
|
144
|
+
});
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"wasmParserExplanation.js","sourceRoot":"","sources":["../../../../test/unit/wasmParserExplanation.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * WASM Parser Explanation Tests\n *\n * Tests for explainDotTransaction, specifically verifying batch transaction\n * handling with proxy deposit costs matches legacy account-lib behavior.\n *\n * Uses WASM-built transactions (not legacy rawTx fixtures) since the WASM\n * parser requires metadata-compatible signed extension encoding.\n *\n * Note: wasm-dot@1.3.0 uses high-level business intents. Batch transactions\n * are produced automatically from the intent (e.g., unstake with stopStaking\n * produces batchAll(removeProxy, chill, unbond)).\n */\n\nimport assert from 'assert';\nimport { coins } from '@bitgo-beta/statics';\nimport { TransactionType } from '@bitgo-beta/sdk-core';\nimport { explainDotTransaction } from '../../src/lib/wasmParser';\nimport { buildTransaction, type BuildContext, type Material as WasmMaterial } from '@bitgo/wasm-dot';\nimport type { Material } from '../../src/lib/iface';\nimport { accounts, westendBlock } from '../fixtures';\nimport utils from '../../src/lib/utils';\n\ndescribe('WASM Parser Explanation', function () {\n  const coin = coins.get('tdot');\n  // utils.getMaterial returns the iface Material shape; cast to WasmMaterial for buildTransaction\n  const material = utils.getMaterial(coin) as Material & WasmMaterial;\n\n  function createWasmContext(overrides: Partial<BuildContext> = {}): BuildContext {\n    return {\n      sender: accounts.account1.address,\n      nonce: 0,\n      tip: 0n,\n      material,\n      validity: {\n        firstValid: westendBlock.blockNumber,\n        maxDuration: 2400,\n      },\n      referenceBlock: westendBlock.hash,\n      ...overrides,\n    };\n  }\n\n  describe('Batch unstake (removeProxy + chill + unbond)', function () {\n    it('should explain batch unstake with proxy deposit cost', function () {\n      const unbondAmount = 5000000000000n; // 5 DOT\n      const proxyDelegate = accounts.account2.address;\n\n      // Build a full unstake: produces batchAll(removeProxy, chill, unbond)\n      const wasmTx = buildTransaction(\n        {\n          type: 'unstake',\n          amount: unbondAmount,\n          stopStaking: true,\n          proxyAddress: proxyDelegate,\n        },\n        createWasmContext()\n      );\n\n      const txHex = wasmTx.toBroadcastFormat();\n      const explanation = explainDotTransaction({\n        txHex,\n        material,\n        senderAddress: accounts.account1.address,\n      });\n\n      // Should be Batch type\n      assert.strictEqual(explanation.type, TransactionType.Batch);\n      assert.ok(explanation.methodName.includes('batchAll'), `Expected batchAll, got ${explanation.methodName}`);\n\n      // Outputs should contain proxy deposit cost, NOT the unbond amount\n      assert.strictEqual(explanation.outputs.length, 1, 'Should have exactly one output (proxy deposit cost)');\n      const output = explanation.outputs[0];\n      assert.strictEqual(output.address, accounts.account1.address, 'Output should go to sender (deposit refund)');\n      const proxyDepositCost = BigInt(output.amount);\n      assert.ok(proxyDepositCost > 0n, 'Proxy deposit cost should be positive');\n      // The proxy deposit cost should NOT equal the unbond amount\n      assert.notStrictEqual(proxyDepositCost, unbondAmount, 'Should use proxy deposit cost, not unbond amount');\n\n      // Input should come from the proxy delegate address\n      assert.strictEqual(explanation.inputs.length, 1, 'Should have exactly one input');\n      assert.strictEqual(explanation.inputs[0].address, proxyDelegate, 'Input should come from proxy delegate');\n      assert.strictEqual(\n        explanation.inputs[0].valueString,\n        output.amount,\n        'Input value should equal proxy deposit cost'\n      );\n    });\n\n    it('proxy deposit cost should be consistent across calls', function () {\n      const proxyDelegate = accounts.account2.address;\n      const wasmTx = buildTransaction(\n        {\n          type: 'unstake',\n          amount: 1000000000000n,\n          stopStaking: true,\n          proxyAddress: proxyDelegate,\n        },\n        createWasmContext()\n      );\n\n      const txHex = wasmTx.toBroadcastFormat();\n      const explanation1 = explainDotTransaction({ txHex, material, senderAddress: accounts.account1.address });\n      const explanation2 = explainDotTransaction({ txHex, material, senderAddress: accounts.account1.address });\n\n      assert.strictEqual(explanation1.outputs[0].amount, explanation2.outputs[0].amount);\n    });\n  });\n\n  describe('Batch stake (bond + addProxy)', function () {\n    it('should explain batch stake with bond amount and proxy deposit cost', function () {\n      const bondAmount = 10000000000000n; // 10 DOT\n      const proxyDelegate = accounts.account2.address;\n\n      // Build a new stake with proxy: produces batchAll(bond, addProxy)\n      const wasmTx = buildTransaction(\n        {\n          type: 'stake',\n          amount: bondAmount,\n          proxyAddress: proxyDelegate,\n          payee: { type: 'staked' },\n        },\n        createWasmContext()\n      );\n\n      const txHex = wasmTx.toBroadcastFormat();\n      const explanation = explainDotTransaction({ txHex, material, senderAddress: accounts.account1.address });\n\n      assert.strictEqual(explanation.type, TransactionType.Batch);\n\n      // Should have two outputs: bond amount (to STAKING_DESTINATION sentinel) + proxy deposit cost (to proxy delegate)\n      assert.strictEqual(explanation.outputs.length, 2, 'Should have bond + proxy deposit outputs');\n\n      const stakingOutput = explanation.outputs.find(\n        (o) => o.address === '5C4hrfjw9DjXZTzV3MwzrrAr9P1MJhSrvWGWqi1eSuyUpnhM'\n      );\n      assert.ok(stakingOutput, 'Should have STAKING_DESTINATION sentinel output for bond amount');\n      assert.strictEqual(BigInt(stakingOutput!.amount), bondAmount, 'Bond amount should match');\n\n      const proxyOutput = explanation.outputs.find(\n        (o) => o.address !== '5C4hrfjw9DjXZTzV3MwzrrAr9P1MJhSrvWGWqi1eSuyUpnhM'\n      );\n      assert.ok(proxyOutput, 'Should have proxy deposit output');\n      assert.strictEqual(proxyOutput!.address, proxyDelegate);\n      assert.ok(BigInt(proxyOutput!.amount) > 0n, 'Proxy deposit cost should be positive');\n\n      // All inputs should come from sender\n      assert.strictEqual(explanation.inputs.length, 2);\n      for (const input of explanation.inputs) {\n        assert.strictEqual(input.address, accounts.account1.address);\n      }\n    });\n  });\n\n  describe('Non-batch transactions (should not be affected)', function () {\n    it('should explain transfer normally', function () {\n      const wasmTx = buildTransaction(\n        { type: 'payment', to: accounts.account2.address, amount: 1000000000000n, keepAlive: true },\n        createWasmContext()\n      );\n\n      const explanation = explainDotTransaction({\n        txHex: wasmTx.toBroadcastFormat(),\n        material,\n        senderAddress: accounts.account1.address,\n      });\n\n      assert.strictEqual(explanation.type, TransactionType.Send);\n      assert.strictEqual(explanation.outputs.length, 1);\n      assert.strictEqual(explanation.outputs[0].address, accounts.account2.address);\n      assert.strictEqual(explanation.outputs[0].amount, '1000000000000');\n    });\n\n    it('should explain single unstake (unbond) normally', function () {\n      const wasmTx = buildTransaction({ type: 'unstake', amount: 5000000000000n }, createWasmContext());\n\n      const explanation = explainDotTransaction({\n        txHex: wasmTx.toBroadcastFormat(),\n        material,\n        senderAddress: accounts.account1.address,\n      });\n\n      assert.strictEqual(explanation.type, TransactionType.StakingUnlock);\n      // Unbond has no outputs (balance stays in account as unlocking, matching legacy behavior)\n      assert.strictEqual(explanation.outputs.length, 0);\n    });\n  });\n});\n"]}
|