@bitgo-beta/sdk-coin-flrp 1.0.1-beta.36 → 1.0.1-beta.361

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (119) hide show
  1. package/dist/src/flrp.d.ts +10 -17
  2. package/dist/src/flrp.d.ts.map +1 -1
  3. package/dist/src/flrp.js +51 -77
  4. package/dist/src/index.d.ts +0 -1
  5. package/dist/src/index.d.ts.map +1 -1
  6. package/dist/src/index.js +1 -2
  7. package/dist/src/lib/ExportInCTxBuilder.d.ts +51 -0
  8. package/dist/src/lib/ExportInCTxBuilder.d.ts.map +1 -0
  9. package/dist/src/lib/ExportInCTxBuilder.js +190 -0
  10. package/dist/src/lib/ExportInPTxBuilder.d.ts +47 -0
  11. package/dist/src/lib/ExportInPTxBuilder.d.ts.map +1 -0
  12. package/dist/src/lib/ExportInPTxBuilder.js +277 -0
  13. package/dist/src/lib/ImportInCTxBuilder.d.ts +48 -0
  14. package/dist/src/lib/ImportInCTxBuilder.d.ts.map +1 -0
  15. package/dist/src/lib/ImportInCTxBuilder.js +268 -0
  16. package/dist/src/lib/ImportInPTxBuilder.d.ts +33 -0
  17. package/dist/src/lib/ImportInPTxBuilder.d.ts.map +1 -0
  18. package/dist/src/lib/ImportInPTxBuilder.js +192 -0
  19. package/dist/src/lib/atomicInCTransactionBuilder.d.ts +18 -16
  20. package/dist/src/lib/atomicInCTransactionBuilder.d.ts.map +1 -1
  21. package/dist/src/lib/atomicInCTransactionBuilder.js +38 -36
  22. package/dist/src/lib/atomicTransactionBuilder.d.ts +57 -71
  23. package/dist/src/lib/atomicTransactionBuilder.d.ts.map +1 -1
  24. package/dist/src/lib/atomicTransactionBuilder.js +246 -209
  25. package/dist/src/lib/iface.d.ts +38 -61
  26. package/dist/src/lib/iface.d.ts.map +1 -1
  27. package/dist/src/lib/iface.js +13 -14
  28. package/dist/src/lib/index.d.ts +5 -0
  29. package/dist/src/lib/index.d.ts.map +1 -1
  30. package/dist/src/lib/index.js +12 -2
  31. package/dist/src/lib/keyPair.d.ts +5 -5
  32. package/dist/src/lib/keyPair.d.ts.map +1 -1
  33. package/dist/src/lib/keyPair.js +17 -9
  34. package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts +43 -0
  35. package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts.map +1 -0
  36. package/dist/src/lib/permissionlessValidatorTxBuilder.js +132 -0
  37. package/dist/src/lib/transaction.d.ts +25 -65
  38. package/dist/src/lib/transaction.d.ts.map +1 -1
  39. package/dist/src/lib/transaction.js +341 -199
  40. package/dist/src/lib/transactionBuilder.d.ts +107 -0
  41. package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
  42. package/dist/src/lib/transactionBuilder.js +210 -0
  43. package/dist/src/lib/transactionBuilderFactory.d.ts +50 -30
  44. package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
  45. package/dist/src/lib/transactionBuilderFactory.js +129 -72
  46. package/dist/src/lib/utils.d.ts +78 -147
  47. package/dist/src/lib/utils.d.ts.map +1 -1
  48. package/dist/src/lib/utils.js +238 -324
  49. package/dist/test/resources/account.d.ts +51 -0
  50. package/dist/test/resources/account.d.ts.map +1 -0
  51. package/dist/test/resources/account.js +54 -0
  52. package/dist/test/resources/transactionData/exportInC.d.ts +20 -0
  53. package/dist/test/resources/transactionData/exportInC.d.ts.map +1 -0
  54. package/dist/test/resources/transactionData/exportInC.js +39 -0
  55. package/dist/test/resources/transactionData/exportInP.d.ts +69 -0
  56. package/dist/test/resources/transactionData/exportInP.d.ts.map +1 -0
  57. package/dist/test/resources/transactionData/exportInP.js +140 -0
  58. package/dist/test/resources/transactionData/importInC.d.ts +27 -0
  59. package/dist/test/resources/transactionData/importInC.d.ts.map +1 -0
  60. package/dist/test/resources/transactionData/importInC.js +44 -0
  61. package/dist/test/resources/transactionData/importInP.d.ts +35 -0
  62. package/dist/test/resources/transactionData/importInP.d.ts.map +1 -0
  63. package/dist/test/resources/transactionData/importInP.js +58 -0
  64. package/dist/test/unit/flrp.js +446 -68
  65. package/dist/test/unit/lib/exportInCTxBuilder.d.ts +2 -0
  66. package/dist/test/unit/lib/exportInCTxBuilder.d.ts.map +1 -0
  67. package/dist/test/unit/lib/exportInCTxBuilder.js +192 -0
  68. package/dist/test/unit/lib/exportInPTxBuilder.d.ts +2 -0
  69. package/dist/test/unit/lib/exportInPTxBuilder.d.ts.map +1 -0
  70. package/dist/test/unit/lib/exportInPTxBuilder.js +325 -0
  71. package/dist/test/unit/lib/importInCTxBuilder.d.ts +2 -0
  72. package/dist/test/unit/lib/importInCTxBuilder.d.ts.map +1 -0
  73. package/dist/test/unit/lib/importInCTxBuilder.js +400 -0
  74. package/dist/test/unit/lib/importInPTxBuilder.d.ts +2 -0
  75. package/dist/test/unit/lib/importInPTxBuilder.d.ts.map +1 -0
  76. package/dist/test/unit/lib/importInPTxBuilder.js +307 -0
  77. package/dist/test/unit/lib/keyPair.d.ts +2 -0
  78. package/dist/test/unit/lib/keyPair.d.ts.map +1 -0
  79. package/dist/test/unit/lib/keyPair.js +158 -0
  80. package/dist/test/unit/lib/signFlowTestSuit.d.ts +20 -0
  81. package/dist/test/unit/lib/signFlowTestSuit.d.ts.map +1 -0
  82. package/dist/test/unit/lib/signFlowTestSuit.js +83 -0
  83. package/dist/test/unit/lib/transactionBuilderFactory.d.ts +2 -0
  84. package/dist/test/unit/lib/transactionBuilderFactory.d.ts.map +1 -0
  85. package/dist/test/unit/lib/transactionBuilderFactory.js +60 -0
  86. package/dist/test/unit/lib/utils.js +539 -207
  87. package/dist/tsconfig.tsbuildinfo +1 -1
  88. package/package.json +18 -10
  89. package/.eslintignore +0 -5
  90. package/.eslintrc.json +0 -7
  91. package/.mocharc.yml +0 -8
  92. package/CHANGELOG.md +0 -0
  93. package/dist/src/iface.d.ts +0 -25
  94. package/dist/src/iface.d.ts.map +0 -1
  95. package/dist/src/iface.js +0 -3
  96. package/dist/src/lib/constants.d.ts +0 -11
  97. package/dist/src/lib/constants.d.ts.map +0 -1
  98. package/dist/src/lib/constants.js +0 -17
  99. package/dist/src/lib/errors.d.ts +0 -8
  100. package/dist/src/lib/errors.d.ts.map +0 -1
  101. package/dist/src/lib/errors.js +0 -19
  102. package/dist/src/lib/exportInCTxBuilder.d.ts +0 -77
  103. package/dist/src/lib/exportInCTxBuilder.d.ts.map +0 -1
  104. package/dist/src/lib/exportInCTxBuilder.js +0 -170
  105. package/dist/src/lib/exportInPTxBuilder.d.ts +0 -30
  106. package/dist/src/lib/exportInPTxBuilder.d.ts.map +0 -1
  107. package/dist/src/lib/exportInPTxBuilder.js +0 -56
  108. package/dist/test/unit/lib/atomicTransactionBuilder.d.ts +0 -2
  109. package/dist/test/unit/lib/atomicTransactionBuilder.d.ts.map +0 -1
  110. package/dist/test/unit/lib/atomicTransactionBuilder.js +0 -222
  111. package/dist/test/unit/lib/exportTxBuilder.d.ts +0 -2
  112. package/dist/test/unit/lib/exportTxBuilder.d.ts.map +0 -1
  113. package/dist/test/unit/lib/exportTxBuilder.js +0 -45
  114. package/dist/test/unit/lib/transaction.d.ts +0 -2
  115. package/dist/test/unit/lib/transaction.d.ts.map +0 -1
  116. package/dist/test/unit/lib/transaction.js +0 -460
  117. package/dist/test/unit/smoke.d.ts +0 -2
  118. package/dist/test/unit/smoke.d.ts.map +0 -1
  119. package/dist/test/unit/smoke.js +0 -23
