@bitgo-beta/sdk-coin-flrp 1.0.1-beta.40 → 1.0.1-beta.401
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/flrp.d.ts +10 -17
- package/dist/src/flrp.d.ts.map +1 -1
- package/dist/src/flrp.js +51 -77
- package/dist/src/index.d.ts +0 -1
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +1 -2
- package/dist/src/lib/ExportInCTxBuilder.d.ts +43 -0
- package/dist/src/lib/ExportInCTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ExportInCTxBuilder.js +150 -0
- package/dist/src/lib/ExportInPTxBuilder.d.ts +28 -0
- package/dist/src/lib/ExportInPTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ExportInPTxBuilder.js +190 -0
- package/dist/src/lib/ImportInCTxBuilder.d.ts +34 -0
- package/dist/src/lib/ImportInCTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ImportInCTxBuilder.js +191 -0
- package/dist/src/lib/ImportInPTxBuilder.d.ts +38 -0
- package/dist/src/lib/ImportInPTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ImportInPTxBuilder.js +224 -0
- package/dist/src/lib/atomicInCTransactionBuilder.d.ts +12 -16
- package/dist/src/lib/atomicInCTransactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/atomicInCTransactionBuilder.js +30 -41
- package/dist/src/lib/atomicTransactionBuilder.d.ts +126 -69
- package/dist/src/lib/atomicTransactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/atomicTransactionBuilder.js +320 -211
- package/dist/src/lib/iface.d.ts +65 -57
- package/dist/src/lib/iface.d.ts.map +1 -1
- package/dist/src/lib/iface.js +20 -14
- package/dist/src/lib/index.d.ts +5 -0
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +12 -2
- package/dist/src/lib/keyPair.d.ts +5 -5
- package/dist/src/lib/keyPair.d.ts.map +1 -1
- package/dist/src/lib/keyPair.js +17 -9
- package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts +41 -0
- package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/permissionlessValidatorTxBuilder.js +126 -0
- package/dist/src/lib/transaction.d.ts +30 -66
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +347 -199
- package/dist/src/lib/transactionBuilder.d.ts +115 -0
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder.js +228 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts +50 -30
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilderFactory.js +129 -72
- package/dist/src/lib/utils.d.ts +131 -146
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +344 -321
- package/dist/test/resources/account.d.ts +81 -0
- package/dist/test/resources/account.d.ts.map +1 -0
- package/dist/test/resources/account.js +79 -0
- package/dist/test/resources/transactionData/exportInC.d.ts +50 -0
- package/dist/test/resources/transactionData/exportInC.d.ts.map +1 -0
- package/dist/test/resources/transactionData/exportInC.js +58 -0
- package/dist/test/resources/transactionData/exportInP.d.ts +60 -0
- package/dist/test/resources/transactionData/exportInP.d.ts.map +1 -0
- package/dist/test/resources/transactionData/exportInP.js +101 -0
- package/dist/test/resources/transactionData/importInC.d.ts +56 -0
- package/dist/test/resources/transactionData/importInC.d.ts.map +1 -0
- package/dist/test/resources/transactionData/importInC.js +120 -0
- package/dist/test/resources/transactionData/importInP.d.ts +66 -0
- package/dist/test/resources/transactionData/importInP.d.ts.map +1 -0
- package/dist/test/resources/transactionData/importInP.js +84 -0
- package/dist/test/unit/flrp.js +449 -68
- package/dist/test/unit/lib/exportInCTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/exportInCTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/exportInCTxBuilder.js +193 -0
- package/dist/test/unit/lib/exportInPTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/exportInPTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/exportInPTxBuilder.js +296 -0
- package/dist/test/unit/lib/importInCTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/importInCTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/importInCTxBuilder.js +309 -0
- package/dist/test/unit/lib/importInPTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/importInPTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/importInPTxBuilder.js +490 -0
- package/dist/test/unit/lib/keyPair.d.ts +2 -0
- package/dist/test/unit/lib/keyPair.d.ts.map +1 -0
- package/dist/test/unit/lib/keyPair.js +158 -0
- package/dist/test/unit/lib/signFlowTestSuit.d.ts +20 -0
- package/dist/test/unit/lib/signFlowTestSuit.d.ts.map +1 -0
- package/dist/test/unit/lib/signFlowTestSuit.js +83 -0
- package/dist/test/unit/lib/signatureIndex.d.ts +13 -0
- package/dist/test/unit/lib/signatureIndex.d.ts.map +1 -0
- package/dist/test/unit/lib/signatureIndex.js +1173 -0
- package/dist/test/unit/lib/transactionBuilderFactory.d.ts +2 -0
- package/dist/test/unit/lib/transactionBuilderFactory.d.ts.map +1 -0
- package/dist/test/unit/lib/transactionBuilderFactory.js +60 -0
- package/dist/test/unit/lib/utils.js +681 -206
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +20 -11
- package/.eslintignore +0 -5
- package/.eslintrc.json +0 -7
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -0
- package/dist/src/iface.d.ts +0 -25
- package/dist/src/iface.d.ts.map +0 -1
- package/dist/src/iface.js +0 -3
- package/dist/src/lib/constants.d.ts +0 -11
- package/dist/src/lib/constants.d.ts.map +0 -1
- package/dist/src/lib/constants.js +0 -17
- package/dist/src/lib/errors.d.ts +0 -8
- package/dist/src/lib/errors.d.ts.map +0 -1
- package/dist/src/lib/errors.js +0 -19
- package/dist/src/lib/exportInCTxBuilder.d.ts +0 -77
- package/dist/src/lib/exportInCTxBuilder.d.ts.map +0 -1
- package/dist/src/lib/exportInCTxBuilder.js +0 -170
- package/dist/src/lib/exportInPTxBuilder.d.ts +0 -30
- package/dist/src/lib/exportInPTxBuilder.d.ts.map +0 -1
- package/dist/src/lib/exportInPTxBuilder.js +0 -56
- package/dist/test/unit/lib/atomicTransactionBuilder.d.ts +0 -2
- package/dist/test/unit/lib/atomicTransactionBuilder.d.ts.map +0 -1
- package/dist/test/unit/lib/atomicTransactionBuilder.js +0 -222
- package/dist/test/unit/lib/exportTxBuilder.d.ts +0 -2
- package/dist/test/unit/lib/exportTxBuilder.d.ts.map +0 -1
- package/dist/test/unit/lib/exportTxBuilder.js +0 -45
- package/dist/test/unit/lib/transaction.d.ts +0 -2
- package/dist/test/unit/lib/transaction.d.ts.map +0 -1
- package/dist/test/unit/lib/transaction.js +0 -460
- package/dist/test/unit/smoke.d.ts +0 -2
- package/dist/test/unit/smoke.d.ts.map +0 -1
- package/dist/test/unit/smoke.js +0 -23
package/dist/src/lib/utils.js
CHANGED
|
@@ -1,27 +1,79 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
2
5
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
6
|
exports.Utils = void 0;
|
|
7
|
+
const flarejs_1 = require("@flarenetwork/flarejs");
|
|
4
8
|
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
5
|
-
const
|
|
9
|
+
const buffer_1 = require("buffer");
|
|
6
10
|
const crypto_1 = require("crypto");
|
|
7
|
-
const
|
|
11
|
+
const secp256k1_1 = require("@bitgo-beta/secp256k1");
|
|
12
|
+
const iface_1 = require("./iface");
|
|
13
|
+
const bs58_1 = __importDefault(require("bs58"));
|
|
14
|
+
const bech32_1 = require("bech32");
|
|
8
15
|
class Utils {
|
|
9
16
|
constructor() {
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
17
|
+
/**
|
|
18
|
+
* Helper method to convert address components to string
|
|
19
|
+
*/
|
|
20
|
+
this.addressToString = (hrp, prefix, address) => {
|
|
21
|
+
// Convert the address bytes to 5-bit words for bech32 encoding
|
|
22
|
+
const words = bech32_1.bech32.toWords(address);
|
|
23
|
+
// Create the full bech32 address with format: P-{hrp}1{bech32_encoded_address}
|
|
24
|
+
return `${prefix}-${bech32_1.bech32.encode(hrp, words)}`;
|
|
25
|
+
};
|
|
26
|
+
// In utils.ts, add this method to the Utils class:
|
|
27
|
+
/**
|
|
28
|
+
* Parse an address string into a Buffer
|
|
29
|
+
* @param address - The address to parse
|
|
30
|
+
* @returns Buffer containing the parsed address
|
|
31
|
+
*/
|
|
32
|
+
this.parseAddress = (address) => {
|
|
33
|
+
return this.stringToAddress(address);
|
|
34
|
+
};
|
|
35
|
+
this.stringToAddress = (address, hrp) => {
|
|
36
|
+
// Handle hex addresses
|
|
37
|
+
if (address.startsWith('0x')) {
|
|
38
|
+
return buffer_1.Buffer.from(address.slice(2), 'hex');
|
|
39
|
+
}
|
|
40
|
+
// Handle raw hex without 0x prefix
|
|
41
|
+
if (/^[0-9a-fA-F]{40}$/.test(address)) {
|
|
42
|
+
return buffer_1.Buffer.from(address, 'hex');
|
|
43
|
+
}
|
|
44
|
+
// Handle Bech32 addresses
|
|
45
|
+
const parts = address.trim().split('-');
|
|
46
|
+
if (parts.length < 2) {
|
|
47
|
+
throw new Error('Error - Valid address should include -');
|
|
48
|
+
}
|
|
49
|
+
const split = parts[1].lastIndexOf('1');
|
|
50
|
+
if (split < 0) {
|
|
51
|
+
throw new Error('Error - Valid bech32 address must include separator (1)');
|
|
52
|
+
}
|
|
53
|
+
const humanReadablePart = parts[1].slice(0, split);
|
|
54
|
+
if (humanReadablePart !== 'flare' && humanReadablePart !== 'costwo') {
|
|
55
|
+
throw new Error('Error - Invalid HRP');
|
|
56
|
+
}
|
|
57
|
+
return buffer_1.Buffer.from(bech32_1.bech32.fromWords(bech32_1.bech32.decode(parts[1]).words));
|
|
13
58
|
};
|
|
14
59
|
}
|
|
60
|
+
isValidTransactionId(txId) {
|
|
61
|
+
throw new Error('Method not implemented.');
|
|
62
|
+
}
|
|
63
|
+
isValidSignature(signature) {
|
|
64
|
+
throw new Error('Method not implemented.');
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Check if addresses in wallet match UTXO output addresses
|
|
68
|
+
*/
|
|
15
69
|
includeIn(walletAddresses, otxoOutputAddresses) {
|
|
16
70
|
return walletAddresses.map((a) => otxoOutputAddresses.includes(a)).reduce((a, b) => a && b, true);
|
|
17
71
|
}
|
|
18
72
|
/**
|
|
19
|
-
*
|
|
20
|
-
*
|
|
21
|
-
* @
|
|
22
|
-
* @returns {boolean} - the validation result
|
|
73
|
+
* Validates a Flare address or array of addresses
|
|
74
|
+
* @param {string | string[]} address - address(es) to validate
|
|
75
|
+
* @returns {boolean} - validation result
|
|
23
76
|
*/
|
|
24
|
-
/** @inheritdoc */
|
|
25
77
|
isValidAddress(address) {
|
|
26
78
|
const addressArr = Array.isArray(address) ? address : address.split('~');
|
|
27
79
|
for (const address of addressArr) {
|
|
@@ -32,65 +84,53 @@ class Utils {
|
|
|
32
84
|
return true;
|
|
33
85
|
}
|
|
34
86
|
isValidAddressRegex(address) {
|
|
35
|
-
return
|
|
87
|
+
return /^(^P||NodeID)-[a-zA-Z0-9]+$/.test(address);
|
|
36
88
|
}
|
|
37
89
|
/**
|
|
38
|
-
*
|
|
39
|
-
*
|
|
40
|
-
* @
|
|
41
|
-
* @returns {boolean} - the validation result
|
|
90
|
+
* Validates a block ID
|
|
91
|
+
* @param {string} hash - block ID to validate
|
|
92
|
+
* @returns {boolean} - validation result
|
|
42
93
|
*/
|
|
43
|
-
/** @inheritdoc */
|
|
44
94
|
isValidBlockId(hash) {
|
|
45
|
-
// FlareJS equivalent - check if it's a valid CB58 hash with correct length
|
|
46
95
|
try {
|
|
47
|
-
const decoded = Buffer.from(hash);
|
|
48
|
-
return decoded.length ===
|
|
96
|
+
const decoded = buffer_1.Buffer.from(hash, 'hex');
|
|
97
|
+
return decoded.length === 32;
|
|
49
98
|
}
|
|
50
99
|
catch {
|
|
51
100
|
return false;
|
|
52
101
|
}
|
|
53
102
|
}
|
|
54
103
|
/**
|
|
55
|
-
*
|
|
56
|
-
*
|
|
57
|
-
*
|
|
58
|
-
* @param {string} pub - the public key to be validated
|
|
59
|
-
* @returns {boolean} - the validation result
|
|
104
|
+
* Validates a public key
|
|
105
|
+
* @param {string} pub - public key to validate
|
|
106
|
+
* @returns {boolean} - validation result
|
|
60
107
|
*/
|
|
61
108
|
isValidPublicKey(pub) {
|
|
62
109
|
if ((0, sdk_core_1.isValidXpub)(pub))
|
|
63
110
|
return true;
|
|
64
111
|
let pubBuf;
|
|
65
|
-
if (pub.length ===
|
|
112
|
+
if (pub.length === 50) {
|
|
66
113
|
try {
|
|
67
|
-
|
|
68
|
-
pubBuf = Buffer.from(pub, 'hex'); // Temporary placeholder
|
|
114
|
+
pubBuf = buffer_1.Buffer.from(pub, 'hex');
|
|
69
115
|
}
|
|
70
116
|
catch {
|
|
71
117
|
return false;
|
|
72
118
|
}
|
|
73
119
|
}
|
|
74
120
|
else {
|
|
75
|
-
if (pub.length !==
|
|
121
|
+
if (pub.length !== 66 && pub.length !== 130)
|
|
76
122
|
return false;
|
|
77
|
-
}
|
|
78
123
|
const firstByte = pub.slice(0, 2);
|
|
79
|
-
|
|
80
|
-
if (pub.length === constants_1.UNCOMPRESSED_PUBLIC_KEY_LENGTH && firstByte !== '04') {
|
|
124
|
+
if (pub.length === 130 && firstByte !== '04')
|
|
81
125
|
return false;
|
|
82
|
-
|
|
83
|
-
// compressed public key
|
|
84
|
-
if (pub.length === constants_1.COMPRESSED_PUBLIC_KEY_LENGTH && firstByte !== '02' && firstByte !== '03') {
|
|
126
|
+
if (pub.length === 66 && firstByte !== '02' && firstByte !== '03')
|
|
85
127
|
return false;
|
|
86
|
-
}
|
|
87
128
|
if (!this.allHexChars(pub))
|
|
88
129
|
return false;
|
|
89
|
-
pubBuf = Buffer.from(pub, 'hex');
|
|
130
|
+
pubBuf = buffer_1.Buffer.from(pub, 'hex');
|
|
90
131
|
}
|
|
91
|
-
// validate the public key using BitGo secp256k1
|
|
92
132
|
try {
|
|
93
|
-
secp256k1_1.ecc.isPoint(pubBuf);
|
|
133
|
+
secp256k1_1.ecc.isPoint(pubBuf);
|
|
94
134
|
return true;
|
|
95
135
|
}
|
|
96
136
|
catch (e) {
|
|
@@ -98,269 +138,134 @@ class Utils {
|
|
|
98
138
|
}
|
|
99
139
|
}
|
|
100
140
|
/**
|
|
101
|
-
*
|
|
102
|
-
* private key
|
|
103
|
-
*
|
|
104
|
-
* The protocol key format is described in the @stacks/transactions npm package, in the
|
|
105
|
-
* createStacksPrivateKey function:
|
|
106
|
-
* https://github.com/blockstack/stacks.js/blob/master/packages/transactions/src/keys.ts#L125
|
|
107
|
-
*
|
|
108
|
-
* @param {string} prv - the private key (or extended private key) to be validated
|
|
109
|
-
* @returns {boolean} - the validation result
|
|
141
|
+
* Validates a private key
|
|
142
|
+
* @param {string} prv - private key to validate
|
|
143
|
+
* @returns {boolean} - validation result
|
|
110
144
|
*/
|
|
111
145
|
isValidPrivateKey(prv) {
|
|
112
146
|
if ((0, sdk_core_1.isValidXprv)(prv))
|
|
113
147
|
return true;
|
|
114
|
-
if (prv.length !==
|
|
148
|
+
if (prv.length !== 64 && prv.length !== 66)
|
|
115
149
|
return false;
|
|
116
|
-
|
|
117
|
-
if (prv.length === constants_1.SUFFIXED_PRIVATE_KEY_LENGTH &&
|
|
118
|
-
prv.slice(constants_1.RAW_PRIVATE_KEY_LENGTH) !== constants_1.PRIVATE_KEY_COMPRESSED_SUFFIX) {
|
|
150
|
+
if (prv.length === 66 && prv.slice(64) !== '01')
|
|
119
151
|
return false;
|
|
120
|
-
}
|
|
121
152
|
return this.allHexChars(prv);
|
|
122
153
|
}
|
|
123
154
|
/**
|
|
124
|
-
*
|
|
125
|
-
*
|
|
126
|
-
* @param {string} maybe - the string to be validated
|
|
127
|
-
* @returns {boolean} - the validation result
|
|
155
|
+
* Checks if a string contains only hex characters
|
|
128
156
|
*/
|
|
129
|
-
allHexChars(
|
|
130
|
-
return
|
|
157
|
+
allHexChars(str) {
|
|
158
|
+
return /^(0x){0,1}([0-9a-f])+$/i.test(str);
|
|
131
159
|
}
|
|
132
160
|
/**
|
|
133
|
-
*
|
|
134
|
-
*
|
|
135
|
-
*
|
|
136
|
-
* @param {string} address - the Ethereum address to validate
|
|
137
|
-
* @returns {boolean} - true if valid Ethereum address format
|
|
161
|
+
* Creates a signature using the Flare network parameters
|
|
162
|
+
* Returns a 65-byte signature (64 bytes signature + 1 byte recovery parameter)
|
|
138
163
|
*/
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
164
|
+
createSignature(network, message, prv) {
|
|
165
|
+
const messageHash = this.sha256(message);
|
|
166
|
+
const signature = secp256k1_1.ecc.sign(messageHash, prv);
|
|
167
|
+
// Get the public key from the private key for recovery parameter determination
|
|
168
|
+
const publicKey = secp256k1_1.ecc.pointFromScalar(prv, true);
|
|
169
|
+
if (!publicKey) {
|
|
170
|
+
throw new Error('Failed to derive public key from private key');
|
|
142
171
|
}
|
|
143
|
-
//
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
*
|
|
151
|
-
* @param {T} obj - the source object
|
|
152
|
-
* @param {K[]} keys - array of property keys to pick
|
|
153
|
-
* @returns {Pick<T, K>} - new object with only the specified properties
|
|
154
|
-
*/
|
|
155
|
-
pick(obj, keys) {
|
|
156
|
-
const result = {};
|
|
157
|
-
for (const key of keys) {
|
|
158
|
-
if (Object.prototype.hasOwnProperty.call(obj, key)) {
|
|
159
|
-
result[key] = obj[key];
|
|
172
|
+
// Try recovery with param 0 and 1 to find the correct one
|
|
173
|
+
let recoveryParam = 0;
|
|
174
|
+
for (let i = 0; i <= 1; i++) {
|
|
175
|
+
const recovered = secp256k1_1.ecc.recoverPublicKey(messageHash, signature, i, true);
|
|
176
|
+
if (recovered && buffer_1.Buffer.from(recovered).equals(buffer_1.Buffer.from(publicKey))) {
|
|
177
|
+
recoveryParam = i;
|
|
178
|
+
break;
|
|
160
179
|
}
|
|
161
180
|
}
|
|
162
|
-
|
|
181
|
+
// Append recovery parameter to create 65-byte signature
|
|
182
|
+
const sigWithRecovery = buffer_1.Buffer.alloc(65);
|
|
183
|
+
buffer_1.Buffer.from(signature).copy(sigWithRecovery, 0);
|
|
184
|
+
sigWithRecovery[64] = recoveryParam;
|
|
185
|
+
return sigWithRecovery;
|
|
163
186
|
}
|
|
164
187
|
/**
|
|
165
|
-
*
|
|
166
|
-
*
|
|
167
|
-
* @param
|
|
168
|
-
* @param
|
|
169
|
-
* @returns
|
|
188
|
+
* Verifies a signature
|
|
189
|
+
* @param messageHash - The SHA256 hash of the message (e.g., signablePayload)
|
|
190
|
+
* @param signature - The 64-byte signature (without recovery parameter)
|
|
191
|
+
* @param publicKey - The public key to verify against
|
|
192
|
+
* @returns true if signature is valid
|
|
170
193
|
*/
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
return
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
if (typeof a !== typeof b)
|
|
194
|
+
verifySignature(messageHash, signature, publicKey) {
|
|
195
|
+
try {
|
|
196
|
+
return secp256k1_1.ecc.verify(messageHash, publicKey, signature);
|
|
197
|
+
}
|
|
198
|
+
catch (e) {
|
|
177
199
|
return false;
|
|
178
|
-
if (typeof a === 'object') {
|
|
179
|
-
if (Array.isArray(a) !== Array.isArray(b))
|
|
180
|
-
return false;
|
|
181
|
-
if (Array.isArray(a)) {
|
|
182
|
-
const arrB = b;
|
|
183
|
-
if (a.length !== arrB.length)
|
|
184
|
-
return false;
|
|
185
|
-
for (let i = 0; i < a.length; i++) {
|
|
186
|
-
if (!this.isEqual(a[i], arrB[i]))
|
|
187
|
-
return false;
|
|
188
|
-
}
|
|
189
|
-
return true;
|
|
190
|
-
}
|
|
191
|
-
const objA = a;
|
|
192
|
-
const objB = b;
|
|
193
|
-
const keysA = Object.keys(objA);
|
|
194
|
-
const keysB = Object.keys(objB);
|
|
195
|
-
if (keysA.length !== keysB.length)
|
|
196
|
-
return false;
|
|
197
|
-
for (const key of keysA) {
|
|
198
|
-
if (!keysB.includes(key))
|
|
199
|
-
return false;
|
|
200
|
-
if (!this.isEqual(objA[key], objB[key]))
|
|
201
|
-
return false;
|
|
202
|
-
}
|
|
203
|
-
return true;
|
|
204
200
|
}
|
|
205
|
-
return false;
|
|
206
|
-
}
|
|
207
|
-
/** @inheritdoc */
|
|
208
|
-
isValidSignature(signature) {
|
|
209
|
-
throw new sdk_core_1.NotImplementedError('isValidSignature not implemented');
|
|
210
|
-
}
|
|
211
|
-
/** @inheritdoc */
|
|
212
|
-
isValidTransactionId(txId) {
|
|
213
|
-
throw new sdk_core_1.NotImplementedError('isValidTransactionId not implemented');
|
|
214
201
|
}
|
|
215
202
|
/**
|
|
216
|
-
*
|
|
217
|
-
* @param network
|
|
218
|
-
* @param message
|
|
219
|
-
* @param prv
|
|
220
|
-
* @return signature
|
|
203
|
+
* Creates a new signature object
|
|
221
204
|
*/
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
const signature = secp256k1_1.ecc.sign(message, prv);
|
|
226
|
-
return Buffer.from(signature);
|
|
227
|
-
}
|
|
228
|
-
catch (error) {
|
|
229
|
-
throw new Error(`Failed to create signature: ${error}`);
|
|
230
|
-
}
|
|
205
|
+
createNewSig(sigHex) {
|
|
206
|
+
const buffer = buffer_1.Buffer.from(sigHex.padStart(130, '0'), 'hex');
|
|
207
|
+
return new flarejs_1.Signature(buffer);
|
|
231
208
|
}
|
|
232
209
|
/**
|
|
233
|
-
*
|
|
234
|
-
*
|
|
235
|
-
*
|
|
236
|
-
* @param
|
|
237
|
-
* @param publicKey - public key instead of private key for verification
|
|
238
|
-
* @return true if it's verify successful
|
|
210
|
+
* Creates an empty signature with embedded address for signature slot identification.
|
|
211
|
+
* The address is embedded at position 90 (after the first 45 zero bytes).
|
|
212
|
+
* This allows the signing logic to determine which slot belongs to which address.
|
|
213
|
+
* @param addressHex The 20-byte address in hex format (40 chars, without 0x prefix)
|
|
239
214
|
*/
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
}
|
|
215
|
+
createEmptySigWithAddress(addressHex) {
|
|
216
|
+
// First 45 bytes (90 hex chars) are zeros, followed by 20-byte address (40 hex chars)
|
|
217
|
+
const cleanAddr = this.removeHexPrefix(addressHex).toLowerCase();
|
|
218
|
+
const sigHex = '0'.repeat(90) + cleanAddr.padStart(40, '0');
|
|
219
|
+
const buffer = buffer_1.Buffer.from(sigHex, 'hex');
|
|
220
|
+
return new flarejs_1.Signature(buffer);
|
|
247
221
|
}
|
|
248
222
|
/**
|
|
249
|
-
*
|
|
250
|
-
*
|
|
251
|
-
* @param message
|
|
252
|
-
* @param signature
|
|
253
|
-
* @return recovered public key
|
|
223
|
+
* Extracts the embedded address from an empty signature.
|
|
224
|
+
* Returns the address hex string (40 chars) or empty string if not found.
|
|
254
225
|
*/
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
//
|
|
259
|
-
|
|
260
|
-
}
|
|
261
|
-
catch (error) {
|
|
262
|
-
throw new Error(`Failed to recover signature: ${error}`);
|
|
226
|
+
getAddressFromEmptySig(sig) {
|
|
227
|
+
const cleanSig = this.removeHexPrefix(sig);
|
|
228
|
+
if (cleanSig.length >= 130) {
|
|
229
|
+
// Address is at position 90-130 (last 40 hex chars = 20 bytes)
|
|
230
|
+
return cleanSig.substring(90, 130).toLowerCase();
|
|
263
231
|
}
|
|
232
|
+
return '';
|
|
264
233
|
}
|
|
234
|
+
/**
|
|
235
|
+
* Computes SHA256 hash
|
|
236
|
+
*/
|
|
265
237
|
sha256(buf) {
|
|
266
238
|
return (0, crypto_1.createHash)('sha256').update(buf).digest();
|
|
267
239
|
}
|
|
268
240
|
/**
|
|
269
|
-
*
|
|
270
|
-
* It's to reuse in TransactionBuilder and TransactionBuilderFactory
|
|
271
|
-
*
|
|
272
|
-
* @param rawTransaction Transaction as hex string
|
|
241
|
+
* Validates raw transaction format
|
|
273
242
|
*/
|
|
274
243
|
validateRawTransaction(rawTransaction) {
|
|
275
244
|
if (!rawTransaction) {
|
|
276
245
|
throw new sdk_core_1.InvalidTransactionError('Raw transaction is empty');
|
|
277
246
|
}
|
|
278
|
-
if (!
|
|
247
|
+
if (!this.allHexChars(rawTransaction)) {
|
|
279
248
|
throw new sdk_core_1.ParseTransactionError('Raw transaction is not hex string');
|
|
280
249
|
}
|
|
281
250
|
}
|
|
282
251
|
/**
|
|
283
|
-
*
|
|
284
|
-
*
|
|
285
|
-
* @param {DeprecatedTx} tx
|
|
286
|
-
* @param {string} blockchainId
|
|
287
|
-
* @returns true if tx is for blockchainId
|
|
288
|
-
*/
|
|
289
|
-
isTransactionOf(tx, blockchainId) {
|
|
290
|
-
// FlareJS equivalent - this would need proper CB58 encoding implementation
|
|
291
|
-
try {
|
|
292
|
-
const txRecord = tx;
|
|
293
|
-
const unsignedTx = txRecord.getUnsignedTx();
|
|
294
|
-
const transaction = unsignedTx.getTransaction();
|
|
295
|
-
const txBlockchainId = transaction.getBlockchainID();
|
|
296
|
-
return Buffer.from(txBlockchainId).toString('hex') === blockchainId;
|
|
297
|
-
}
|
|
298
|
-
catch (error) {
|
|
299
|
-
return false;
|
|
300
|
-
}
|
|
301
|
-
}
|
|
302
|
-
/**
|
|
303
|
-
* Check if Output is from PVM.
|
|
304
|
-
* Output could be EVM or PVM output.
|
|
305
|
-
* @param {DeprecatedOutput} output
|
|
306
|
-
* @returns {boolean} output has transferable output structure
|
|
307
|
-
*/
|
|
308
|
-
deprecatedIsTransferableOutput(output) {
|
|
309
|
-
return 'getOutput' in output;
|
|
310
|
-
}
|
|
311
|
-
/**
|
|
312
|
-
* Check if Output is from PVM.
|
|
313
|
-
* Output could be EVM or PVM output.
|
|
314
|
-
* @param {Output} output
|
|
315
|
-
* @returns {boolean} output is TransferableOutput
|
|
252
|
+
* Checks if output is TransferableOutput type
|
|
316
253
|
*/
|
|
317
254
|
isTransferableOutput(output) {
|
|
318
|
-
return
|
|
319
|
-
}
|
|
320
|
-
/**
|
|
321
|
-
* Return a mapper function to that network address representation.
|
|
322
|
-
* @param network required to stringify addresses
|
|
323
|
-
* @return mapper function
|
|
324
|
-
*/
|
|
325
|
-
deprecatedMapOutputToEntry(network) {
|
|
326
|
-
return (output) => {
|
|
327
|
-
if (this.deprecatedIsTransferableOutput(output)) {
|
|
328
|
-
// Simplified implementation for FlareJS
|
|
329
|
-
try {
|
|
330
|
-
const transferableOutput = output;
|
|
331
|
-
const amount = transferableOutput.amount();
|
|
332
|
-
// Simplified address handling - would need proper FlareJS address utilities
|
|
333
|
-
const address = 'flare-address-placeholder'; // TODO: implement proper address conversion
|
|
334
|
-
return {
|
|
335
|
-
value: amount.toString(),
|
|
336
|
-
address,
|
|
337
|
-
};
|
|
338
|
-
}
|
|
339
|
-
catch (error) {
|
|
340
|
-
throw new Error(`Failed to map output: ${error}`);
|
|
341
|
-
}
|
|
342
|
-
}
|
|
343
|
-
else {
|
|
344
|
-
// Handle EVM output case - simplified
|
|
345
|
-
return {
|
|
346
|
-
value: '0', // TODO: implement proper amount extraction
|
|
347
|
-
address: '0x0000000000000000000000000000000000000000', // TODO: implement proper address extraction
|
|
348
|
-
};
|
|
349
|
-
}
|
|
350
|
-
};
|
|
255
|
+
return output?._type === flarejs_1.TypeSymbols.TransferableOutput;
|
|
351
256
|
}
|
|
352
257
|
/**
|
|
353
|
-
*
|
|
354
|
-
* @param network required to stringify addresses
|
|
355
|
-
* @return mapper function
|
|
258
|
+
* Maps outputs to entry format
|
|
356
259
|
*/
|
|
357
260
|
mapOutputToEntry(network) {
|
|
358
261
|
return (output) => {
|
|
359
262
|
if (this.isTransferableOutput(output)) {
|
|
360
|
-
const
|
|
361
|
-
const
|
|
362
|
-
|
|
363
|
-
|
|
263
|
+
const outputAmount = output.amount();
|
|
264
|
+
const address = output.output
|
|
265
|
+
.getOwners()
|
|
266
|
+
.map((a) => this.addressToString(network.hrp, network.alias, buffer_1.Buffer.from(a)))
|
|
267
|
+
.sort()
|
|
268
|
+
.join(iface_1.ADDRESS_SEPARATOR);
|
|
364
269
|
return {
|
|
365
270
|
value: outputAmount.toString(),
|
|
366
271
|
address,
|
|
@@ -372,116 +277,234 @@ class Utils {
|
|
|
372
277
|
};
|
|
373
278
|
}
|
|
374
279
|
/**
|
|
375
|
-
*
|
|
376
|
-
* @param hex string
|
|
377
|
-
* @returns hex without 0x
|
|
280
|
+
* Removes 0x prefix from hex string
|
|
378
281
|
*/
|
|
379
282
|
removeHexPrefix(hex) {
|
|
380
|
-
|
|
381
|
-
return hex.substring(2);
|
|
382
|
-
}
|
|
383
|
-
return hex;
|
|
283
|
+
return hex.startsWith('0x') ? hex.substring(2) : hex;
|
|
384
284
|
}
|
|
385
285
|
/**
|
|
386
|
-
*
|
|
387
|
-
* @param {string} outputidx number
|
|
388
|
-
* @return {Buffer} buffer of size 4 with that number value
|
|
286
|
+
* Converts output index to buffer
|
|
389
287
|
*/
|
|
390
288
|
outputidxNumberToBuffer(outputidx) {
|
|
391
|
-
return Buffer.from(Number(outputidx).toString(16).padStart(
|
|
289
|
+
return buffer_1.Buffer.from(Number(outputidx).toString(16).padStart(8, '0'), 'hex');
|
|
392
290
|
}
|
|
393
291
|
/**
|
|
394
|
-
*
|
|
395
|
-
* @param {Buffer} outputidx
|
|
396
|
-
* @return {string} outputidx number
|
|
292
|
+
* Converts output index buffer to number string
|
|
397
293
|
*/
|
|
398
294
|
outputidxBufferToNumber(outputidx) {
|
|
399
295
|
return parseInt(outputidx.toString('hex'), 16).toString();
|
|
400
296
|
}
|
|
401
297
|
/**
|
|
402
|
-
*
|
|
403
|
-
* @param {string} data - CB58 encoded string
|
|
404
|
-
* @returns {Buffer} decoded buffer
|
|
298
|
+
* Decodes a base58 string with checksum to a Buffer
|
|
405
299
|
*/
|
|
406
|
-
cb58Decode(
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
return Buffer.from(data, 'hex');
|
|
411
|
-
}
|
|
412
|
-
catch {
|
|
413
|
-
// Fallback to buffer from string
|
|
414
|
-
return Buffer.from(data);
|
|
300
|
+
cb58Decode(str) {
|
|
301
|
+
const decoded = bs58_1.default.decode(str);
|
|
302
|
+
if (!this.validateChecksum(buffer_1.Buffer.from(decoded))) {
|
|
303
|
+
throw new Error('Invalid checksum');
|
|
415
304
|
}
|
|
305
|
+
return buffer_1.Buffer.from(decoded.slice(0, decoded.length - 4));
|
|
416
306
|
}
|
|
417
307
|
/**
|
|
418
|
-
*
|
|
419
|
-
* @param {string} hrp - Human readable part
|
|
420
|
-
* @param {string} chainid - Chain identifier
|
|
421
|
-
* @param {Buffer} addressBuffer - Address buffer
|
|
422
|
-
* @returns {string} Address string
|
|
308
|
+
* Validates a checksum on a Buffer and returns true if valid, false if not
|
|
423
309
|
*/
|
|
424
|
-
|
|
425
|
-
|
|
426
|
-
|
|
310
|
+
validateChecksum(buff) {
|
|
311
|
+
const hashSlice = buff.slice(buff.length - 4);
|
|
312
|
+
const calculatedHashSlice = (0, crypto_1.createHash)('sha256')
|
|
313
|
+
.update(buff.slice(0, buff.length - 4))
|
|
314
|
+
.digest()
|
|
315
|
+
.slice(28);
|
|
316
|
+
return hashSlice.toString('hex') === calculatedHashSlice.toString('hex');
|
|
427
317
|
}
|
|
428
318
|
/**
|
|
429
|
-
*
|
|
430
|
-
* Follows FlareJS utils.stringToBytes pattern
|
|
431
|
-
* @param {string} text - Text to convert
|
|
432
|
-
* @returns {Uint8Array} Byte array
|
|
319
|
+
* Encodes a Buffer as a base58 string with checksum
|
|
433
320
|
*/
|
|
434
|
-
|
|
435
|
-
|
|
321
|
+
cb58Encode(bytes) {
|
|
322
|
+
const withChecksum = this.addChecksum(bytes);
|
|
323
|
+
return bs58_1.default.encode(withChecksum);
|
|
436
324
|
}
|
|
437
325
|
/**
|
|
438
|
-
*
|
|
439
|
-
*
|
|
440
|
-
* @returns {string} Decoded string
|
|
326
|
+
* Adds a checksum to a Buffer and returns the concatenated result
|
|
327
|
+
* Uses last 4 bytes of SHA256 hash as checksum (matching avaxp behavior)
|
|
441
328
|
*/
|
|
442
|
-
|
|
443
|
-
|
|
329
|
+
addChecksum(buff) {
|
|
330
|
+
const buffer = buffer_1.Buffer.from(buff);
|
|
331
|
+
const hashSlice = (0, crypto_1.createHash)('sha256').update(buffer).digest().slice(28);
|
|
332
|
+
return new Uint8Array(buffer_1.Buffer.concat([buffer, hashSlice]));
|
|
333
|
+
}
|
|
334
|
+
flareIdString(value) {
|
|
335
|
+
return new flarejs_1.Id(buffer_1.Buffer.from(value, 'hex'));
|
|
444
336
|
}
|
|
445
337
|
/**
|
|
446
|
-
*
|
|
447
|
-
*
|
|
448
|
-
*
|
|
449
|
-
*
|
|
338
|
+
* Sort addresses lexicographically by their byte representation.
|
|
339
|
+
* This matches how addresses are stored on-chain in Avalanche/Flare P-chain UTXOs.
|
|
340
|
+
*
|
|
341
|
+
* IMPORTANT: This sorting MUST be consistent with FlareJS's internal address sorting.
|
|
342
|
+
* FlareJS uses the same lexicographic comparison: `hexA.localeCompare(hexB)`.
|
|
343
|
+
* The sigIndices in transaction inputs depend on this sorted order, so any deviation
|
|
344
|
+
* would cause signature order mismatches and on-chain verification failures.
|
|
345
|
+
*
|
|
346
|
+
* @param addresses - Array of bech32 address strings (e.g., "P-costwo1...")
|
|
347
|
+
* @returns Array of addresses sorted by hex value (ascending lexicographic order)
|
|
450
348
|
*/
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
349
|
+
sortAddressesByHex(addresses) {
|
|
350
|
+
return [...addresses].sort((a, b) => {
|
|
351
|
+
const aHex = this.parseAddress(a).toString('hex');
|
|
352
|
+
const bHex = this.parseAddress(b).toString('hex');
|
|
353
|
+
return aHex.localeCompare(bHex);
|
|
354
|
+
});
|
|
355
|
+
}
|
|
356
|
+
/**
|
|
357
|
+
* Sort address buffers lexicographically by their byte representation.
|
|
358
|
+
* This matches how addresses are stored on-chain in Avalanche/Flare P-chain UTXOs.
|
|
359
|
+
* @param addressBuffers - Array of address byte buffers
|
|
360
|
+
* @returns Array of address buffers sorted by hex value
|
|
361
|
+
*/
|
|
362
|
+
sortAddressBuffersByHex(addressBuffers) {
|
|
363
|
+
return [...addressBuffers].sort((a, b) => {
|
|
364
|
+
return a.toString('hex').localeCompare(b.toString('hex'));
|
|
365
|
+
});
|
|
366
|
+
}
|
|
367
|
+
/**
|
|
368
|
+
* Recover public key from signature
|
|
369
|
+
* @param messageHash - The SHA256 hash of the message (e.g., signablePayload)
|
|
370
|
+
* @param signature - 65-byte signature (64 bytes signature + 1 byte recovery parameter)
|
|
371
|
+
* @return recovered public key
|
|
372
|
+
*/
|
|
373
|
+
recoverySignature(messageHash, signature) {
|
|
374
|
+
try {
|
|
375
|
+
// Extract recovery parameter and signature
|
|
376
|
+
if (signature.length !== 65) {
|
|
377
|
+
throw new Error('Invalid signature length - expected 65 bytes (64 bytes signature + 1 byte recovery)');
|
|
378
|
+
}
|
|
379
|
+
const recoveryParam = signature[64];
|
|
380
|
+
const sigOnly = signature.slice(0, 64);
|
|
381
|
+
// Recover public key using the provided recovery parameter
|
|
382
|
+
// messageHash should already be the SHA256 hash (signablePayload)
|
|
383
|
+
const recovered = secp256k1_1.ecc.recoverPublicKey(messageHash, sigOnly, recoveryParam, true);
|
|
384
|
+
if (!recovered) {
|
|
385
|
+
throw new Error('Failed to recover public key');
|
|
386
|
+
}
|
|
387
|
+
return buffer_1.Buffer.from(recovered);
|
|
457
388
|
}
|
|
458
|
-
|
|
459
|
-
|
|
389
|
+
catch (error) {
|
|
390
|
+
throw new Error(`Failed to recover signature: ${error.message}`);
|
|
460
391
|
}
|
|
461
|
-
throw new sdk_core_1.InvalidTransactionError('Invalid memo format');
|
|
462
392
|
}
|
|
463
393
|
/**
|
|
464
|
-
*
|
|
465
|
-
*
|
|
466
|
-
* @
|
|
394
|
+
* Check if tx is for the blockchainId
|
|
395
|
+
*
|
|
396
|
+
* @param {Tx} tx
|
|
397
|
+
* @param {string} blockchainId - blockchain ID in hex format
|
|
398
|
+
* @returns true if tx is for blockchainId
|
|
467
399
|
*/
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
|
|
400
|
+
isTransactionOf(tx, blockchainId) {
|
|
401
|
+
// Note: getBlockchainId() and BlockchainId.value() return CB58-encoded strings,
|
|
402
|
+
// but we need hex format, so we use toBytes() and convert to hex
|
|
403
|
+
const extractBlockchainId = (txObj) => {
|
|
404
|
+
if (typeof txObj.getTx === 'function') {
|
|
405
|
+
const innerTx = txObj.getTx();
|
|
406
|
+
if (innerTx.baseTx?.BlockchainId?.toBytes) {
|
|
407
|
+
return buffer_1.Buffer.from(innerTx.baseTx.BlockchainId.toBytes()).toString('hex');
|
|
408
|
+
}
|
|
409
|
+
if (innerTx.blockchainId?.toBytes) {
|
|
410
|
+
return buffer_1.Buffer.from(innerTx.blockchainId.toBytes()).toString('hex');
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
if (txObj.tx?.baseTx?.BlockchainId?.toBytes) {
|
|
414
|
+
return buffer_1.Buffer.from(txObj.tx.baseTx.BlockchainId.toBytes()).toString('hex');
|
|
415
|
+
}
|
|
416
|
+
if (txObj.baseTx?.BlockchainId?.toBytes) {
|
|
417
|
+
return buffer_1.Buffer.from(txObj.baseTx.BlockchainId.toBytes()).toString('hex');
|
|
418
|
+
}
|
|
419
|
+
if (txObj.blockchainId?.toBytes) {
|
|
420
|
+
return buffer_1.Buffer.from(txObj.blockchainId.toBytes()).toString('hex');
|
|
421
|
+
}
|
|
422
|
+
return null;
|
|
423
|
+
};
|
|
424
|
+
const txBlockchainId = extractBlockchainId(tx);
|
|
425
|
+
return txBlockchainId === blockchainId;
|
|
426
|
+
}
|
|
427
|
+
/**
|
|
428
|
+
* Convert FlareJS native Utxo to DecodedUtxoObj for internal use
|
|
429
|
+
* @param utxo - FlareJS Utxo object
|
|
430
|
+
* @param network - Flare network configuration
|
|
431
|
+
* @returns DecodedUtxoObj compatible with existing methods
|
|
432
|
+
*/
|
|
433
|
+
utxoToDecoded(utxo, network) {
|
|
434
|
+
const outputOwners = utxo.getOutputOwners();
|
|
435
|
+
const output = utxo.output;
|
|
436
|
+
// Get amount from output
|
|
437
|
+
const amount = output.amount().toString();
|
|
438
|
+
// Get txid from utxoId (cb58 encoded)
|
|
439
|
+
const txid = this.cb58Encode(buffer_1.Buffer.from(utxo.utxoId.txID.toBytes()));
|
|
440
|
+
// Get output index
|
|
441
|
+
const outputidx = utxo.utxoId.outputIdx.value().toString();
|
|
442
|
+
// Get threshold
|
|
443
|
+
const threshold = outputOwners.threshold.value();
|
|
444
|
+
// Get locktime
|
|
445
|
+
const locktime = outputOwners.locktime.value().toString();
|
|
446
|
+
// Get addresses as bech32 strings
|
|
447
|
+
const addresses = outputOwners.addrs.map((addr) => this.addressToString(network.hrp, network.alias, buffer_1.Buffer.from(addr.toBytes())));
|
|
448
|
+
return {
|
|
449
|
+
outputID: iface_1.SECP256K1_Transfer_Output,
|
|
450
|
+
locktime,
|
|
451
|
+
amount,
|
|
452
|
+
txid,
|
|
453
|
+
outputidx,
|
|
454
|
+
threshold,
|
|
455
|
+
addresses,
|
|
456
|
+
};
|
|
457
|
+
}
|
|
458
|
+
/**
|
|
459
|
+
* Convert array of FlareJS Utxos to DecodedUtxoObj array
|
|
460
|
+
* @param utxos - Array of FlareJS Utxo objects
|
|
461
|
+
* @param network - Flare network configuration
|
|
462
|
+
* @returns Array of DecodedUtxoObj
|
|
463
|
+
*/
|
|
464
|
+
utxosToDecoded(utxos, network) {
|
|
465
|
+
return utxos.map((utxo) => this.utxoToDecoded(utxo, network));
|
|
466
|
+
}
|
|
467
|
+
/**
|
|
468
|
+
* Convert DecodedUtxoObj to native FlareJS Utxo object
|
|
469
|
+
* This is the reverse of utxoToDecoded
|
|
470
|
+
*
|
|
471
|
+
* IMPORTANT: Addresses are sorted lexicographically by byte value to match
|
|
472
|
+
* on-chain storage order. The API may return addresses in arbitrary order, but
|
|
473
|
+
* on-chain UTXOs always store addresses in sorted order.
|
|
474
|
+
*
|
|
475
|
+
* @param decoded - DecodedUtxoObj to convert
|
|
476
|
+
* @param assetId - Asset ID as cb58 encoded string
|
|
477
|
+
* @returns Native FlareJS Utxo object
|
|
478
|
+
*/
|
|
479
|
+
decodedToUtxo(decoded, assetId) {
|
|
480
|
+
// Create UTXOID from txid and output index
|
|
481
|
+
const utxoId = flarejs_1.avaxSerial.UTXOID.fromNative(decoded.txid, parseInt(decoded.outputidx, 10));
|
|
482
|
+
// Parse addresses from bech32 strings to byte buffers
|
|
483
|
+
const addressBytes = decoded.addresses.map((addr) => this.parseAddress(addr));
|
|
484
|
+
// Sort addresses lexicographically by byte value to match on-chain order
|
|
485
|
+
// This is critical because the P-chain stores addresses in sorted order,
|
|
486
|
+
// and sigIndices must reference the correct positions.
|
|
487
|
+
const sortedAddressBytes = this.sortAddressBuffersByHex(addressBytes);
|
|
488
|
+
// Create OutputOwners with locktime, threshold, and SORTED addresses
|
|
489
|
+
const locktime = decoded.locktime ? BigInt(decoded.locktime) : BigInt(0);
|
|
490
|
+
const outputOwners = flarejs_1.OutputOwners.fromNative(sortedAddressBytes, locktime, decoded.threshold);
|
|
491
|
+
// Create TransferOutput with amount and owners
|
|
492
|
+
const amount = BigInt(decoded.amount);
|
|
493
|
+
const transferOutput = new flarejs_1.TransferOutput(new flarejs_1.BigIntPr(amount), outputOwners);
|
|
494
|
+
// Create and return the Utxo
|
|
495
|
+
return new flarejs_1.Utxo(utxoId, flarejs_1.Id.fromString(assetId), transferOutput);
|
|
473
496
|
}
|
|
474
497
|
/**
|
|
475
|
-
*
|
|
476
|
-
* @param
|
|
477
|
-
* @param
|
|
478
|
-
* @returns
|
|
498
|
+
* Convert array of DecodedUtxoObj to native FlareJS Utxo objects
|
|
499
|
+
* @param decodedUtxos - Array of DecodedUtxoObj
|
|
500
|
+
* @param assetId - Asset ID as cb58 encoded string
|
|
501
|
+
* @returns Array of native FlareJS Utxo objects
|
|
479
502
|
*/
|
|
480
|
-
|
|
481
|
-
return
|
|
503
|
+
decodedToUtxos(decodedUtxos, assetId) {
|
|
504
|
+
return decodedUtxos.map((decoded) => this.decodedToUtxo(decoded, assetId));
|
|
482
505
|
}
|
|
483
506
|
}
|
|
484
507
|
exports.Utils = Utils;
|
|
485
508
|
const utils = new Utils();
|
|
486
509
|
exports.default = utils;
|
|
487
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":";;;AACA,mDAQ8B;AAE9B,qDAA4C;AAC5C,mCAAoC;AAEpC,2CAWqB;AAErB,MAAa,KAAK;IAAlB;QA4FS,iBAAY,GAAG,CAAC,GAAW,EAAU,EAAE;YAC5C,yCAAyC;YACzC,OAAO,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC,CAAC,4BAA4B;QAC9D,CAAC,CAAC;IA8ZJ,CAAC;IA5fQ,SAAS,CAAC,eAAyB,EAAE,mBAA6B;QACvE,OAAO,eAAe,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,mBAAmB,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,IAAI,CAAC,CAAC;IACpG,CAAC;IAED;;;;;OAKG;IACH,kBAAkB;IAClB,cAAc,CAAC,OAA0B;QACvC,MAAM,UAAU,GAAa,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAEnF,KAAK,MAAM,OAAO,IAAI,UAAU,EAAE,CAAC;YACjC,IAAI,CAAC,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,EAAE,CAAC;gBACvC,OAAO,KAAK,CAAC;YACf,CAAC;QACH,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,mBAAmB,CAAC,OAAe;QACzC,OAAO,yBAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACrC,CAAC;IAED;;;;;OAKG;IACH,kBAAkB;IAClB,cAAc,CAAC,IAAY;QACzB,2EAA2E;QAC3E,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,wCAAwC;YAC3E,OAAO,OAAO,CAAC,MAAM,KAAK,mCAAuB,CAAC;QACpD,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACH,gBAAgB,CAAC,GAAW;QAC1B,IAAI,IAAA,sBAAW,EAAC,GAAG,CAAC;YAAE,OAAO,IAAI,CAAC;QAElC,IAAI,MAAc,CAAC;QACnB,IAAI,GAAG,CAAC,MAAM,KAAK,gCAAoB,EAAE,CAAC;YACxC,IAAI,CAAC;gBACH,iEAAiE;gBACjE,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC,CAAC,wBAAwB;YAC5D,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,KAAK,CAAC;YACf,CAAC;QACH,CAAC;aAAM,CAAC;YACN,IAAI,GAAG,CAAC,MAAM,KAAK,wCAA4B,IAAI,GAAG,CAAC,MAAM,KAAK,0CAA8B,EAAE,CAAC;gBACjG,OAAO,KAAK,CAAC;YACf,CAAC;YAED,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;YAElC,0BAA0B;YAC1B,IAAI,GAAG,CAAC,MAAM,KAAK,0CAA8B,IAAI,SAAS,KAAK,IAAI,EAAE,CAAC;gBACxE,OAAO,KAAK,CAAC;YACf,CAAC;YAED,wBAAwB;YACxB,IAAI,GAAG,CAAC,MAAM,KAAK,wCAA4B,IAAI,SAAS,KAAK,IAAI,IAAI,SAAS,KAAK,IAAI,EAAE,CAAC;gBAC5F,OAAO,KAAK,CAAC;YACf,CAAC;YAED,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC;gBAAE,OAAO,KAAK,CAAC;YACzC,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QACnC,CAAC;QACD,gDAAgD;QAChD,IAAI,CAAC;YACH,eAAG,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,8BAA8B;YACnD,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAOD;;;;;;;;;;OAUG;IACH,iBAAiB,CAAC,GAAW;QAC3B,IAAI,IAAA,sBAAW,EAAC,GAAG,CAAC;YAAE,OAAO,IAAI,CAAC;QAElC,IAAI,GAAG,CAAC,MAAM,KAAK,kCAAsB,IAAI,GAAG,CAAC,MAAM,KAAK,uCAA2B,EAAE,CAAC;YACxF,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IACE,GAAG,CAAC,MAAM,KAAK,uCAA2B;YAC1C,GAAG,CAAC,KAAK,CAAC,kCAAsB,CAAC,KAAK,yCAA6B,EACnE,CAAC;YACD,OAAO,KAAK,CAAC;QACf,CAAC;QAED,OAAO,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;IAC/B,CAAC;IAED;;;;;OAKG;IACH,WAAW,CAAC,KAAa;QACvB,OAAO,qBAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED;;;;;;OAMG;IACH,sBAAsB,CAAC,OAAe;QACpC,IAAI,CAAC,OAAO,IAAI,OAAO,OAAO,KAAK,QAAQ,EAAE,CAAC;YAC5C,OAAO,KAAK,CAAC;QACf,CAAC;QAED,8BAA8B;QAC9B,MAAM,YAAY,GAAG,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;QAE3E,0CAA0C;QAC1C,OAAO,YAAY,CAAC,MAAM,KAAK,EAAE,IAAI,mBAAmB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IAC9E,CAAC;IAED;;;;;;OAMG;IACH,IAAI,CAAuB,GAAM,EAAE,IAAS;QAC1C,MAAM,MAAM,GAAG,EAAgB,CAAC;QAChC,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACvB,IAAI,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC;gBACnD,MAAM,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC;YACzB,CAAC;QACH,CAAC;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;OAMG;IACH,OAAO,CAAC,CAAU,EAAE,CAAU;QAC5B,IAAI,CAAC,KAAK,CAAC;YAAE,OAAO,IAAI,CAAC;QAEzB,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,SAAS,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,SAAS;YAAE,OAAO,CAAC,KAAK,CAAC,CAAC;QAEnF,IAAI,OAAO,CAAC,KAAK,OAAO,CAAC;YAAE,OAAO,KAAK,CAAC;QAExC,IAAI,OAAO,CAAC,KAAK,QAAQ,EAAE,CAAC;YAC1B,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC;gBAAE,OAAO,KAAK,CAAC;YAExD,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC;gBACrB,MAAM,IAAI,GAAG,CAAc,CAAC;gBAC5B,IAAI,CAAC,CAAC,MAAM,KAAK,IAAI,CAAC,MAAM;oBAAE,OAAO,KAAK,CAAC;gBAC3C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;oBAClC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC;wBAAE,OAAO,KAAK,CAAC;gBACjD,CAAC;gBACD,OAAO,IAAI,CAAC;YACd,CAAC;YAED,MAAM,IAAI,GAAG,CAA4B,CAAC;YAC1C,MAAM,IAAI,GAAG,CAA4B,CAAC;YAC1C,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAChC,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAChC,IAAI,KAAK,CAAC,MAAM,KAAK,KAAK,CAAC,MAAM;gBAAE,OAAO,KAAK,CAAC;YAEhD,KAAK,MAAM,GAAG,IAAI,KAAK,EAAE,CAAC;gBACxB,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC;oBAAE,OAAO,KAAK,CAAC;gBACvC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;oBAAE,OAAO,KAAK,CAAC;YACxD,CAAC;YACD,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED,kBAAkB;IAClB,gBAAgB,CAAC,SAAiB;QAChC,MAAM,IAAI,8BAAmB,CAAC,kCAAkC,CAAC,CAAC;IACpE,CAAC;IAED,kBAAkB;IAClB,oBAAoB,CAAC,IAAY;QAC/B,MAAM,IAAI,8BAAmB,CAAC,sCAAsC,CAAC,CAAC;IACxE,CAAC;IAED;;;;;;OAMG;IACH,eAAe,CAAC,OAAqB,EAAE,OAAe,EAAE,GAAW;QACjE,2EAA2E;QAC3E,IAAI,CAAC;YACH,MAAM,SAAS,GAAG,eAAG,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;YACzC,OAAO,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAChC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,IAAI,KAAK,CAAC,+BAA+B,KAAK,EAAE,CAAC,CAAC;QAC1D,CAAC;IACH,CAAC;IAED;;;;;;;OAOG;IACH,eAAe,CAAC,OAAqB,EAAE,OAAe,EAAE,SAAiB,EAAE,SAAiB;QAC1F,IAAI,CAAC;YACH,OAAO,eAAG,CAAC,MAAM,CAAC,OAAO,EAAE,SAAS,EAAE,SAAS,CAAC,CAAC;QACnD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACH,iBAAiB,CAAC,OAAqB,EAAE,OAAe,EAAE,SAAiB;QACzE,IAAI,CAAC;YACH,4DAA4D;YAC5D,wEAAwE;YACxE,MAAM,IAAI,8BAAmB,CAAC,qDAAqD,CAAC,CAAC;QACvF,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,IAAI,KAAK,CAAC,gCAAgC,KAAK,EAAE,CAAC,CAAC;QAC3D,CAAC;IACH,CAAC;IAED,MAAM,CAAC,GAAe;QACpB,OAAO,IAAA,mBAAU,EAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,CAAC;IACnD,CAAC;IAED;;;;;OAKG;IACH,sBAAsB,CAAC,cAAsB;QAC3C,IAAI,CAAC,cAAc,EAAE,CAAC;YACpB,MAAM,IAAI,kCAAuB,CAAC,0BAA0B,CAAC,CAAC;QAChE,CAAC;QACD,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,cAAc,CAAC,EAAE,CAAC;YACvC,MAAM,IAAI,gCAAqB,CAAC,mCAAmC,CAAC,CAAC;QACvE,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACH,eAAe,CAAC,EAAgB,EAAE,YAAoB;QACpD,2EAA2E;QAC3E,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,EAAwC,CAAC;YAC1D,MAAM,UAAU,GAAI,QAAQ,CAAC,aAA+C,EAAE,CAAC;YAC/E,MAAM,WAAW,GAAI,UAAU,CAAC,cAAgD,EAAE,CAAC;YACnF,MAAM,cAAc,GAAI,WAAW,CAAC,eAAiC,EAAE,CAAC;YACxE,OAAO,MAAM,CAAC,IAAI,CAAC,cAAwB,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,KAAK,YAAY,CAAC;QAChF,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,8BAA8B,CAAC,MAAwB;QACrD,OAAO,WAAW,IAAK,MAAkC,CAAC;IAC5D,CAAC;IAED;;;;;OAKG;IACH,oBAAoB,CAAC,MAAc;QACjC,OAAO,OAAQ,MAA6C,CAAC,SAAS,KAAK,UAAU,CAAC;IACxF,CAAC;IAED;;;;OAIG;IACH,0BAA0B,CAAC,OAAqB;QAC9C,OAAO,CAAC,MAAwB,EAAE,EAAE;YAClC,IAAI,IAAI,CAAC,8BAA8B,CAAC,MAAM,CAAC,EAAE,CAAC;gBAChD,wCAAwC;gBACxC,IAAI,CAAC;oBACH,MAAM,kBAAkB,GAAG,MAAuC,CAAC;oBACnE,MAAM,MAAM,GAAG,kBAAkB,CAAC,MAAM,EAAE,CAAC;oBAE3C,4EAA4E;oBAC5E,MAAM,OAAO,GAAG,2BAA2B,CAAC,CAAC,4CAA4C;oBAEzF,OAAO;wBACL,KAAK,EAAE,MAAM,CAAC,QAAQ,EAAE;wBACxB,OAAO;qBACR,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBACf,MAAM,IAAI,KAAK,CAAC,yBAAyB,KAAK,EAAE,CAAC,CAAC;gBACpD,CAAC;YACH,CAAC;iBAAM,CAAC;gBACN,sCAAsC;gBACtC,OAAO;oBACL,KAAK,EAAE,GAAG,EAAE,2CAA2C;oBACvD,OAAO,EAAE,4CAA4C,EAAE,4CAA4C;iBACpG,CAAC;YACJ,CAAC;QACH,CAAC,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,gBAAgB,CAAC,OAAqB;QACpC,OAAO,CAAC,MAAc,EAAE,EAAE;YACxB,IAAI,IAAI,CAAC,oBAAoB,CAAC,MAAM,CAAC,EAAE,CAAC;gBACtC,MAAM,kBAAkB,GAAG,MAA4B,CAAC;gBACxD,MAAM,YAAY,GAAG,kBAAkB,CAAC,MAAM,EAAE,CAAC;gBAEjD,0CAA0C;gBAC1C,MAAM,OAAO,GAAG,2BAA2B,CAAC,CAAC,4CAA4C;gBAEzF,OAAO;oBACL,KAAK,EAAE,YAAY,CAAC,QAAQ,EAAE;oBAC9B,OAAO;iBACR,CAAC;YACJ,CAAC;iBAAM,CAAC;gBACN,MAAM,IAAI,KAAK,CAAC,qBAAqB,CAAC,CAAC;YACzC,CAAC;QACH,CAAC,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,eAAe,CAAC,GAAW;QACzB,IAAI,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC;YACzB,OAAO,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC;QAC1B,CAAC;QACD,OAAO,GAAG,CAAC;IACb,CAAC;IAED;;;;OAIG;IACH,uBAAuB,CAAC,SAAiB;QACvC,OAAO,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,mCAAuB,EAAE,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;IACnG,CAAC;IAED;;;;OAIG;IACH,uBAAuB,CAAC,SAAiB;QACvC,OAAO,QAAQ,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC;IAC5D,CAAC;IAED;;;;OAIG;IACH,UAAU,CAAC,IAAY;QACrB,kDAAkD;QAClD,+DAA+D;QAC/D,IAAI,CAAC;YACH,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;QAClC,CAAC;QAAC,MAAM,CAAC;YACP,iCAAiC;YACjC,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC3B,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACH,eAAe,CAAC,GAAW,EAAE,OAAe,EAAE,aAAqB;QACjE,qEAAqE;QACrE,OAAO,GAAG,OAAO,IAAI,aAAa,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;IACvD,CAAC;IAED;;;;;OAKG;IACH,aAAa,CAAC,IAAY;QACxB,OAAO,IAAI,WAAW,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IACxC,CAAC;IAED;;;;OAIG;IACH,aAAa,CAAC,KAAiB;QAC7B,OAAO,IAAI,WAAW,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IACzC,CAAC;IAED;;;;;OAKG;IACH,eAAe,CAAC,IAAmD;QACjE,IAAI,IAAI,YAAY,UAAU,EAAE,CAAC;YAC/B,OAAO,IAAI,CAAC;QACd,CAAC;QAED,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC7B,OAAO,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;QAClC,CAAC;QAED,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC7B,OAAO,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;QAClD,CAAC;QAED,MAAM,IAAI,kCAAuB,CAAC,qBAAqB,CAAC,CAAC;IAC3D,CAAC;IAED;;;;OAIG;IACH,cAAc,CAAC,SAAqB;QAClC,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC3B,OAAO,EAAE,CAAC;QACZ,CAAC;QACD,OAAO,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC;IACvC,CAAC;IAED;;;;;OAKG;IACH,gBAAgB,CAAC,SAAqB,EAAE,OAAO,GAAG,IAAI;QACpD,OAAO,SAAS,CAAC,MAAM,IAAI,OAAO,CAAC;IACrC,CAAC;CACF;AA7fD,sBA6fC;AAED,MAAM,KAAK,GAAG,IAAI,KAAK,EAAE,CAAC;AAE1B,kBAAe,KAAK,CAAC","sourcesContent":["import { TransferableOutput } from '@flarenetwork/flarejs';\nimport {\n  BaseUtils,\n  Entry,\n  InvalidTransactionError,\n  isValidXprv,\n  isValidXpub,\n  NotImplementedError,\n  ParseTransactionError,\n} from '@bitgo-beta/sdk-core';\nimport { FlareNetwork } from '@bitgo-beta/statics';\nimport { ecc } from '@bitgo-beta/secp256k1';\nimport { createHash } from 'crypto';\nimport { DeprecatedOutput, DeprecatedTx, Output } from './iface';\nimport {\n  DECODED_BLOCK_ID_LENGTH,\n  SHORT_PUB_KEY_LENGTH,\n  COMPRESSED_PUBLIC_KEY_LENGTH,\n  UNCOMPRESSED_PUBLIC_KEY_LENGTH,\n  RAW_PRIVATE_KEY_LENGTH,\n  SUFFIXED_PRIVATE_KEY_LENGTH,\n  PRIVATE_KEY_COMPRESSED_SUFFIX,\n  OUTPUT_INDEX_HEX_LENGTH,\n  ADDRESS_REGEX,\n  HEX_REGEX,\n} from './constants';\n\nexport class Utils implements BaseUtils {\n  public includeIn(walletAddresses: string[], otxoOutputAddresses: string[]): boolean {\n    return walletAddresses.map((a) => otxoOutputAddresses.includes(a)).reduce((a, b) => a && b, true);\n  }\n\n  /**\n   * Checks if it is a valid address no illegal characters\n   *\n   * @param {string} address - address to be validated\n   * @returns {boolean} - the validation result\n   */\n  /** @inheritdoc */\n  isValidAddress(address: string | string[]): boolean {\n    const addressArr: string[] = Array.isArray(address) ? address : address.split('~');\n\n    for (const address of addressArr) {\n      if (!this.isValidAddressRegex(address)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  private isValidAddressRegex(address: string): boolean {\n    return ADDRESS_REGEX.test(address);\n  }\n\n  /**\n   * Checks if it is a valid blockId with length 66 including 0x\n   *\n   * @param {string} hash - blockId to be validated\n   * @returns {boolean} - the validation result\n   */\n  /** @inheritdoc */\n  isValidBlockId(hash: string): boolean {\n    // FlareJS equivalent - check if it's a valid CB58 hash with correct length\n    try {\n      const decoded = Buffer.from(hash); // FlareJS should provide CB58 utilities\n      return decoded.length === DECODED_BLOCK_ID_LENGTH;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Checks if the string is a valid protocol public key or\n   * extended public key.\n   *\n   * @param {string} pub - the  public key to be validated\n   * @returns {boolean} - the validation result\n   */\n  isValidPublicKey(pub: string): boolean {\n    if (isValidXpub(pub)) return true;\n\n    let pubBuf: Buffer;\n    if (pub.length === SHORT_PUB_KEY_LENGTH) {\n      try {\n        // For FlareJS, we'll need to implement CB58 decode functionality\n        pubBuf = Buffer.from(pub, 'hex'); // Temporary placeholder\n      } catch {\n        return false;\n      }\n    } else {\n      if (pub.length !== COMPRESSED_PUBLIC_KEY_LENGTH && pub.length !== UNCOMPRESSED_PUBLIC_KEY_LENGTH) {\n        return false;\n      }\n\n      const firstByte = pub.slice(0, 2);\n\n      // uncompressed public key\n      if (pub.length === UNCOMPRESSED_PUBLIC_KEY_LENGTH && firstByte !== '04') {\n        return false;\n      }\n\n      // compressed public key\n      if (pub.length === COMPRESSED_PUBLIC_KEY_LENGTH && firstByte !== '02' && firstByte !== '03') {\n        return false;\n      }\n\n      if (!this.allHexChars(pub)) return false;\n      pubBuf = Buffer.from(pub, 'hex');\n    }\n    // validate the public key using BitGo secp256k1\n    try {\n      ecc.isPoint(pubBuf); // Check if it's a valid point\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  public parseAddress = (pub: string): Buffer => {\n    // FlareJS equivalent for address parsing\n    return Buffer.from(pub, 'hex'); // Simplified implementation\n  };\n\n  /**\n   * Returns whether or not the string is a valid protocol private key, or extended\n   * private key.\n   *\n   * The protocol key format is described in the @stacks/transactions npm package, in the\n   * createStacksPrivateKey function:\n   * https://github.com/blockstack/stacks.js/blob/master/packages/transactions/src/keys.ts#L125\n   *\n   * @param {string} prv - the private key (or extended private key) to be validated\n   * @returns {boolean} - the validation result\n   */\n  isValidPrivateKey(prv: string): boolean {\n    if (isValidXprv(prv)) return true;\n\n    if (prv.length !== RAW_PRIVATE_KEY_LENGTH && prv.length !== SUFFIXED_PRIVATE_KEY_LENGTH) {\n      return false;\n    }\n\n    if (\n      prv.length === SUFFIXED_PRIVATE_KEY_LENGTH &&\n      prv.slice(RAW_PRIVATE_KEY_LENGTH) !== PRIVATE_KEY_COMPRESSED_SUFFIX\n    ) {\n      return false;\n    }\n\n    return this.allHexChars(prv);\n  }\n\n  /**\n   * Returns whether or not the string is a composed of hex chars only\n   *\n   * @param {string} maybe - the  string to be validated\n   * @returns {boolean} - the validation result\n   */\n  allHexChars(maybe: string): boolean {\n    return HEX_REGEX.test(maybe);\n  }\n\n  /**\n   * Lightweight Ethereum address validation\n   * Validates that an address is a 40-character hex string (optionally prefixed with 0x)\n   *\n   * @param {string} address - the Ethereum address to validate\n   * @returns {boolean} - true if valid Ethereum address format\n   */\n  isValidEthereumAddress(address: string): boolean {\n    if (!address || typeof address !== 'string') {\n      return false;\n    }\n\n    // Remove 0x prefix if present\n    const cleanAddress = address.startsWith('0x') ? address.slice(2) : address;\n\n    // Check if it's exactly 40 hex characters\n    return cleanAddress.length === 40 && /^[0-9a-fA-F]{40}$/.test(cleanAddress);\n  }\n\n  /**\n   * Pick specific properties from an object (replaces lodash.pick)\n   *\n   * @param {T} obj - the source object\n   * @param {K[]} keys - array of property keys to pick\n   * @returns {Pick<T, K>} - new object with only the specified properties\n   */\n  pick<T, K extends keyof T>(obj: T, keys: K[]): Pick<T, K> {\n    const result = {} as Pick<T, K>;\n    for (const key of keys) {\n      if (Object.prototype.hasOwnProperty.call(obj, key)) {\n        result[key] = obj[key];\n      }\n    }\n    return result;\n  }\n\n  /**\n   * Deep equality comparison (replaces lodash.isEqual)\n   *\n   * @param {unknown} a - first value to compare\n   * @param {unknown} b - second value to compare\n   * @returns {boolean} - true if values are deeply equal\n   */\n  isEqual(a: unknown, b: unknown): boolean {\n    if (a === b) return true;\n\n    if (a === null || a === undefined || b === null || b === undefined) return a === b;\n\n    if (typeof a !== typeof b) return false;\n\n    if (typeof a === 'object') {\n      if (Array.isArray(a) !== Array.isArray(b)) return false;\n\n      if (Array.isArray(a)) {\n        const arrB = b as unknown[];\n        if (a.length !== arrB.length) return false;\n        for (let i = 0; i < a.length; i++) {\n          if (!this.isEqual(a[i], arrB[i])) return false;\n        }\n        return true;\n      }\n\n      const objA = a as Record<string, unknown>;\n      const objB = b as Record<string, unknown>;\n      const keysA = Object.keys(objA);\n      const keysB = Object.keys(objB);\n      if (keysA.length !== keysB.length) return false;\n\n      for (const key of keysA) {\n        if (!keysB.includes(key)) return false;\n        if (!this.isEqual(objA[key], objB[key])) return false;\n      }\n      return true;\n    }\n\n    return false;\n  }\n\n  /** @inheritdoc */\n  isValidSignature(signature: string): boolean {\n    throw new NotImplementedError('isValidSignature not implemented');\n  }\n\n  /** @inheritdoc */\n  isValidTransactionId(txId: string): boolean {\n    throw new NotImplementedError('isValidTransactionId not implemented');\n  }\n\n  /**\n   * FlareJS wrapper to create signature and return it for credentials\n   * @param network\n   * @param message\n   * @param prv\n   * @return signature\n   */\n  createSignature(network: FlareNetwork, message: Buffer, prv: Buffer): Buffer {\n    // Use BitGo secp256k1 since FlareJS may not expose KeyPair in the same way\n    try {\n      const signature = ecc.sign(message, prv);\n      return Buffer.from(signature);\n    } catch (error) {\n      throw new Error(`Failed to create signature: ${error}`);\n    }\n  }\n\n  /**\n   * FlareJS wrapper to verify signature\n   * @param network\n   * @param message\n   * @param signature\n   * @param publicKey - public key instead of private key for verification\n   * @return true if it's verify successful\n   */\n  verifySignature(network: FlareNetwork, message: Buffer, signature: Buffer, publicKey: Buffer): boolean {\n    try {\n      return ecc.verify(message, publicKey, signature);\n    } catch (error) {\n      return false;\n    }\n  }\n\n  /**\n   * FlareJS wrapper to recover signature\n   * @param network\n   * @param message\n   * @param signature\n   * @return recovered public key\n   */\n  recoverySignature(network: FlareNetwork, message: Buffer, signature: Buffer): Buffer {\n    try {\n      // This would need to be implemented with secp256k1 recovery\n      // For now, throwing error since recovery logic would need to be adapted\n      throw new NotImplementedError('recoverySignature not fully implemented for FlareJS');\n    } catch (error) {\n      throw new Error(`Failed to recover signature: ${error}`);\n    }\n  }\n\n  sha256(buf: Uint8Array): Buffer {\n    return createHash('sha256').update(buf).digest();\n  }\n\n  /**\n   * Check the raw transaction has a valid format in the blockchain context, throw otherwise.\n   * It's to reuse in TransactionBuilder and TransactionBuilderFactory\n   *\n   * @param rawTransaction Transaction as hex string\n   */\n  validateRawTransaction(rawTransaction: string): void {\n    if (!rawTransaction) {\n      throw new InvalidTransactionError('Raw transaction is empty');\n    }\n    if (!utils.allHexChars(rawTransaction)) {\n      throw new ParseTransactionError('Raw transaction is not hex string');\n    }\n  }\n\n  /**\n   * Check if tx is for the blockchainId\n   *\n   * @param {DeprecatedTx} tx\n   * @param {string} blockchainId\n   * @returns true if tx is for blockchainId\n   */\n  isTransactionOf(tx: DeprecatedTx, blockchainId: string): boolean {\n    // FlareJS equivalent - this would need proper CB58 encoding implementation\n    try {\n      const txRecord = tx as unknown as Record<string, unknown>;\n      const unsignedTx = (txRecord.getUnsignedTx as () => Record<string, unknown>)();\n      const transaction = (unsignedTx.getTransaction as () => Record<string, unknown>)();\n      const txBlockchainId = (transaction.getBlockchainID as () => unknown)();\n      return Buffer.from(txBlockchainId as string).toString('hex') === blockchainId;\n    } catch (error) {\n      return false;\n    }\n  }\n\n  /**\n   * Check if Output is from PVM.\n   * Output could be EVM or PVM output.\n   * @param {DeprecatedOutput} output\n   * @returns {boolean} output has transferable output structure\n   */\n  deprecatedIsTransferableOutput(output: DeprecatedOutput): boolean {\n    return 'getOutput' in (output as Record<string, unknown>);\n  }\n\n  /**\n   * Check if Output is from PVM.\n   * Output could be EVM or PVM output.\n   * @param {Output} output\n   * @returns {boolean} output is TransferableOutput\n   */\n  isTransferableOutput(output: Output): output is TransferableOutput {\n    return typeof (output as unknown as Record<string, unknown>).getOutput === 'function';\n  }\n\n  /**\n   * Return a mapper function to that network address representation.\n   * @param network required to stringify addresses\n   * @return mapper function\n   */\n  deprecatedMapOutputToEntry(network: FlareNetwork): (output: DeprecatedOutput) => Entry {\n    return (output: DeprecatedOutput) => {\n      if (this.deprecatedIsTransferableOutput(output)) {\n        // Simplified implementation for FlareJS\n        try {\n          const transferableOutput = output as unknown as TransferableOutput;\n          const amount = transferableOutput.amount();\n\n          // Simplified address handling - would need proper FlareJS address utilities\n          const address = 'flare-address-placeholder'; // TODO: implement proper address conversion\n\n          return {\n            value: amount.toString(),\n            address,\n          };\n        } catch (error) {\n          throw new Error(`Failed to map output: ${error}`);\n        }\n      } else {\n        // Handle EVM output case - simplified\n        return {\n          value: '0', // TODO: implement proper amount extraction\n          address: '0x0000000000000000000000000000000000000000', // TODO: implement proper address extraction\n        };\n      }\n    };\n  }\n\n  /**\n   * Return a mapper function to that network address representation.\n   * @param network required to stringify addresses\n   * @return mapper function\n   */\n  mapOutputToEntry(network: FlareNetwork): (Output) => Entry {\n    return (output: Output) => {\n      if (this.isTransferableOutput(output)) {\n        const transferableOutput = output as TransferableOutput;\n        const outputAmount = transferableOutput.amount();\n\n        // Simplified address handling for FlareJS\n        const address = 'flare-address-placeholder'; // TODO: implement proper address conversion\n\n        return {\n          value: outputAmount.toString(),\n          address,\n        };\n      } else {\n        throw new Error('Invalid output type');\n      }\n    };\n  }\n\n  /**\n   * remove hex prefix (0x)\n   * @param hex string\n   * @returns hex without 0x\n   */\n  removeHexPrefix(hex: string): string {\n    if (hex.startsWith('0x')) {\n      return hex.substring(2);\n    }\n    return hex;\n  }\n\n  /**\n   * Outputidx convert from number (as string) to buffer.\n   * @param {string} outputidx number\n   * @return {Buffer} buffer of size 4 with that number value\n   */\n  outputidxNumberToBuffer(outputidx: string): Buffer {\n    return Buffer.from(Number(outputidx).toString(16).padStart(OUTPUT_INDEX_HEX_LENGTH, '0'), 'hex');\n  }\n\n  /**\n   * Outputidx buffer to number (as string)\n   * @param {Buffer} outputidx\n   * @return {string} outputidx number\n   */\n  outputidxBufferToNumber(outputidx: Buffer): string {\n    return parseInt(outputidx.toString('hex'), 16).toString();\n  }\n\n  /**\n   * CB58 decode function - simple Base58 decode implementation\n   * @param {string} data - CB58 encoded string\n   * @returns {Buffer} decoded buffer\n   */\n  cb58Decode(data: string): Buffer {\n    // For now, use a simple hex decode as placeholder\n    // In a full implementation, this would be proper CB58 decoding\n    try {\n      return Buffer.from(data, 'hex');\n    } catch {\n      // Fallback to buffer from string\n      return Buffer.from(data);\n    }\n  }\n\n  /**\n   * Convert address buffer to bech32 string\n   * @param {string} hrp - Human readable part\n   * @param {string} chainid - Chain identifier\n   * @param {Buffer} addressBuffer - Address buffer\n   * @returns {string} Address string\n   */\n  addressToString(hrp: string, chainid: string, addressBuffer: Buffer): string {\n    // Simple implementation - in practice this would use bech32 encoding\n    return `${chainid}-${addressBuffer.toString('hex')}`;\n  }\n\n  /**\n   * Convert string to bytes for FlareJS memo\n   * Follows FlareJS utils.stringToBytes pattern\n   * @param {string} text - Text to convert\n   * @returns {Uint8Array} Byte array\n   */\n  stringToBytes(text: string): Uint8Array {\n    return new TextEncoder().encode(text);\n  }\n\n  /**\n   * Convert bytes to string from FlareJS memo\n   * @param {Uint8Array} bytes - Bytes to convert\n   * @returns {string} Decoded string\n   */\n  bytesToString(bytes: Uint8Array): string {\n    return new TextDecoder().decode(bytes);\n  }\n\n  /**\n   * Create memo bytes from various input formats\n   * Supports string, JSON object, or raw bytes\n   * @param {string | Record<string, unknown> | Uint8Array} memo - Memo data\n   * @returns {Uint8Array} Memo bytes for FlareJS\n   */\n  createMemoBytes(memo: string | Record<string, unknown> | Uint8Array): Uint8Array {\n    if (memo instanceof Uint8Array) {\n      return memo;\n    }\n\n    if (typeof memo === 'string') {\n      return this.stringToBytes(memo);\n    }\n\n    if (typeof memo === 'object') {\n      return this.stringToBytes(JSON.stringify(memo));\n    }\n\n    throw new InvalidTransactionError('Invalid memo format');\n  }\n\n  /**\n   * Parse memo bytes to string\n   * @param {Uint8Array} memoBytes - Memo bytes from FlareJS transaction\n   * @returns {string} Decoded memo string\n   */\n  parseMemoBytes(memoBytes: Uint8Array): string {\n    if (memoBytes.length === 0) {\n      return '';\n    }\n    return this.bytesToString(memoBytes);\n  }\n\n  /**\n   * Validate memo size (FlareJS has transaction size limits)\n   * @param {Uint8Array} memoBytes - Memo bytes\n   * @param {number} maxSize - Maximum size in bytes (default 4KB)\n   * @returns {boolean} Whether memo is within size limits\n   */\n  validateMemoSize(memoBytes: Uint8Array, maxSize = 4096): boolean {\n    return memoBytes.length <= maxSize;\n  }\n}\n\nconst utils = new Utils();\n\nexport default utils;\n"]}
|
|
510
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Signature,\n  TransferableOutput,\n  TransferOutput,\n  TypeSymbols,\n  Id,\n  Utxo,\n  BigIntPr,\n  OutputOwners,\n  avaxSerial,\n} from '@flarenetwork/flarejs';\nimport {\n  BaseUtils,\n  Entry,\n  InvalidTransactionError,\n  isValidXprv,\n  isValidXpub,\n  ParseTransactionError,\n} from '@bitgo-beta/sdk-core';\nimport { FlareNetwork } from '@bitgo-beta/statics';\nimport { Buffer } from 'buffer';\nimport { createHash } from 'crypto';\nimport { ecc } from '@bitgo-beta/secp256k1';\nimport { ADDRESS_SEPARATOR, DecodedUtxoObj, Output, SECP256K1_Transfer_Output, Tx } from './iface';\nimport bs58 from 'bs58';\nimport { bech32 } from 'bech32';\n\nexport class Utils implements BaseUtils {\n  isValidTransactionId(txId: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n  isValidSignature(signature: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n  /**\n   * Check if addresses in wallet match UTXO output addresses\n   */\n  public includeIn(walletAddresses: string[], otxoOutputAddresses: string[]): boolean {\n    return walletAddresses.map((a) => otxoOutputAddresses.includes(a)).reduce((a, b) => a && b, true);\n  }\n\n  /**\n   * Validates a Flare address or array of addresses\n   * @param {string | string[]} address - address(es) to validate\n   * @returns {boolean} - validation result\n   */\n  isValidAddress(address: string | string[]): boolean {\n    const addressArr: string[] = Array.isArray(address) ? address : address.split('~');\n\n    for (const address of addressArr) {\n      if (!this.isValidAddressRegex(address)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  private isValidAddressRegex(address: string): boolean {\n    return /^(^P||NodeID)-[a-zA-Z0-9]+$/.test(address);\n  }\n\n  /**\n   * Validates a block ID\n   * @param {string} hash - block ID to validate\n   * @returns {boolean} - validation result\n   */\n  isValidBlockId(hash: string): boolean {\n    try {\n      const decoded = Buffer.from(hash, 'hex');\n      return decoded.length === 32;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Validates a public key\n   * @param {string} pub - public key to validate\n   * @returns {boolean} - validation result\n   */\n  isValidPublicKey(pub: string): boolean {\n    if (isValidXpub(pub)) return true;\n\n    let pubBuf: Buffer;\n    if (pub.length === 50) {\n      try {\n        pubBuf = Buffer.from(pub, 'hex');\n      } catch {\n        return false;\n      }\n    } else {\n      if (pub.length !== 66 && pub.length !== 130) return false;\n\n      const firstByte = pub.slice(0, 2);\n      if (pub.length === 130 && firstByte !== '04') return false;\n      if (pub.length === 66 && firstByte !== '02' && firstByte !== '03') return false;\n      if (!this.allHexChars(pub)) return false;\n\n      pubBuf = Buffer.from(pub, 'hex');\n    }\n\n    try {\n      ecc.isPoint(pubBuf);\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Validates a private key\n   * @param {string} prv - private key to validate\n   * @returns {boolean} - validation result\n   */\n  isValidPrivateKey(prv: string): boolean {\n    if (isValidXprv(prv)) return true;\n    if (prv.length !== 64 && prv.length !== 66) return false;\n    if (prv.length === 66 && prv.slice(64) !== '01') return false;\n    return this.allHexChars(prv);\n  }\n\n  /**\n   * Checks if a string contains only hex characters\n   */\n  allHexChars(str: string): boolean {\n    return /^(0x){0,1}([0-9a-f])+$/i.test(str);\n  }\n\n  /**\n   * Creates a signature using the Flare network parameters\n   * Returns a 65-byte signature (64 bytes signature + 1 byte recovery parameter)\n   */\n  createSignature(network: FlareNetwork, message: Buffer, prv: Buffer): Buffer {\n    const messageHash = this.sha256(message);\n    const signature = ecc.sign(messageHash, prv);\n\n    // Get the public key from the private key for recovery parameter determination\n    const publicKey = ecc.pointFromScalar(prv, true);\n    if (!publicKey) {\n      throw new Error('Failed to derive public key from private key');\n    }\n\n    // Try recovery with param 0 and 1 to find the correct one\n    let recoveryParam = 0;\n    for (let i = 0; i <= 1; i++) {\n      const recovered = ecc.recoverPublicKey(messageHash, signature, i, true);\n      if (recovered && Buffer.from(recovered).equals(Buffer.from(publicKey))) {\n        recoveryParam = i;\n        break;\n      }\n    }\n\n    // Append recovery parameter to create 65-byte signature\n    const sigWithRecovery = Buffer.alloc(65);\n    Buffer.from(signature).copy(sigWithRecovery, 0);\n    sigWithRecovery[64] = recoveryParam;\n\n    return sigWithRecovery;\n  }\n\n  /**\n   * Verifies a signature\n   * @param messageHash - The SHA256 hash of the message (e.g., signablePayload)\n   * @param signature - The 64-byte signature (without recovery parameter)\n   * @param publicKey - The public key to verify against\n   * @returns true if signature is valid\n   */\n  verifySignature(messageHash: Buffer, signature: Buffer, publicKey: Buffer): boolean {\n    try {\n      return ecc.verify(messageHash, publicKey, signature);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Creates a new signature object\n   */\n  createNewSig(sigHex: string): Signature {\n    const buffer = Buffer.from(sigHex.padStart(130, '0'), 'hex');\n    return new Signature(buffer);\n  }\n\n  /**\n   * Creates an empty signature with embedded address for signature slot identification.\n   * The address is embedded at position 90 (after the first 45 zero bytes).\n   * This allows the signing logic to determine which slot belongs to which address.\n   * @param addressHex The 20-byte address in hex format (40 chars, without 0x prefix)\n   */\n  createEmptySigWithAddress(addressHex: string): Signature {\n    // First 45 bytes (90 hex chars) are zeros, followed by 20-byte address (40 hex chars)\n    const cleanAddr = this.removeHexPrefix(addressHex).toLowerCase();\n    const sigHex = '0'.repeat(90) + cleanAddr.padStart(40, '0');\n    const buffer = Buffer.from(sigHex, 'hex');\n    return new Signature(buffer);\n  }\n\n  /**\n   * Extracts the embedded address from an empty signature.\n   * Returns the address hex string (40 chars) or empty string if not found.\n   */\n  getAddressFromEmptySig(sig: string): string {\n    const cleanSig = this.removeHexPrefix(sig);\n    if (cleanSig.length >= 130) {\n      // Address is at position 90-130 (last 40 hex chars = 20 bytes)\n      return cleanSig.substring(90, 130).toLowerCase();\n    }\n    return '';\n  }\n\n  /**\n   * Computes SHA256 hash\n   */\n  sha256(buf: Uint8Array): Buffer {\n    return createHash('sha256').update(buf).digest();\n  }\n\n  /**\n   * Validates raw transaction format\n   */\n  validateRawTransaction(rawTransaction: string): void {\n    if (!rawTransaction) {\n      throw new InvalidTransactionError('Raw transaction is empty');\n    }\n    if (!this.allHexChars(rawTransaction)) {\n      throw new ParseTransactionError('Raw transaction is not hex string');\n    }\n  }\n\n  /**\n   * Checks if output is TransferableOutput type\n   */\n  isTransferableOutput(output: Output): output is TransferableOutput {\n    return output?._type === TypeSymbols.TransferableOutput;\n  }\n\n  /**\n   * Maps outputs to entry format\n   */\n  mapOutputToEntry(network: FlareNetwork): (Output) => Entry {\n    return (output: Output) => {\n      if (this.isTransferableOutput(output)) {\n        const outputAmount = output.amount();\n        const address = (output.output as TransferOutput)\n          .getOwners()\n          .map((a) => this.addressToString(network.hrp, network.alias, Buffer.from(a)))\n          .sort()\n          .join(ADDRESS_SEPARATOR);\n        return {\n          value: outputAmount.toString(),\n          address,\n        };\n      } else {\n        throw new Error('Invalid output type');\n      }\n    };\n  }\n\n  /**\n   * Removes 0x prefix from hex string\n   */\n  removeHexPrefix(hex: string): string {\n    return hex.startsWith('0x') ? hex.substring(2) : hex;\n  }\n\n  /**\n   * Converts output index to buffer\n   */\n  outputidxNumberToBuffer(outputidx: string): Buffer {\n    return Buffer.from(Number(outputidx).toString(16).padStart(8, '0'), 'hex');\n  }\n\n  /**\n   * Converts output index buffer to number string\n   */\n  outputidxBufferToNumber(outputidx: Buffer): string {\n    return parseInt(outputidx.toString('hex'), 16).toString();\n  }\n\n  /**\n   * Helper method to convert address components to string\n   */\n  public addressToString = (hrp: string, prefix: string, address: Buffer): string => {\n    // Convert the address bytes to 5-bit words for bech32 encoding\n    const words = bech32.toWords(address);\n    // Create the full bech32 address with format: P-{hrp}1{bech32_encoded_address}\n    return `${prefix}-${bech32.encode(hrp, words)}`;\n  };\n\n  /**\n   * Decodes a base58 string with checksum to a Buffer\n   */\n  public cb58Decode(str: string): Buffer {\n    const decoded = bs58.decode(str);\n    if (!this.validateChecksum(Buffer.from(decoded))) {\n      throw new Error('Invalid checksum');\n    }\n    return Buffer.from(decoded.slice(0, decoded.length - 4));\n  }\n\n  /**\n   * Validates a checksum on a Buffer and returns true if valid, false if not\n   */\n  private validateChecksum(buff: Buffer): boolean {\n    const hashSlice = buff.slice(buff.length - 4);\n    const calculatedHashSlice = createHash('sha256')\n      .update(buff.slice(0, buff.length - 4))\n      .digest()\n      .slice(28);\n    return hashSlice.toString('hex') === calculatedHashSlice.toString('hex');\n  }\n\n  /**\n   * Encodes a Buffer as a base58 string with checksum\n   */\n  public cb58Encode(bytes: Buffer): string {\n    const withChecksum = this.addChecksum(bytes);\n    return bs58.encode(withChecksum);\n  }\n\n  /**\n   * Adds a checksum to a Buffer and returns the concatenated result\n   * Uses last 4 bytes of SHA256 hash as checksum (matching avaxp behavior)\n   */\n  public addChecksum(buff: Buffer | Uint8Array): Uint8Array {\n    const buffer = Buffer.from(buff);\n    const hashSlice = createHash('sha256').update(buffer).digest().slice(28);\n    return new Uint8Array(Buffer.concat([buffer, hashSlice]));\n  }\n\n  // In utils.ts, add this method to the Utils class:\n\n  /**\n   * Parse an address string into a Buffer\n   * @param address - The address to parse\n   * @returns Buffer containing the parsed address\n   */\n  public parseAddress = (address: string): Buffer => {\n    return this.stringToAddress(address);\n  };\n\n  public stringToAddress = (address: string, hrp?: string): Buffer => {\n    // Handle hex addresses\n    if (address.startsWith('0x')) {\n      return Buffer.from(address.slice(2), 'hex');\n    }\n\n    // Handle raw hex without 0x prefix\n    if (/^[0-9a-fA-F]{40}$/.test(address)) {\n      return Buffer.from(address, 'hex');\n    }\n\n    // Handle Bech32 addresses\n    const parts = address.trim().split('-');\n    if (parts.length < 2) {\n      throw new Error('Error - Valid address should include -');\n    }\n\n    const split = parts[1].lastIndexOf('1');\n    if (split < 0) {\n      throw new Error('Error - Valid bech32 address must include separator (1)');\n    }\n\n    const humanReadablePart = parts[1].slice(0, split);\n    if (humanReadablePart !== 'flare' && humanReadablePart !== 'costwo') {\n      throw new Error('Error - Invalid HRP');\n    }\n\n    return Buffer.from(bech32.fromWords(bech32.decode(parts[1]).words));\n  };\n\n  flareIdString(value: string): Id {\n    return new Id(Buffer.from(value, 'hex'));\n  }\n\n  /**\n   * Sort addresses lexicographically by their byte representation.\n   * This matches how addresses are stored on-chain in Avalanche/Flare P-chain UTXOs.\n   *\n   * IMPORTANT: This sorting MUST be consistent with FlareJS's internal address sorting.\n   * FlareJS uses the same lexicographic comparison: `hexA.localeCompare(hexB)`.\n   * The sigIndices in transaction inputs depend on this sorted order, so any deviation\n   * would cause signature order mismatches and on-chain verification failures.\n   *\n   * @param addresses - Array of bech32 address strings (e.g., \"P-costwo1...\")\n   * @returns Array of addresses sorted by hex value (ascending lexicographic order)\n   */\n  public sortAddressesByHex(addresses: string[]): string[] {\n    return [...addresses].sort((a, b) => {\n      const aHex = this.parseAddress(a).toString('hex');\n      const bHex = this.parseAddress(b).toString('hex');\n      return aHex.localeCompare(bHex);\n    });\n  }\n\n  /**\n   * Sort address buffers lexicographically by their byte representation.\n   * This matches how addresses are stored on-chain in Avalanche/Flare P-chain UTXOs.\n   * @param addressBuffers - Array of address byte buffers\n   * @returns Array of address buffers sorted by hex value\n   */\n  public sortAddressBuffersByHex(addressBuffers: Buffer[]): Buffer[] {\n    return [...addressBuffers].sort((a, b) => {\n      return a.toString('hex').localeCompare(b.toString('hex'));\n    });\n  }\n\n  /**\n   * Recover public key from signature\n   * @param messageHash - The SHA256 hash of the message (e.g., signablePayload)\n   * @param signature - 65-byte signature (64 bytes signature + 1 byte recovery parameter)\n   * @return recovered public key\n   */\n  recoverySignature(messageHash: Buffer, signature: Buffer): Buffer {\n    try {\n      // Extract recovery parameter and signature\n      if (signature.length !== 65) {\n        throw new Error('Invalid signature length - expected 65 bytes (64 bytes signature + 1 byte recovery)');\n      }\n\n      const recoveryParam = signature[64];\n      const sigOnly = signature.slice(0, 64);\n\n      // Recover public key using the provided recovery parameter\n      // messageHash should already be the SHA256 hash (signablePayload)\n      const recovered = ecc.recoverPublicKey(messageHash, sigOnly, recoveryParam, true);\n      if (!recovered) {\n        throw new Error('Failed to recover public key');\n      }\n\n      return Buffer.from(recovered);\n    } catch (error) {\n      throw new Error(`Failed to recover signature: ${error.message}`);\n    }\n  }\n\n  /**\n   * Check if tx is for the blockchainId\n   *\n   * @param {Tx} tx\n   * @param {string} blockchainId - blockchain ID in hex format\n   * @returns true if tx is for blockchainId\n   */\n  isTransactionOf(tx: Tx, blockchainId: string): boolean {\n    // Note: getBlockchainId() and BlockchainId.value() return CB58-encoded strings,\n    // but we need hex format, so we use toBytes() and convert to hex\n    const extractBlockchainId = (txObj: any): string | null => {\n      if (typeof txObj.getTx === 'function') {\n        const innerTx = txObj.getTx();\n        if (innerTx.baseTx?.BlockchainId?.toBytes) {\n          return Buffer.from(innerTx.baseTx.BlockchainId.toBytes()).toString('hex');\n        }\n        if (innerTx.blockchainId?.toBytes) {\n          return Buffer.from(innerTx.blockchainId.toBytes()).toString('hex');\n        }\n      }\n\n      if (txObj.tx?.baseTx?.BlockchainId?.toBytes) {\n        return Buffer.from(txObj.tx.baseTx.BlockchainId.toBytes()).toString('hex');\n      }\n\n      if (txObj.baseTx?.BlockchainId?.toBytes) {\n        return Buffer.from(txObj.baseTx.BlockchainId.toBytes()).toString('hex');\n      }\n      if (txObj.blockchainId?.toBytes) {\n        return Buffer.from(txObj.blockchainId.toBytes()).toString('hex');\n      }\n\n      return null;\n    };\n\n    const txBlockchainId = extractBlockchainId(tx);\n    return txBlockchainId === blockchainId;\n  }\n\n  /**\n   * Convert FlareJS native Utxo to DecodedUtxoObj for internal use\n   * @param utxo - FlareJS Utxo object\n   * @param network - Flare network configuration\n   * @returns DecodedUtxoObj compatible with existing methods\n   */\n  public utxoToDecoded(utxo: Utxo, network: FlareNetwork): DecodedUtxoObj {\n    const outputOwners = utxo.getOutputOwners();\n    const output = utxo.output as TransferOutput;\n\n    // Get amount from output\n    const amount = output.amount().toString();\n\n    // Get txid from utxoId (cb58 encoded)\n    const txid = this.cb58Encode(Buffer.from(utxo.utxoId.txID.toBytes()));\n\n    // Get output index\n    const outputidx = utxo.utxoId.outputIdx.value().toString();\n\n    // Get threshold\n    const threshold = outputOwners.threshold.value();\n\n    // Get locktime\n    const locktime = outputOwners.locktime.value().toString();\n\n    // Get addresses as bech32 strings\n    const addresses = outputOwners.addrs.map((addr) =>\n      this.addressToString(network.hrp, network.alias, Buffer.from(addr.toBytes()))\n    );\n\n    return {\n      outputID: SECP256K1_Transfer_Output,\n      locktime,\n      amount,\n      txid,\n      outputidx,\n      threshold,\n      addresses,\n    };\n  }\n\n  /**\n   * Convert array of FlareJS Utxos to DecodedUtxoObj array\n   * @param utxos - Array of FlareJS Utxo objects\n   * @param network - Flare network configuration\n   * @returns Array of DecodedUtxoObj\n   */\n  public utxosToDecoded(utxos: Utxo[], network: FlareNetwork): DecodedUtxoObj[] {\n    return utxos.map((utxo) => this.utxoToDecoded(utxo, network));\n  }\n\n  /**\n   * Convert DecodedUtxoObj to native FlareJS Utxo object\n   * This is the reverse of utxoToDecoded\n   *\n   * IMPORTANT: Addresses are sorted lexicographically by byte value to match\n   * on-chain storage order. The API may return addresses in arbitrary order, but\n   * on-chain UTXOs always store addresses in sorted order.\n   *\n   * @param decoded - DecodedUtxoObj to convert\n   * @param assetId - Asset ID as cb58 encoded string\n   * @returns Native FlareJS Utxo object\n   */\n  public decodedToUtxo(decoded: DecodedUtxoObj, assetId: string): Utxo {\n    // Create UTXOID from txid and output index\n    const utxoId = avaxSerial.UTXOID.fromNative(decoded.txid, parseInt(decoded.outputidx, 10));\n\n    // Parse addresses from bech32 strings to byte buffers\n    const addressBytes = decoded.addresses.map((addr) => this.parseAddress(addr));\n\n    // Sort addresses lexicographically by byte value to match on-chain order\n    // This is critical because the P-chain stores addresses in sorted order,\n    // and sigIndices must reference the correct positions.\n    const sortedAddressBytes = this.sortAddressBuffersByHex(addressBytes);\n\n    // Create OutputOwners with locktime, threshold, and SORTED addresses\n    const locktime = decoded.locktime ? BigInt(decoded.locktime) : BigInt(0);\n    const outputOwners = OutputOwners.fromNative(sortedAddressBytes, locktime, decoded.threshold);\n\n    // Create TransferOutput with amount and owners\n    const amount = BigInt(decoded.amount);\n    const transferOutput = new TransferOutput(new BigIntPr(amount), outputOwners);\n\n    // Create and return the Utxo\n    return new Utxo(utxoId, Id.fromString(assetId), transferOutput);\n  }\n\n  /**\n   * Convert array of DecodedUtxoObj to native FlareJS Utxo objects\n   * @param decodedUtxos - Array of DecodedUtxoObj\n   * @param assetId - Asset ID as cb58 encoded string\n   * @returns Array of native FlareJS Utxo objects\n   */\n  public decodedToUtxos(decodedUtxos: DecodedUtxoObj[], assetId: string): Utxo[] {\n    return decodedUtxos.map((decoded) => this.decodedToUtxo(decoded, assetId));\n  }\n}\n\nconst utils = new Utils();\nexport default utils;\n"]}
|