@bitgo/sdk-coin-ton 3.17.1 → 3.19.0

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.
@@ -0,0 +1,176 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ var __importDefault = (this && this.__importDefault) || function (mod) {
36
+ return (mod && mod.__esModule) ? mod : { "default": mod };
37
+ };
38
+ Object.defineProperty(exports, "__esModule", { value: true });
39
+ const should_1 = __importDefault(require("should"));
40
+ const statics_1 = require("@bitgo/statics");
41
+ const sdk_core_1 = require("@bitgo/sdk-core");
42
+ const wasm_ton_1 = require("@bitgo/wasm-ton");
43
+ const transactionBuilderFactory_1 = require("../../src/lib/transactionBuilderFactory");
44
+ const testData = __importStar(require("../resources/ton"));
45
+ /**
46
+ * Cross-compatibility tests between WASM (@bitgo/wasm-ton) and legacy
47
+ * (tonweb-based) transaction building/parsing for TON.
48
+ *
49
+ * Direction 1: WASM parses legacy-built transactions (covered by explainTransactionWasm.ts)
50
+ * Direction 2: Legacy parses WASM-built transactions (this file)
51
+ */
52
+ describe('TON WASM Cross-Compatibility', function () {
53
+ const coin = statics_1.coins.get('tton');
54
+ const factory = new transactionBuilderFactory_1.TransactionBuilderFactory(coin);
55
+ // Use sender from test fixtures
56
+ const senderAddress = testData.sender.address;
57
+ const senderPublicKey = testData.sender.publicKey;
58
+ const recipientAddress = testData.addresses.validAddresses[0];
59
+ function createWasmContext(overrides = {}) {
60
+ return {
61
+ sender: senderAddress,
62
+ seqno: 6,
63
+ expireTime: 1695997582n,
64
+ publicKey: senderPublicKey,
65
+ ...overrides,
66
+ };
67
+ }
68
+ // =========================================================================
69
+ // WASM -> Legacy: Legacy can parse WASM-built transactions
70
+ // =========================================================================
71
+ describe('Legacy parses WASM-built transactions', function () {
72
+ it('should parse a WASM-built payment transaction via legacy factory.from()', async function () {
73
+ const amount = 10000000n; // 0.01 TON
74
+ const memo = 'test';
75
+ // Build with WASM
76
+ const wasmTx = (0, wasm_ton_1.buildTransaction)({
77
+ type: 'payment',
78
+ to: recipientAddress,
79
+ amount,
80
+ bounceable: false,
81
+ memo,
82
+ }, createWasmContext());
83
+ // Convert to base64 (the format legacy expects)
84
+ const bocBytes = wasmTx.toBroadcastFormat();
85
+ const base64Tx = Buffer.from(bocBytes).toString('base64');
86
+ // Parse with legacy
87
+ const legacyBuilder = factory.from(base64Tx);
88
+ const legacyTx = await legacyBuilder.build();
89
+ const json = legacyTx.toJson();
90
+ // Verify the legacy builder can extract the correct fields
91
+ legacyTx.type.should.equal(sdk_core_1.TransactionType.Send);
92
+ json.seqno.should.equal(6);
93
+ json.expirationTime.should.equal(1695997582);
94
+ json.amount.should.equal(amount.toString());
95
+ should_1.default.exist(json.sender);
96
+ should_1.default.exist(json.destination);
97
+ });
98
+ it('should parse a WASM-built payment without memo via legacy', async function () {
99
+ const amount = 50000000n; // 0.05 TON
100
+ const wasmTx = (0, wasm_ton_1.buildTransaction)({
101
+ type: 'payment',
102
+ to: recipientAddress,
103
+ amount,
104
+ bounceable: false,
105
+ }, createWasmContext({ seqno: 10, expireTime: 1700000000n }));
106
+ const base64Tx = Buffer.from(wasmTx.toBroadcastFormat()).toString('base64');
107
+ const legacyBuilder = factory.from(base64Tx);
108
+ const legacyTx = await legacyBuilder.build();
109
+ const json = legacyTx.toJson();
110
+ legacyTx.type.should.equal(sdk_core_1.TransactionType.Send);
111
+ json.seqno.should.equal(10);
112
+ json.expirationTime.should.equal(1700000000);
113
+ json.amount.should.equal(amount.toString());
114
+ });
115
+ it('should round-trip: WASM build -> legacy parse -> legacy build -> WASM parse', async function () {
116
+ const amount = 123400000n;
117
+ const memo = 'hello';
118
+ const wasmTx = (0, wasm_ton_1.buildTransaction)({
119
+ type: 'payment',
120
+ to: recipientAddress,
121
+ amount,
122
+ bounceable: false,
123
+ memo,
124
+ }, createWasmContext({ seqno: 3, expireTime: 1234567890n }));
125
+ const base64Tx = Buffer.from(wasmTx.toBroadcastFormat()).toString('base64');
126
+ // Parse with legacy
127
+ const legacyBuilder = factory.from(base64Tx);
128
+ const legacyTx = await legacyBuilder.build();
129
+ // Rebuild with legacy should produce valid base64
130
+ const rebuiltBase64 = legacyTx.toBroadcastFormat();
131
+ // Parse the rebuilt transaction with WASM to confirm fields match
132
+ const wasmParsed = (0, wasm_ton_1.parseTransaction)(wasm_ton_1.Transaction.fromBytes(Buffer.from(rebuiltBase64, 'base64')));
133
+ wasmParsed.transactionType.should.equal('Transfer');
134
+ wasmParsed.seqno.should.equal(3);
135
+ wasmParsed.sendActions.length.should.be.greaterThan(0);
136
+ String(wasmParsed.sendActions[0].amount).should.equal(amount.toString());
137
+ });
138
+ it('should parse a WASM-built bounceable payment via legacy', async function () {
139
+ const amount = 10000000n;
140
+ const wasmTx = (0, wasm_ton_1.buildTransaction)({
141
+ type: 'payment',
142
+ to: recipientAddress,
143
+ amount,
144
+ bounceable: true,
145
+ }, createWasmContext());
146
+ const base64Tx = Buffer.from(wasmTx.toBroadcastFormat()).toString('base64');
147
+ const legacyBuilder = factory.from(base64Tx);
148
+ const legacyTx = await legacyBuilder.build();
149
+ const json = legacyTx.toJson();
150
+ legacyTx.type.should.equal(sdk_core_1.TransactionType.Send);
151
+ json.amount.should.equal(amount.toString());
152
+ });
153
+ });
154
+ // =========================================================================
155
+ // Both WASM and legacy agree on signed fixture data
156
+ // =========================================================================
157
+ describe('Both WASM and legacy agree on signed fixture data', function () {
158
+ it('should produce matching fields for signedSendTransaction', async function () {
159
+ const txBase64 = testData.signedSendTransaction.tx;
160
+ // Parse with WASM
161
+ const wasmTx = wasm_ton_1.Transaction.fromBytes(Buffer.from(txBase64, 'base64'));
162
+ const wasmParsed = (0, wasm_ton_1.parseTransaction)(wasmTx);
163
+ // Parse with legacy
164
+ const legacyBuilder = factory.from(txBase64);
165
+ const legacyTx = await legacyBuilder.build();
166
+ const json = legacyTx.toJson();
167
+ // Both should agree on core fields
168
+ wasmParsed.transactionType.should.equal('Transfer');
169
+ legacyTx.type.should.equal(sdk_core_1.TransactionType.Send);
170
+ wasmParsed.seqno.should.equal(json.seqno);
171
+ wasmParsed.sendActions.length.should.be.greaterThan(0);
172
+ String(wasmParsed.sendActions[0].amount).should.equal(json.amount);
173
+ });
174
+ });
175
+ });
176
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"wasmCrossCompatibility.js","sourceRoot":"","sources":["../../../test/unit/wasmCrossCompatibility.ts"],"names":[],"mappings":"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","sourcesContent":["import should from 'should';\nimport { coins } from '@bitgo/statics';\nimport { TransactionType } from '@bitgo/sdk-core';\nimport {\n  buildTransaction,\n  parseTransaction,\n  Transaction as WasmTonTransaction,\n  type BuildContext,\n  type PaymentIntent,\n  type ParsedTransaction,\n} from '@bitgo/wasm-ton';\nimport { TransactionBuilderFactory } from '../../src/lib/transactionBuilderFactory';\nimport * as testData from '../resources/ton';\n\n/**\n * Cross-compatibility tests between WASM (@bitgo/wasm-ton) and legacy\n * (tonweb-based) transaction building/parsing for TON.\n *\n * Direction 1: WASM parses legacy-built transactions (covered by explainTransactionWasm.ts)\n * Direction 2: Legacy parses WASM-built transactions (this file)\n */\ndescribe('TON WASM Cross-Compatibility', function () {\n  const coin = coins.get('tton');\n  const factory = new TransactionBuilderFactory(coin);\n\n  // Use sender from test fixtures\n  const senderAddress = testData.sender.address;\n  const senderPublicKey = testData.sender.publicKey;\n  const recipientAddress = testData.addresses.validAddresses[0];\n\n  function createWasmContext(overrides: Partial<BuildContext> = {}): BuildContext {\n    return {\n      sender: senderAddress,\n      seqno: 6,\n      expireTime: 1695997582n,\n      publicKey: senderPublicKey,\n      ...overrides,\n    };\n  }\n\n  // =========================================================================\n  // WASM -> Legacy: Legacy can parse WASM-built transactions\n  // =========================================================================\n  describe('Legacy parses WASM-built transactions', function () {\n    it('should parse a WASM-built payment transaction via legacy factory.from()', async function () {\n      const amount = 10000000n; // 0.01 TON\n      const memo = 'test';\n\n      // Build with WASM\n      const wasmTx = buildTransaction(\n        {\n          type: 'payment',\n          to: recipientAddress,\n          amount,\n          bounceable: false,\n          memo,\n        } as PaymentIntent,\n        createWasmContext()\n      );\n\n      // Convert to base64 (the format legacy expects)\n      const bocBytes = wasmTx.toBroadcastFormat();\n      const base64Tx = Buffer.from(bocBytes).toString('base64');\n\n      // Parse with legacy\n      const legacyBuilder = factory.from(base64Tx);\n      const legacyTx = await legacyBuilder.build();\n      const json = legacyTx.toJson();\n\n      // Verify the legacy builder can extract the correct fields\n      legacyTx.type.should.equal(TransactionType.Send);\n      json.seqno.should.equal(6);\n      json.expirationTime.should.equal(1695997582);\n      json.amount.should.equal(amount.toString());\n      should.exist(json.sender);\n      should.exist(json.destination);\n    });\n\n    it('should parse a WASM-built payment without memo via legacy', async function () {\n      const amount = 50000000n; // 0.05 TON\n\n      const wasmTx = buildTransaction(\n        {\n          type: 'payment',\n          to: recipientAddress,\n          amount,\n          bounceable: false,\n        } as PaymentIntent,\n        createWasmContext({ seqno: 10, expireTime: 1700000000n })\n      );\n\n      const base64Tx = Buffer.from(wasmTx.toBroadcastFormat()).toString('base64');\n\n      const legacyBuilder = factory.from(base64Tx);\n      const legacyTx = await legacyBuilder.build();\n      const json = legacyTx.toJson();\n\n      legacyTx.type.should.equal(TransactionType.Send);\n      json.seqno.should.equal(10);\n      json.expirationTime.should.equal(1700000000);\n      json.amount.should.equal(amount.toString());\n    });\n\n    it('should round-trip: WASM build -> legacy parse -> legacy build -> WASM parse', async function () {\n      const amount = 123400000n;\n      const memo = 'hello';\n\n      const wasmTx = buildTransaction(\n        {\n          type: 'payment',\n          to: recipientAddress,\n          amount,\n          bounceable: false,\n          memo,\n        } as PaymentIntent,\n        createWasmContext({ seqno: 3, expireTime: 1234567890n })\n      );\n\n      const base64Tx = Buffer.from(wasmTx.toBroadcastFormat()).toString('base64');\n\n      // Parse with legacy\n      const legacyBuilder = factory.from(base64Tx);\n      const legacyTx = await legacyBuilder.build();\n\n      // Rebuild with legacy should produce valid base64\n      const rebuiltBase64 = legacyTx.toBroadcastFormat();\n\n      // Parse the rebuilt transaction with WASM to confirm fields match\n      const wasmParsed: ParsedTransaction = parseTransaction(\n        WasmTonTransaction.fromBytes(Buffer.from(rebuiltBase64, 'base64'))\n      );\n\n      wasmParsed.transactionType.should.equal('Transfer');\n      wasmParsed.seqno.should.equal(3);\n      wasmParsed.sendActions.length.should.be.greaterThan(0);\n      String(wasmParsed.sendActions[0].amount).should.equal(amount.toString());\n    });\n\n    it('should parse a WASM-built bounceable payment via legacy', async function () {\n      const amount = 10000000n;\n\n      const wasmTx = buildTransaction(\n        {\n          type: 'payment',\n          to: recipientAddress,\n          amount,\n          bounceable: true,\n        } as PaymentIntent,\n        createWasmContext()\n      );\n\n      const base64Tx = Buffer.from(wasmTx.toBroadcastFormat()).toString('base64');\n\n      const legacyBuilder = factory.from(base64Tx);\n      const legacyTx = await legacyBuilder.build();\n      const json = legacyTx.toJson();\n\n      legacyTx.type.should.equal(TransactionType.Send);\n      json.amount.should.equal(amount.toString());\n    });\n  });\n\n  // =========================================================================\n  // Both WASM and legacy agree on signed fixture data\n  // =========================================================================\n  describe('Both WASM and legacy agree on signed fixture data', function () {\n    it('should produce matching fields for signedSendTransaction', async function () {\n      const txBase64 = testData.signedSendTransaction.tx;\n\n      // Parse with WASM\n      const wasmTx = WasmTonTransaction.fromBytes(Buffer.from(txBase64, 'base64'));\n      const wasmParsed: ParsedTransaction = parseTransaction(wasmTx);\n\n      // Parse with legacy\n      const legacyBuilder = factory.from(txBase64);\n      const legacyTx = await legacyBuilder.build();\n      const json = legacyTx.toJson();\n\n      // Both should agree on core fields\n      wasmParsed.transactionType.should.equal('Transfer');\n      legacyTx.type.should.equal(TransactionType.Send);\n      wasmParsed.seqno.should.equal(json.seqno);\n      wasmParsed.sendActions.length.should.be.greaterThan(0);\n      String(wasmParsed.sendActions[0].amount).should.equal(json.amount);\n    });\n  });\n});\n"]}