@@ -32,87 +32,465 @@ var __importStar = (this && this.__importStar) || (function () {
32
32
  return result;
33
33
  };
34
34
  })();
35
+ var __importDefault = (this && this.__importDefault) || function (mod) {
36
+ return (mod && mod.__esModule) ? mod : { "default": mod };
37
+ };
35
38
  Object.defineProperty(exports, "__esModule", { value: true });
36
- const statics_1 = require("@bitgo-beta/statics");
37
- const assert = __importStar(require("assert"));
38
- const flrp_1 = require("../../src/flrp");
39
+ const FlrpLib = __importStar(require("../../src/lib"));
39
40
  const sdk_test_1 = require("@bitgo-beta/sdk-test");
41
+ const src_1 = require("../../src/");
42
+ const crypto_1 = require("crypto");
40
43
  const sdk_api_1 = require("@bitgo-beta/sdk-api");
41
- describe('Flrp', function () {
44
+ const account_1 = require("../resources/account");
45
+ const exportInC_1 = require("../resources/transactionData/exportInC");
46
+ const exportInP_1 = require("../resources/transactionData/exportInP");
47
+ const importInP_1 = require("../resources/transactionData/importInP");
48
+ const importInC_1 = require("../resources/transactionData/importInC");
49
+ const sdk_core_1 = require("@bitgo-beta/sdk-core");
50
+ const assert_1 = __importDefault(require("assert"));
51
+ describe('Flrp test cases', function () {
52
+ const coinName = 'flrp';
53
+ const tcoinName = 't' + coinName;
42
54
  let bitgo;
43
- const staticsCoin = statics_1.coins.get('flrp');
55
+ let basecoin;
56
+ const keychains = [{ pub: account_1.SEED_ACCOUNT.publicKey }, { pub: account_1.ACCOUNT_1.publicKey }, { pub: account_1.ACCOUNT_2.publicKey }];
44
57
  before(function () {
45
- bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'mock' });
58
+ bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, {
59
+ env: 'mock',
60
+ });
46
61
  bitgo.initializeTestVars();
47
- // Attempt to register the coin symbol; safeRegister is idempotent.
48
- bitgo.safeRegister?.('flrp', flrp_1.Flrp.createInstance);
62
+ bitgo.safeRegister(coinName, src_1.Flrp.createInstance);
63
+ bitgo.safeRegister(tcoinName, src_1.TflrP.createInstance);
64
+ basecoin = bitgo.coin(tcoinName);
65
+ });
66
+ it('should instantiate the coin', function () {
67
+ let localBasecoin = bitgo.coin(tcoinName);
68
+ localBasecoin.should.be.an.instanceof(src_1.TflrP);
69
+ localBasecoin = bitgo.coin(coinName);
70
+ localBasecoin.should.be.an.instanceof(src_1.Flrp);
71
+ });
72
+ it('should return ' + tcoinName, function () {
73
+ basecoin.getChain().should.equal(tcoinName);
74
+ });
75
+ it('should return full name', function () {
76
+ basecoin.getFullName().should.equal('Testnet Flare P-Chain');
77
+ });
78
+ it('should return base factor', function () {
79
+ basecoin.getBaseFactor().should.equal(1e9);
80
+ });
81
+ it('should return coin family', function () {
82
+ basecoin.getFamily().should.equal('flrp');
83
+ });
84
+ it('should return default multisig type', function () {
85
+ basecoin.getDefaultMultisigType().should.equal('onchain');
86
+ });
87
+ describe('Keypairs:', () => {
88
+ it('should generate a keypair from random seed', function () {
89
+ const keyPair = basecoin.generateKeyPair();
90
+ keyPair.should.have.property('pub');
91
+ keyPair.should.have.property('prv');
92
+ });
93
+ it('should generate a keypair from a seed', function () {
94
+ const seed = Buffer.from(account_1.SEED_ACCOUNT.seed, 'hex');
95
+ const keyPair = basecoin.generateKeyPair(seed);
96
+ keyPair.pub.should.equal(account_1.SEED_ACCOUNT.publicKey);
97
+ keyPair.prv.should.equal(account_1.SEED_ACCOUNT.privateKey);
98
+ });
99
+ it('should validate a public key', function () {
100
+ basecoin.isValidPub(account_1.SEED_ACCOUNT.publicKey).should.equal(true);
101
+ basecoin.isValidPub(account_1.ACCOUNT_1.publicKey).should.equal(true);
102
+ });
103
+ it('should fail to validate an invalid public key', function () {
104
+ basecoin.isValidPub('invalid').should.equal(false);
105
+ });
106
+ it('should validate a private key', function () {
107
+ basecoin.isValidPrv(account_1.SEED_ACCOUNT.privateKey).should.equal(true);
108
+ basecoin.isValidPrv(account_1.ACCOUNT_1.privateKey).should.equal(true);
109
+ });
110
+ it('should fail to validate an invalid private key', function () {
111
+ basecoin.isValidPrv('invalid').should.equal(false);
112
+ });
113
+ });
114
+ describe('Sign Transaction', () => {
115
+ it('should sign an export from C-chain transaction', async () => {
116
+ const params = {
117
+ txPrebuild: {
118
+ txHex: exportInC_1.EXPORT_IN_C.unsignedHex,
119
+ },
120
+ prv: exportInC_1.EXPORT_IN_C.privateKey,
121
+ };
122
+ const signedTx = await basecoin.signTransaction(params);
123
+ signedTx.should.have.property('halfSigned');
124
+ const halfSigned = signedTx.halfSigned;
125
+ (0, assert_1.default)(halfSigned, 'halfSigned should be defined');
126
+ (0, assert_1.default)(halfSigned.txHex, 'txHex should be defined');
127
+ halfSigned.txHex.should.equal(exportInC_1.EXPORT_IN_C.signedHex);
128
+ });
129
+ it('should sign an export from P-chain transaction', async () => {
130
+ const params = {
131
+ txPrebuild: {
132
+ txHex: exportInP_1.EXPORT_IN_P.unsignedHex,
133
+ },
134
+ prv: exportInP_1.EXPORT_IN_P.privateKeys[0],
135
+ };
136
+ const signedTx = await basecoin.signTransaction(params);
137
+ signedTx.should.have.property('halfSigned');
138
+ const halfSigned = signedTx.halfSigned;
139
+ (0, assert_1.default)(halfSigned, 'halfSigned should be defined');
140
+ (0, assert_1.default)(halfSigned.txHex, 'txHex should be defined');
141
+ halfSigned.txHex.should.equal(exportInP_1.EXPORT_IN_P.halfSigntxHex);
142
+ });
143
+ it('should sign an import to P-chain transaction', async () => {
144
+ const params = {
145
+ txPrebuild: {
146
+ txHex: importInP_1.IMPORT_IN_P.unsignedHex,
147
+ },
148
+ prv: importInP_1.IMPORT_IN_P.privateKeys[0],
149
+ };
150
+ const signedTx = await basecoin.signTransaction(params);
151
+ signedTx.should.have.property('halfSigned');
152
+ const halfSigned = signedTx.halfSigned;
153
+ (0, assert_1.default)(halfSigned, 'halfSigned should be defined');
154
+ (0, assert_1.default)(halfSigned.txHex, 'txHex should be defined');
155
+ halfSigned.txHex.should.equal(importInP_1.IMPORT_IN_P.halfSigntxHex);
156
+ });
157
+ it('should sign an import to C-chain transaction', async () => {
158
+ const params = {
159
+ txPrebuild: {
160
+ txHex: importInC_1.IMPORT_IN_C.unsignedHex,
161
+ },
162
+ prv: importInC_1.IMPORT_IN_C.privateKeys[0],
163
+ };
164
+ const signedTx = await basecoin.signTransaction(params);
165
+ signedTx.should.have.property('halfSigned');
166
+ const halfSigned = signedTx.halfSigned;
167
+ (0, assert_1.default)(halfSigned, 'halfSigned should be defined');
168
+ (0, assert_1.default)(halfSigned.txHex, 'txHex should be defined');
169
+ halfSigned.txHex.should.equal(importInC_1.IMPORT_IN_C.halfSigntxHex);
170
+ });
171
+ it('should reject signing with an invalid key', async () => {
172
+ const params = {
173
+ txPrebuild: {
174
+ txHex: exportInC_1.EXPORT_IN_C.unsignedHex,
175
+ },
176
+ prv: 'invalid-key',
177
+ };
178
+ await basecoin.signTransaction(params).should.be.rejected();
179
+ });
180
+ });
181
+ describe('Sign Message', () => {
182
+ it('should sign a message', async () => {
183
+ const keyPair = new FlrpLib.KeyPair({ prv: account_1.SEED_ACCOUNT.privateKey });
184
+ const keys = keyPair.getKeys();
185
+ const messageToSign = Buffer.from(account_1.SEED_ACCOUNT.message, 'utf8');
186
+ const signature = await basecoin.signMessage(keys, messageToSign.toString('hex'));
187
+ signature.should.be.instanceOf(Buffer);
188
+ signature.length.should.equal(65);
189
+ });
190
+ it('should sign a random message', async () => {
191
+ const keyPair = new FlrpLib.KeyPair();
192
+ const pubKey = keyPair.getKeys().pub;
193
+ const keys = keyPair.getKeys();
194
+ const messageToSign = Buffer.from((0, crypto_1.randomBytes)(32));
195
+ const signature = await basecoin.signMessage(keys, messageToSign.toString('hex'));
196
+ const verify = FlrpLib.Utils.verifySignature(FlrpLib.Utils.sha256(messageToSign), signature.slice(0, 64), // Remove recovery byte for verification
197
+ Buffer.from(pubKey, 'hex'));
198
+ verify.should.be.true();
199
+ });
200
+ it('should fail to sign with missing private key', async () => {
201
+ const keyPair = new FlrpLib.KeyPair({ pub: account_1.SEED_ACCOUNT.publicKey });
202
+ const keys = keyPair.getKeys();
203
+ const messageToSign = Buffer.from(account_1.SEED_ACCOUNT.message, 'utf8');
204
+ await basecoin
205
+ .signMessage(keys, messageToSign.toString('hex'))
206
+ .should.be.rejectedWith('Invalid key pair options');
207
+ });
208
+ });
209
+ describe('Explain Transaction', () => {
210
+ it('should explain a half signed export from P-chain transaction', async () => {
211
+ const txExplain = await basecoin.explainTransaction({
212
+ halfSigned: { txHex: exportInP_1.EXPORT_IN_P.halfSigntxHex },
213
+ });
214
+ txExplain.type.should.equal(sdk_core_1.TransactionType.Export);
215
+ txExplain.fee.fee.should.equal(exportInP_1.EXPORT_IN_P.fee);
216
+ txExplain.inputs.should.be.an.Array();
217
+ txExplain.changeAmount.should.equal('498459568');
218
+ txExplain.changeOutputs.should.be.an.Array();
219
+ txExplain.changeOutputs[0].address.should.equal('P-costwo106gc5h5qswhye8e0pmthq4wzf0ekv5qppsrvpu~P-costwo1cueygd7fd37g56s49k3rshqakhp6k8u3adzt6m~P-costwo1xv5mulgpe5lt4tnx2ntnylwe79azu9vpja6lut');
220
+ });
221
+ it('should explain a signed export from P-chain transaction', async () => {
222
+ const txExplain = await basecoin.explainTransaction({ txHex: exportInP_1.EXPORT_IN_P.fullSigntxHex });
223
+ txExplain.type.should.equal(sdk_core_1.TransactionType.Export);
224
+ txExplain.id.should.equal(exportInP_1.EXPORT_IN_P.txhash);
225
+ txExplain.fee.fee.should.equal(exportInP_1.EXPORT_IN_P.fee);
226
+ txExplain.inputs.should.be.an.Array();
227
+ txExplain.changeAmount.should.equal('498459568');
228
+ txExplain.changeOutputs.should.be.an.Array();
229
+ txExplain.changeOutputs[0].address.should.equal('P-costwo106gc5h5qswhye8e0pmthq4wzf0ekv5qppsrvpu~P-costwo1cueygd7fd37g56s49k3rshqakhp6k8u3adzt6m~P-costwo1xv5mulgpe5lt4tnx2ntnylwe79azu9vpja6lut');
230
+ });
231
+ it('should explain a half signed import to P-chain transaction', async () => {
232
+ const txExplain = await basecoin.explainTransaction({
233
+ halfSigned: { txHex: importInP_1.IMPORT_IN_P.halfSigntxHex },
234
+ });
235
+ txExplain.type.should.equal(sdk_core_1.TransactionType.Import);
236
+ txExplain.fee.fee.should.equal(importInP_1.IMPORT_IN_P.fee);
237
+ txExplain.inputs.should.be.an.Array();
238
+ txExplain.outputAmount.should.equal('48739000');
239
+ txExplain.outputs.should.be.an.Array();
240
+ txExplain.outputs.length.should.equal(1);
241
+ txExplain.changeOutputs.should.be.empty();
242
+ });
243
+ it('should explain a signed import to P-chain transaction', async () => {
244
+ const txExplain = await basecoin.explainTransaction({ txHex: importInP_1.IMPORT_IN_P.fullSigntxHex });
245
+ txExplain.type.should.equal(sdk_core_1.TransactionType.Import);
246
+ txExplain.id.should.equal(importInP_1.IMPORT_IN_P.txhash);
247
+ txExplain.fee.fee.should.equal(importInP_1.IMPORT_IN_P.fee);
248
+ txExplain.inputs.should.be.an.Array();
249
+ txExplain.outputAmount.should.equal('48739000');
250
+ txExplain.outputs.should.be.an.Array();
251
+ txExplain.outputs.length.should.equal(1);
252
+ txExplain.changeOutputs.should.be.empty();
253
+ });
254
+ it('should fail when transaction hex is not provided', async () => {
255
+ await basecoin.explainTransaction({}).should.be.rejectedWith('missing transaction hex');
256
+ });
257
+ it('should fail for invalid transaction hex', async () => {
258
+ await basecoin.explainTransaction({ txHex: 'invalid' }).should.be.rejected();
259
+ });
49
260
  });
50
- describe('createInstance', function () {
51
- it('should return a Flrp instance', function () {
52
- const coin = flrp_1.Flrp.createInstance(bitgo, staticsCoin);
53
- assert.ok(coin instanceof flrp_1.Flrp);
54
- });
55
- it('should produce distinct objects on multiple calls', function () {
56
- const a = flrp_1.Flrp.createInstance(bitgo, staticsCoin);
57
- const b = flrp_1.Flrp.createInstance(bitgo, staticsCoin);
58
- assert.notStrictEqual(a, b);
59
- assert.ok(a instanceof flrp_1.Flrp);
60
- assert.ok(b instanceof flrp_1.Flrp);
261
+ describe('Verify Transaction', () => {
262
+ it('should verify an export from C-chain transaction', async () => {
263
+ const txPrebuild = {
264
+ txHex: exportInC_1.EXPORT_IN_C.signedHex,
265
+ txInfo: {},
266
+ };
267
+ const txParams = {
268
+ recipients: [
269
+ {
270
+ address: '',
271
+ amount: exportInC_1.EXPORT_IN_C.amount,
272
+ },
273
+ ],
274
+ type: 'Export',
275
+ locktime: 0,
276
+ };
277
+ const isVerified = await basecoin.verifyTransaction({ txParams, txPrebuild });
278
+ isVerified.should.equal(true);
279
+ });
280
+ it('should verify an export from P-chain transaction', async () => {
281
+ const txPrebuild = {
282
+ txHex: exportInP_1.EXPORT_IN_P.fullSigntxHex,
283
+ txInfo: {},
284
+ };
285
+ const txParams = {
286
+ recipients: [
287
+ {
288
+ address: '',
289
+ amount: exportInP_1.EXPORT_IN_P.amount,
290
+ },
291
+ ],
292
+ type: 'Export',
293
+ locktime: 0,
294
+ };
295
+ const isVerified = await basecoin.verifyTransaction({ txParams, txPrebuild });
296
+ isVerified.should.equal(true);
297
+ });
298
+ it('should verify an import to C-chain transaction', async () => {
299
+ const txPrebuild = {
300
+ txHex: importInC_1.IMPORT_IN_C.fullSigntxHex,
301
+ txInfo: {},
302
+ };
303
+ const txParams = {
304
+ recipients: [
305
+ {
306
+ address: importInC_1.IMPORT_IN_C.to,
307
+ amount: '1',
308
+ },
309
+ ],
310
+ type: 'ImportToC',
311
+ locktime: 0,
312
+ };
313
+ const isVerified = await basecoin.verifyTransaction({ txParams, txPrebuild });
314
+ isVerified.should.equal(true);
315
+ });
316
+ it('should verify an import to P-chain transaction', async () => {
317
+ const txPrebuild = {
318
+ txHex: importInP_1.IMPORT_IN_P.fullSigntxHex,
319
+ txInfo: {},
320
+ };
321
+ const txParams = {
322
+ recipients: [],
323
+ type: 'Import',
324
+ locktime: 0,
325
+ };
326
+ const isVerified = await basecoin.verifyTransaction({ txParams, txPrebuild });
327
+ isVerified.should.equal(true);
328
+ });
329
+ it('should fail to verify export transaction with wrong amount', async () => {
330
+ const txPrebuild = {
331
+ txHex: exportInC_1.EXPORT_IN_C.signedHex,
332
+ txInfo: {},
333
+ };
334
+ const txParams = {
335
+ recipients: [
336
+ {
337
+ address: '',
338
+ amount: '999999999999',
339
+ },
340
+ ],
341
+ type: 'Export',
342
+ locktime: 0,
343
+ };
344
+ await basecoin
345
+ .verifyTransaction({ txParams, txPrebuild })
346
+ .should.be.rejectedWith(/Tx total amount .* does not match with expected total amount/);
347
+ });
348
+ it('should fail to verify transaction with wrong type', async () => {
349
+ const txPrebuild = {
350
+ txHex: exportInC_1.EXPORT_IN_C.signedHex,
351
+ txInfo: {},
352
+ };
353
+ const txParams = {
354
+ recipients: [
355
+ {
356
+ address: '',
357
+ amount: exportInC_1.EXPORT_IN_C.amount,
358
+ },
359
+ ],
360
+ type: 'Import',
361
+ locktime: 0,
362
+ };
363
+ await basecoin
364
+ .verifyTransaction({ txParams, txPrebuild })
365
+ .should.be.rejectedWith('Tx type does not match with expected txParams type');
366
+ });
367
+ it('should fail to verify transaction without txHex', async () => {
368
+ const txPrebuild = {
369
+ txInfo: {},
370
+ };
371
+ const txParams = {
372
+ recipients: [],
373
+ type: 'Export',
374
+ locktime: 0,
375
+ };
376
+ await basecoin
377
+ .verifyTransaction({ txParams, txPrebuild })
378
+ .should.be.rejectedWith('missing required tx prebuild property txHex');
379
+ });
380
+ it('should fail to verify transaction with invalid txHex', async () => {
381
+ const txPrebuild = {
382
+ txHex: 'invalidhex',
383
+ txInfo: {},
384
+ };
385
+ const txParams = {
386
+ recipients: [],
387
+ type: 'Export',
388
+ locktime: 0,
389
+ };
390
+ await basecoin
391
+ .verifyTransaction({ txParams, txPrebuild })
392
+ .should.be.rejectedWith('Invalid transaction: Raw transaction is not hex string');
393
+ });
394
+ it('should fail to verify import to C-chain without recipients', async () => {
395
+ const txPrebuild = {
396
+ txHex: importInC_1.IMPORT_IN_C.fullSigntxHex,
397
+ txInfo: {},
398
+ };
399
+ const txParams = {
400
+ recipients: [],
401
+ type: 'ImportToC',
402
+ locktime: 0,
403
+ };
404
+ await basecoin
405
+ .verifyTransaction({ txParams, txPrebuild })
406
+ .should.be.rejectedWith('Expected 1 recipient in import transaction');
61
407
  });
62
408
  });
63
- describe('coin properties', function () {
64
- let coin;
65
- beforeEach(function () {
66
- coin = flrp_1.Flrp.createInstance(bitgo, staticsCoin);
67
- });
68
- it('should have correct coin family', function () {
69
- assert.strictEqual(coin.getFamily(), staticsCoin.family);
70
- });
71
- it('should have correct coin name', function () {
72
- assert.strictEqual(coin.getFullName(), staticsCoin.fullName);
73
- });
74
- it('should have correct base factor', function () {
75
- assert.strictEqual(coin.getBaseFactor(), Math.pow(10, staticsCoin.decimalPlaces));
76
- });
77
- it('should validate addresses using utils', function () {
78
- const validAddress = 'flare1qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqq6f4avh';
79
- const result = coin.isValidAddress(validAddress);
80
- assert.strictEqual(typeof result, 'boolean');
81
- });
82
- it('should generate key pairs', function () {
83
- const keyPair = coin.generateKeyPair();
84
- assert.ok('pub' in keyPair);
85
- assert.ok('prv' in keyPair);
86
- if (keyPair.pub && keyPair.prv) {
87
- assert.strictEqual(typeof keyPair.pub, 'string');
88
- assert.strictEqual(typeof keyPair.prv, 'string');
89
- }
409
+ describe('Address Validation', () => {
410
+ it('should validate mainnet P-chain address', function () {
411
+ basecoin.isValidAddress(account_1.SEED_ACCOUNT.addressMainnet).should.be.true();
412
+ });
413
+ it('should validate testnet P-chain address', function () {
414
+ basecoin.isValidAddress(account_1.SEED_ACCOUNT.addressTestnet).should.be.true();
415
+ });
416
+ it('should validate array of P-chain addresses', function () {
417
+ basecoin.isValidAddress(exportInC_1.EXPORT_IN_C.pAddresses).should.be.true();
418
+ });
419
+ it('should validate tilde-separated multisig address', function () {
420
+ const multiSigAddress = exportInC_1.EXPORT_IN_C.pAddresses.join('~');
421
+ basecoin.isValidAddress(multiSigAddress).should.be.true();
422
+ });
423
+ it('should validate C-chain hex address', function () {
424
+ basecoin.isValidAddress(exportInC_1.EXPORT_IN_C.cHexAddress).should.be.true();
425
+ });
426
+ it('should validate lowercase C-chain address', function () {
427
+ basecoin.isValidAddress(importInC_1.IMPORT_IN_C.to.toLowerCase()).should.be.true();
428
+ });
429
+ it('should fail to validate undefined address', function () {
430
+ basecoin.isValidAddress(undefined).should.be.false();
431
+ });
432
+ it('should fail to validate empty string', function () {
433
+ basecoin.isValidAddress('').should.be.false();
434
+ });
435
+ it('should fail to validate invalid address', function () {
436
+ basecoin.isValidAddress('invalid-address').should.be.false();
437
+ });
438
+ it('should fail to validate array with invalid address', function () {
439
+ const addresses = [...exportInC_1.EXPORT_IN_C.pAddresses, 'invalid'];
440
+ basecoin.isValidAddress(addresses).should.be.false();
90
441
  });
91
442
  });
92
- describe('error handling', function () {
93
- it('should handle construction with invalid parameters', function () {
94
- assert.throws(() => flrp_1.Flrp.createInstance(null, staticsCoin));
95
- assert.throws(() => flrp_1.Flrp.createInstance(bitgo, null));
443
+ describe('Wallet Address Verification', () => {
444
+ it('should verify wallet address with matching keychains', async () => {
445
+ const keyPairs = [{ pub: account_1.SEED_ACCOUNT.publicKey }, { pub: account_1.ACCOUNT_1.publicKey }, { pub: account_1.ACCOUNT_2.publicKey }];
446
+ // Derive addresses from public keys to ensure they match
447
+ const derivedAddresses = keyPairs.map((kp) => new FlrpLib.KeyPair({ pub: kp.pub }).getAddress('testnet'));
448
+ const address = derivedAddresses.join('~');
449
+ const isValid = await basecoin.isWalletAddress({
450
+ address,
451
+ keychains: keyPairs,
452
+ });
453
+ isValid.should.be.true();
454
+ });
455
+ it('should throw for address with wrong number of keychains', async () => {
456
+ const address = account_1.SEED_ACCOUNT.addressTestnet;
457
+ await assert_1.default.rejects(async () => basecoin.isWalletAddress({
458
+ address,
459
+ keychains: [{ pub: account_1.SEED_ACCOUNT.publicKey }],
460
+ }), /Invalid keychains/);
461
+ });
462
+ it('should throw for invalid address', async () => {
463
+ await assert_1.default.rejects(async () => basecoin.isWalletAddress({
464
+ address: 'invalid',
465
+ keychains,
466
+ }), /invalid address/);
467
+ });
468
+ it('should throw when address length does not match keychain length', async () => {
469
+ const address = account_1.SEED_ACCOUNT.addressTestnet;
470
+ await assert_1.default.rejects(async () => basecoin.isWalletAddress({
471
+ address,
472
+ keychains,
473
+ }));
474
+ });
475
+ it('should throw when addresses do not match keychains', async () => {
476
+ // Use addresses that don't match the keychains
477
+ const address = exportInC_1.EXPORT_IN_C.pAddresses.join('~');
478
+ await assert_1.default.rejects(async () => basecoin.isWalletAddress({
479
+ address,
480
+ keychains,
481
+ }));
96
482
  });
97
483
  });
98
- describe('inheritance and methods', function () {
99
- let coin;
100
- beforeEach(function () {
101
- coin = flrp_1.Flrp.createInstance(bitgo, staticsCoin);
102
- });
103
- it('should have required base coin methods', function () {
104
- assert.ok('getFamily' in coin);
105
- assert.ok('getFullName' in coin);
106
- assert.ok('getBaseFactor' in coin);
107
- assert.ok('isValidAddress' in coin);
108
- assert.ok('generateKeyPair' in coin);
109
- });
110
- it('should handle address validation consistently', function () {
111
- const validAddress = 'flare1test';
112
- const invalidAddress = 'invalid-address';
113
- assert.strictEqual(typeof coin.isValidAddress(validAddress), 'boolean');
114
- assert.strictEqual(typeof coin.isValidAddress(invalidAddress), 'boolean');
484
+ describe('Recovery Signature', () => {
485
+ it('should recover signature from signed message', async () => {
486
+ const message = Buffer.from(account_1.SEED_ACCOUNT.message, 'utf8');
487
+ const privateKey = Buffer.from(account_1.SEED_ACCOUNT.privateKey, 'hex');
488
+ const signature = FlrpLib.Utils.createSignature(basecoin._staticsCoin.network, message, privateKey);
489
+ const messageHash = FlrpLib.Utils.sha256(message);
490
+ const recoveredPubKey = basecoin.recoverySignature(messageHash, signature);
491
+ recoveredPubKey.should.be.instanceOf(Buffer);
492
+ recoveredPubKey.length.should.equal(33);
115
493
  });
116
494
  });
117
495
  });
118
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"flrp.js","sourceRoot":"","sources":["../../../test/unit/flrp.ts"],"names":[],"mappings":"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","sourcesContent":["import { coins } from '@bitgo-beta/statics';\nimport * as assert from 'assert';\nimport { Flrp } from '../../src/flrp';\nimport { BitGoBase } from '@bitgo-beta/sdk-core';\nimport { TestBitGo, TestBitGoAPI } from '@bitgo-beta/sdk-test';\nimport { BitGoAPI } from '@bitgo-beta/sdk-api';\n\ndescribe('Flrp', function () {\n  let bitgo: TestBitGoAPI;\n  const staticsCoin = coins.get('flrp');\n\n  before(function () {\n    bitgo = TestBitGo.decorate(BitGoAPI, { env: 'mock' });\n    bitgo.initializeTestVars();\n    // Attempt to register the coin symbol; safeRegister is idempotent.\n    (bitgo as unknown as { safeRegister?: (n: string, f: (bg: BitGoBase) => unknown) => void }).safeRegister?.(\n      'flrp',\n      Flrp.createInstance\n    );\n  });\n\n  describe('createInstance', function () {\n    it('should return a Flrp instance', function () {\n      const coin = Flrp.createInstance(bitgo as unknown as BitGoBase, staticsCoin);\n      assert.ok(coin instanceof Flrp);\n    });\n\n    it('should produce distinct objects on multiple calls', function () {\n      const a = Flrp.createInstance(bitgo as unknown as BitGoBase, staticsCoin);\n      const b = Flrp.createInstance(bitgo as unknown as BitGoBase, staticsCoin);\n      assert.notStrictEqual(a, b);\n      assert.ok(a instanceof Flrp);\n      assert.ok(b instanceof Flrp);\n    });\n  });\n\n  describe('coin properties', function () {\n    let coin: Flrp;\n\n    beforeEach(function () {\n      coin = Flrp.createInstance(bitgo as unknown as BitGoBase, staticsCoin) as Flrp;\n    });\n\n    it('should have correct coin family', function () {\n      assert.strictEqual(coin.getFamily(), staticsCoin.family);\n    });\n\n    it('should have correct coin name', function () {\n      assert.strictEqual(coin.getFullName(), staticsCoin.fullName);\n    });\n\n    it('should have correct base factor', function () {\n      assert.strictEqual(coin.getBaseFactor(), Math.pow(10, staticsCoin.decimalPlaces));\n    });\n\n    it('should validate addresses using utils', function () {\n      const validAddress = 'flare1qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqq6f4avh';\n      const result = coin.isValidAddress(validAddress);\n      assert.strictEqual(typeof result, 'boolean');\n    });\n\n    it('should generate key pairs', function () {\n      const keyPair = coin.generateKeyPair();\n      assert.ok('pub' in keyPair);\n      assert.ok('prv' in keyPair);\n      if (keyPair.pub && keyPair.prv) {\n        assert.strictEqual(typeof keyPair.pub, 'string');\n        assert.strictEqual(typeof keyPair.prv, 'string');\n      }\n    });\n  });\n\n  describe('error handling', function () {\n    it('should handle construction with invalid parameters', function () {\n      assert.throws(() => Flrp.createInstance(null as unknown as BitGoBase, staticsCoin));\n      assert.throws(() => Flrp.createInstance(bitgo as unknown as BitGoBase, null as unknown as typeof staticsCoin));\n    });\n  });\n\n  describe('inheritance and methods', function () {\n    let coin: Flrp;\n\n    beforeEach(function () {\n      coin = Flrp.createInstance(bitgo as unknown as BitGoBase, staticsCoin) as Flrp;\n    });\n\n    it('should have required base coin methods', function () {\n      assert.ok('getFamily' in coin);\n      assert.ok('getFullName' in coin);\n      assert.ok('getBaseFactor' in coin);\n      assert.ok('isValidAddress' in coin);\n      assert.ok('generateKeyPair' in coin);\n    });\n\n    it('should handle address validation consistently', function () {\n      const validAddress = 'flare1test';\n      const invalidAddress = 'invalid-address';\n\n      assert.strictEqual(typeof coin.isValidAddress(validAddress), 'boolean');\n      assert.strictEqual(typeof coin.isValidAddress(invalidAddress), 'boolean');\n    });\n  });\n});\n"]}
496
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"flrp.js","sourceRoot":"","sources":["../../../test/unit/flrp.ts"],"names":[],"mappings":"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","sourcesContent":["import * as FlrpLib from '../../src/lib';\nimport { TestBitGo, TestBitGoAPI } from '@bitgo-beta/sdk-test';\nimport { Flrp, TflrP } from '../../src/';\nimport { randomBytes } from 'crypto';\nimport { BitGoAPI } from '@bitgo-beta/sdk-api';\nimport { SEED_ACCOUNT, ACCOUNT_1, ACCOUNT_2 } from '../resources/account';\nimport { EXPORT_IN_C } from '../resources/transactionData/exportInC';\nimport { EXPORT_IN_P } from '../resources/transactionData/exportInP';\nimport { IMPORT_IN_P } from '../resources/transactionData/importInP';\nimport { IMPORT_IN_C } from '../resources/transactionData/importInC';\nimport { HalfSignedAccountTransaction, TransactionType } from '@bitgo-beta/sdk-core';\nimport assert from 'assert';\n\ndescribe('Flrp test cases', function () {\n  const coinName = 'flrp';\n  const tcoinName = 't' + coinName;\n  let bitgo: TestBitGoAPI;\n  let basecoin;\n\n  const keychains = [{ pub: SEED_ACCOUNT.publicKey }, { pub: ACCOUNT_1.publicKey }, { pub: ACCOUNT_2.publicKey }];\n\n  before(function () {\n    bitgo = TestBitGo.decorate(BitGoAPI, {\n      env: 'mock',\n    });\n    bitgo.initializeTestVars();\n    bitgo.safeRegister(coinName, Flrp.createInstance);\n    bitgo.safeRegister(tcoinName, TflrP.createInstance);\n    basecoin = bitgo.coin(tcoinName);\n  });\n\n  it('should instantiate the coin', function () {\n    let localBasecoin = bitgo.coin(tcoinName);\n    localBasecoin.should.be.an.instanceof(TflrP);\n\n    localBasecoin = bitgo.coin(coinName);\n    localBasecoin.should.be.an.instanceof(Flrp);\n  });\n\n  it('should return ' + tcoinName, function () {\n    basecoin.getChain().should.equal(tcoinName);\n  });\n\n  it('should return full name', function () {\n    basecoin.getFullName().should.equal('Testnet Flare P-Chain');\n  });\n\n  it('should return base factor', function () {\n    basecoin.getBaseFactor().should.equal(1e9);\n  });\n\n  it('should return coin family', function () {\n    basecoin.getFamily().should.equal('flrp');\n  });\n\n  it('should return default multisig type', function () {\n    basecoin.getDefaultMultisigType().should.equal('onchain');\n  });\n\n  describe('Keypairs:', () => {\n    it('should generate a keypair from random seed', function () {\n      const keyPair = basecoin.generateKeyPair();\n      keyPair.should.have.property('pub');\n      keyPair.should.have.property('prv');\n    });\n\n    it('should generate a keypair from a seed', function () {\n      const seed = Buffer.from(SEED_ACCOUNT.seed, 'hex');\n      const keyPair = basecoin.generateKeyPair(seed);\n      keyPair.pub.should.equal(SEED_ACCOUNT.publicKey);\n      keyPair.prv.should.equal(SEED_ACCOUNT.privateKey);\n    });\n\n    it('should validate a public key', function () {\n      basecoin.isValidPub(SEED_ACCOUNT.publicKey).should.equal(true);\n      basecoin.isValidPub(ACCOUNT_1.publicKey).should.equal(true);\n    });\n\n    it('should fail to validate an invalid public key', function () {\n      basecoin.isValidPub('invalid').should.equal(false);\n    });\n\n    it('should validate a private key', function () {\n      basecoin.isValidPrv(SEED_ACCOUNT.privateKey).should.equal(true);\n      basecoin.isValidPrv(ACCOUNT_1.privateKey).should.equal(true);\n    });\n\n    it('should fail to validate an invalid private key', function () {\n      basecoin.isValidPrv('invalid').should.equal(false);\n    });\n  });\n\n  describe('Sign Transaction', () => {\n    it('should sign an export from C-chain transaction', async () => {\n      const params = {\n        txPrebuild: {\n          txHex: EXPORT_IN_C.unsignedHex,\n        },\n        prv: EXPORT_IN_C.privateKey,\n      };\n\n      const signedTx = await basecoin.signTransaction(params);\n      signedTx.should.have.property('halfSigned');\n      const halfSigned = (signedTx as HalfSignedAccountTransaction).halfSigned;\n      assert(halfSigned, 'halfSigned should be defined');\n      assert(halfSigned.txHex, 'txHex should be defined');\n      halfSigned.txHex.should.equal(EXPORT_IN_C.signedHex);\n    });\n\n    it('should sign an export from P-chain transaction', async () => {\n      const params = {\n        txPrebuild: {\n          txHex: EXPORT_IN_P.unsignedHex,\n        },\n        prv: EXPORT_IN_P.privateKeys[0],\n      };\n\n      const signedTx = await basecoin.signTransaction(params);\n      signedTx.should.have.property('halfSigned');\n      const halfSigned = (signedTx as HalfSignedAccountTransaction).halfSigned;\n      assert(halfSigned, 'halfSigned should be defined');\n      assert(halfSigned.txHex, 'txHex should be defined');\n      halfSigned.txHex.should.equal(EXPORT_IN_P.halfSigntxHex);\n    });\n\n    it('should sign an import to P-chain transaction', async () => {\n      const params = {\n        txPrebuild: {\n          txHex: IMPORT_IN_P.unsignedHex,\n        },\n        prv: IMPORT_IN_P.privateKeys[0],\n      };\n\n      const signedTx = await basecoin.signTransaction(params);\n      signedTx.should.have.property('halfSigned');\n      const halfSigned = (signedTx as HalfSignedAccountTransaction).halfSigned;\n      assert(halfSigned, 'halfSigned should be defined');\n      assert(halfSigned.txHex, 'txHex should be defined');\n      halfSigned.txHex.should.equal(IMPORT_IN_P.halfSigntxHex);\n    });\n\n    it('should sign an import to C-chain transaction', async () => {\n      const params = {\n        txPrebuild: {\n          txHex: IMPORT_IN_C.unsignedHex,\n        },\n        prv: IMPORT_IN_C.privateKeys[0],\n      };\n\n      const signedTx = await basecoin.signTransaction(params);\n      signedTx.should.have.property('halfSigned');\n      const halfSigned = (signedTx as HalfSignedAccountTransaction).halfSigned;\n      assert(halfSigned, 'halfSigned should be defined');\n      assert(halfSigned.txHex, 'txHex should be defined');\n      halfSigned.txHex.should.equal(IMPORT_IN_C.halfSigntxHex);\n    });\n\n    it('should reject signing with an invalid key', async () => {\n      const params = {\n        txPrebuild: {\n          txHex: EXPORT_IN_C.unsignedHex,\n        },\n        prv: 'invalid-key',\n      };\n\n      await basecoin.signTransaction(params).should.be.rejected();\n    });\n  });\n\n  describe('Sign Message', () => {\n    it('should sign a message', async () => {\n      const keyPair = new FlrpLib.KeyPair({ prv: SEED_ACCOUNT.privateKey });\n      const keys = keyPair.getKeys();\n      const messageToSign = Buffer.from(SEED_ACCOUNT.message, 'utf8');\n      const signature = await basecoin.signMessage(keys, messageToSign.toString('hex'));\n\n      signature.should.be.instanceOf(Buffer);\n      signature.length.should.equal(65);\n    });\n\n    it('should sign a random message', async () => {\n      const keyPair = new FlrpLib.KeyPair();\n      const pubKey = keyPair.getKeys().pub;\n      const keys = keyPair.getKeys();\n      const messageToSign = Buffer.from(randomBytes(32));\n      const signature = await basecoin.signMessage(keys, messageToSign.toString('hex'));\n\n      const verify = FlrpLib.Utils.verifySignature(\n        FlrpLib.Utils.sha256(messageToSign),\n        signature.slice(0, 64), // Remove recovery byte for verification\n        Buffer.from(pubKey, 'hex')\n      );\n      verify.should.be.true();\n    });\n\n    it('should fail to sign with missing private key', async () => {\n      const keyPair = new FlrpLib.KeyPair({ pub: SEED_ACCOUNT.publicKey });\n      const keys = keyPair.getKeys();\n      const messageToSign = Buffer.from(SEED_ACCOUNT.message, 'utf8');\n\n      await basecoin\n        .signMessage(keys, messageToSign.toString('hex'))\n        .should.be.rejectedWith('Invalid key pair options');\n    });\n  });\n\n  describe('Explain Transaction', () => {\n    it('should explain a half signed export from P-chain transaction', async () => {\n      const txExplain = await basecoin.explainTransaction({\n        halfSigned: { txHex: EXPORT_IN_P.halfSigntxHex },\n      });\n\n      txExplain.type.should.equal(TransactionType.Export);\n      txExplain.fee.fee.should.equal(EXPORT_IN_P.fee);\n      txExplain.inputs.should.be.an.Array();\n      txExplain.changeAmount.should.equal('498459568');\n      txExplain.changeOutputs.should.be.an.Array();\n      txExplain.changeOutputs[0].address.should.equal(\n        'P-costwo106gc5h5qswhye8e0pmthq4wzf0ekv5qppsrvpu~P-costwo1cueygd7fd37g56s49k3rshqakhp6k8u3adzt6m~P-costwo1xv5mulgpe5lt4tnx2ntnylwe79azu9vpja6lut'\n      );\n    });\n\n    it('should explain a signed export from P-chain transaction', async () => {\n      const txExplain = await basecoin.explainTransaction({ txHex: EXPORT_IN_P.fullSigntxHex });\n\n      txExplain.type.should.equal(TransactionType.Export);\n      txExplain.id.should.equal(EXPORT_IN_P.txhash);\n      txExplain.fee.fee.should.equal(EXPORT_IN_P.fee);\n      txExplain.inputs.should.be.an.Array();\n      txExplain.changeAmount.should.equal('498459568');\n      txExplain.changeOutputs.should.be.an.Array();\n      txExplain.changeOutputs[0].address.should.equal(\n        'P-costwo106gc5h5qswhye8e0pmthq4wzf0ekv5qppsrvpu~P-costwo1cueygd7fd37g56s49k3rshqakhp6k8u3adzt6m~P-costwo1xv5mulgpe5lt4tnx2ntnylwe79azu9vpja6lut'\n      );\n    });\n\n    it('should explain a half signed import to P-chain transaction', async () => {\n      const txExplain = await basecoin.explainTransaction({\n        halfSigned: { txHex: IMPORT_IN_P.halfSigntxHex },\n      });\n\n      txExplain.type.should.equal(TransactionType.Import);\n      txExplain.fee.fee.should.equal(IMPORT_IN_P.fee);\n      txExplain.inputs.should.be.an.Array();\n      txExplain.outputAmount.should.equal('48739000');\n      txExplain.outputs.should.be.an.Array();\n      txExplain.outputs.length.should.equal(1);\n      txExplain.changeOutputs.should.be.empty();\n    });\n\n    it('should explain a signed import to P-chain transaction', async () => {\n      const txExplain = await basecoin.explainTransaction({ txHex: IMPORT_IN_P.fullSigntxHex });\n\n      txExplain.type.should.equal(TransactionType.Import);\n      txExplain.id.should.equal(IMPORT_IN_P.txhash);\n      txExplain.fee.fee.should.equal(IMPORT_IN_P.fee);\n      txExplain.inputs.should.be.an.Array();\n      txExplain.outputAmount.should.equal('48739000');\n      txExplain.outputs.should.be.an.Array();\n      txExplain.outputs.length.should.equal(1);\n      txExplain.changeOutputs.should.be.empty();\n    });\n\n    it('should fail when transaction hex is not provided', async () => {\n      await basecoin.explainTransaction({}).should.be.rejectedWith('missing transaction hex');\n    });\n\n    it('should fail for invalid transaction hex', async () => {\n      await basecoin.explainTransaction({ txHex: 'invalid' }).should.be.rejected();\n    });\n  });\n\n  describe('Verify Transaction', () => {\n    it('should verify an export from C-chain transaction', async () => {\n      const txPrebuild = {\n        txHex: EXPORT_IN_C.signedHex,\n        txInfo: {},\n      };\n      const txParams = {\n        recipients: [\n          {\n            address: '',\n            amount: EXPORT_IN_C.amount,\n          },\n        ],\n        type: 'Export',\n        locktime: 0,\n      };\n\n      const isVerified = await basecoin.verifyTransaction({ txParams, txPrebuild });\n      isVerified.should.equal(true);\n    });\n\n    it('should verify an export from P-chain transaction', async () => {\n      const txPrebuild = {\n        txHex: EXPORT_IN_P.fullSigntxHex,\n        txInfo: {},\n      };\n      const txParams = {\n        recipients: [\n          {\n            address: '',\n            amount: EXPORT_IN_P.amount,\n          },\n        ],\n        type: 'Export',\n        locktime: 0,\n      };\n\n      const isVerified = await basecoin.verifyTransaction({ txParams, txPrebuild });\n      isVerified.should.equal(true);\n    });\n\n    it('should verify an import to C-chain transaction', async () => {\n      const txPrebuild = {\n        txHex: IMPORT_IN_C.fullSigntxHex,\n        txInfo: {},\n      };\n      const txParams = {\n        recipients: [\n          {\n            address: IMPORT_IN_C.to,\n            amount: '1',\n          },\n        ],\n        type: 'ImportToC',\n        locktime: 0,\n      };\n\n      const isVerified = await basecoin.verifyTransaction({ txParams, txPrebuild });\n      isVerified.should.equal(true);\n    });\n\n    it('should verify an import to P-chain transaction', async () => {\n      const txPrebuild = {\n        txHex: IMPORT_IN_P.fullSigntxHex,\n        txInfo: {},\n      };\n      const txParams = {\n        recipients: [],\n        type: 'Import',\n        locktime: 0,\n      };\n\n      const isVerified = await basecoin.verifyTransaction({ txParams, txPrebuild });\n      isVerified.should.equal(true);\n    });\n\n    it('should fail to verify export transaction with wrong amount', async () => {\n      const txPrebuild = {\n        txHex: EXPORT_IN_C.signedHex,\n        txInfo: {},\n      };\n      const txParams = {\n        recipients: [\n          {\n            address: '',\n            amount: '999999999999',\n          },\n        ],\n        type: 'Export',\n        locktime: 0,\n      };\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild })\n        .should.be.rejectedWith(/Tx total amount .* does not match with expected total amount/);\n    });\n\n    it('should fail to verify transaction with wrong type', async () => {\n      const txPrebuild = {\n        txHex: EXPORT_IN_C.signedHex,\n        txInfo: {},\n      };\n      const txParams = {\n        recipients: [\n          {\n            address: '',\n            amount: EXPORT_IN_C.amount,\n          },\n        ],\n        type: 'Import',\n        locktime: 0,\n      };\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild })\n        .should.be.rejectedWith('Tx type does not match with expected txParams type');\n    });\n\n    it('should fail to verify transaction without txHex', async () => {\n      const txPrebuild = {\n        txInfo: {},\n      };\n      const txParams = {\n        recipients: [],\n        type: 'Export',\n        locktime: 0,\n      };\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild })\n        .should.be.rejectedWith('missing required tx prebuild property txHex');\n    });\n\n    it('should fail to verify transaction with invalid txHex', async () => {\n      const txPrebuild = {\n        txHex: 'invalidhex',\n        txInfo: {},\n      };\n      const txParams = {\n        recipients: [],\n        type: 'Export',\n        locktime: 0,\n      };\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild })\n        .should.be.rejectedWith('Invalid transaction: Raw transaction is not hex string');\n    });\n\n    it('should fail to verify import to C-chain without recipients', async () => {\n      const txPrebuild = {\n        txHex: IMPORT_IN_C.fullSigntxHex,\n        txInfo: {},\n      };\n      const txParams = {\n        recipients: [],\n        type: 'ImportToC',\n        locktime: 0,\n      };\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild })\n        .should.be.rejectedWith('Expected 1 recipient in import transaction');\n    });\n  });\n\n  describe('Address Validation', () => {\n    it('should validate mainnet P-chain address', function () {\n      basecoin.isValidAddress(SEED_ACCOUNT.addressMainnet).should.be.true();\n    });\n\n    it('should validate testnet P-chain address', function () {\n      basecoin.isValidAddress(SEED_ACCOUNT.addressTestnet).should.be.true();\n    });\n\n    it('should validate array of P-chain addresses', function () {\n      basecoin.isValidAddress(EXPORT_IN_C.pAddresses).should.be.true();\n    });\n\n    it('should validate tilde-separated multisig address', function () {\n      const multiSigAddress = EXPORT_IN_C.pAddresses.join('~');\n      basecoin.isValidAddress(multiSigAddress).should.be.true();\n    });\n\n    it('should validate C-chain hex address', function () {\n      basecoin.isValidAddress(EXPORT_IN_C.cHexAddress).should.be.true();\n    });\n\n    it('should validate lowercase C-chain address', function () {\n      basecoin.isValidAddress(IMPORT_IN_C.to.toLowerCase()).should.be.true();\n    });\n\n    it('should fail to validate undefined address', function () {\n      basecoin.isValidAddress(undefined).should.be.false();\n    });\n\n    it('should fail to validate empty string', function () {\n      basecoin.isValidAddress('').should.be.false();\n    });\n\n    it('should fail to validate invalid address', function () {\n      basecoin.isValidAddress('invalid-address').should.be.false();\n    });\n\n    it('should fail to validate array with invalid address', function () {\n      const addresses = [...EXPORT_IN_C.pAddresses, 'invalid'];\n      basecoin.isValidAddress(addresses).should.be.false();\n    });\n  });\n\n  describe('Wallet Address Verification', () => {\n    it('should verify wallet address with matching keychains', async () => {\n      const keyPairs = [{ pub: SEED_ACCOUNT.publicKey }, { pub: ACCOUNT_1.publicKey }, { pub: ACCOUNT_2.publicKey }];\n\n      // Derive addresses from public keys to ensure they match\n      const derivedAddresses = keyPairs.map((kp) => new FlrpLib.KeyPair({ pub: kp.pub }).getAddress('testnet'));\n      const address = derivedAddresses.join('~');\n\n      const isValid = await basecoin.isWalletAddress({\n        address,\n        keychains: keyPairs,\n      });\n\n      isValid.should.be.true();\n    });\n\n    it('should throw for address with wrong number of keychains', async () => {\n      const address = SEED_ACCOUNT.addressTestnet;\n\n      await assert.rejects(\n        async () =>\n          basecoin.isWalletAddress({\n            address,\n            keychains: [{ pub: SEED_ACCOUNT.publicKey }],\n          }),\n        /Invalid keychains/\n      );\n    });\n\n    it('should throw for invalid address', async () => {\n      await assert.rejects(\n        async () =>\n          basecoin.isWalletAddress({\n            address: 'invalid',\n            keychains,\n          }),\n        /invalid address/\n      );\n    });\n\n    it('should throw when address length does not match keychain length', async () => {\n      const address = SEED_ACCOUNT.addressTestnet;\n\n      await assert.rejects(async () =>\n        basecoin.isWalletAddress({\n          address,\n          keychains,\n        })\n      );\n    });\n\n    it('should throw when addresses do not match keychains', async () => {\n      // Use addresses that don't match the keychains\n      const address = EXPORT_IN_C.pAddresses.join('~');\n\n      await assert.rejects(async () =>\n        basecoin.isWalletAddress({\n          address,\n          keychains,\n        })\n      );\n    });\n  });\n\n  describe('Recovery Signature', () => {\n    it('should recover signature from signed message', async () => {\n      const message = Buffer.from(SEED_ACCOUNT.message, 'utf8');\n      const privateKey = Buffer.from(SEED_ACCOUNT.privateKey, 'hex');\n      const signature = FlrpLib.Utils.createSignature(basecoin._staticsCoin.network, message, privateKey);\n      const messageHash = FlrpLib.Utils.sha256(message);\n      const recoveredPubKey = basecoin.recoverySignature(messageHash, signature);\n\n      recoveredPubKey.should.be.instanceOf(Buffer);\n      recoveredPubKey.length.should.equal(33);\n    });\n  });\n});\n"]}