@bitgo-beta/sdk-coin-flrp 1.0.1-beta.266 → 1.0.1-beta.268
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 +6 -75
- package/dist/src/flrp.d.ts.map +1 -1
- package/dist/src/flrp.js +13 -298
- 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 +50 -0
- package/dist/src/lib/ExportInCTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ExportInCTxBuilder.js +163 -0
- package/dist/src/lib/ExportInPTxBuilder.d.ts +36 -0
- package/dist/src/lib/ExportInPTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ExportInPTxBuilder.js +128 -0
- package/dist/src/lib/ImportInCTxBuilder.d.ts +47 -0
- package/dist/src/lib/ImportInCTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ImportInCTxBuilder.js +213 -0
- package/dist/src/lib/ImportInPTxBuilder.d.ts +23 -0
- package/dist/src/lib/ImportInPTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ImportInPTxBuilder.js +101 -0
- package/dist/src/lib/atomicInCTransactionBuilder.d.ts +18 -16
- package/dist/src/lib/atomicInCTransactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/atomicInCTransactionBuilder.js +38 -36
- package/dist/src/lib/atomicTransactionBuilder.d.ts +34 -84
- package/dist/src/lib/atomicTransactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/atomicTransactionBuilder.js +119 -288
- package/dist/src/lib/iface.d.ts +50 -51
- package/dist/src/lib/iface.d.ts.map +1 -1
- package/dist/src/lib/iface.js +22 -10
- package/dist/src/lib/index.d.ts +2 -3
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +5 -6
- package/dist/src/lib/keyPair.d.ts +1 -1
- package/dist/src/lib/keyPair.d.ts.map +1 -1
- package/dist/src/lib/keyPair.js +9 -5
- package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts +32 -67
- package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts.map +1 -1
- package/dist/src/lib/permissionlessValidatorTxBuilder.js +91 -205
- package/dist/src/lib/transaction.d.ts +8 -74
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +61 -210
- package/dist/src/lib/transactionBuilder.d.ts +56 -34
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilder.js +112 -69
- package/dist/src/lib/transactionBuilderFactory.d.ts +27 -30
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilderFactory.js +69 -75
- package/dist/src/lib/utils.d.ts +58 -157
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +134 -399
- package/dist/test/resources/transactionData/exportInC.d.ts +20 -0
- package/dist/test/resources/transactionData/exportInC.d.ts.map +1 -0
- package/dist/test/resources/transactionData/exportInC.js +30 -0
- package/dist/test/unit/lib/exportInCTxBuilder.js +92 -513
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +9 -11
- 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 -170
- package/dist/src/lib/constants.d.ts.map +0 -1
- package/dist/src/lib/constants.js +0 -227
- package/dist/src/lib/delegatorTxBuilder.d.ts +0 -58
- package/dist/src/lib/delegatorTxBuilder.d.ts.map +0 -1
- package/dist/src/lib/delegatorTxBuilder.js +0 -224
- 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 -199
- 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 -120
- package/dist/src/lib/importInCTxBuilder.d.ts +0 -67
- package/dist/src/lib/importInCTxBuilder.d.ts.map +0 -1
- package/dist/src/lib/importInCTxBuilder.js +0 -403
- package/dist/src/lib/importInPTxBuilder.d.ts +0 -73
- package/dist/src/lib/importInPTxBuilder.d.ts.map +0 -1
- package/dist/src/lib/importInPTxBuilder.js +0 -464
- package/dist/src/lib/types.d.ts +0 -78
- package/dist/src/lib/types.d.ts.map +0 -1
- package/dist/src/lib/types.js +0 -5
- package/dist/src/lib/validatorTxBuilder.d.ts +0 -40
- package/dist/src/lib/validatorTxBuilder.d.ts.map +0 -1
- package/dist/src/lib/validatorTxBuilder.js +0 -180
- package/dist/test/unit/delegatorTxBuilder.test.d.ts +0 -2
- package/dist/test/unit/delegatorTxBuilder.test.d.ts.map +0 -1
- package/dist/test/unit/delegatorTxBuilder.test.js +0 -233
- package/dist/test/unit/flrp.d.ts +0 -2
- package/dist/test/unit/flrp.d.ts.map +0 -1
- package/dist/test/unit/flrp.js +0 -118
- 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/exportInPTxBuilder.d.ts +0 -2
- package/dist/test/unit/lib/exportInPTxBuilder.d.ts.map +0 -1
- package/dist/test/unit/lib/exportInPTxBuilder.js +0 -377
- package/dist/test/unit/lib/importInCTxBuilder.d.ts +0 -2
- package/dist/test/unit/lib/importInCTxBuilder.d.ts.map +0 -1
- package/dist/test/unit/lib/importInCTxBuilder.js +0 -258
- package/dist/test/unit/lib/importInPTxBuilder.d.ts +0 -2
- package/dist/test/unit/lib/importInPTxBuilder.d.ts.map +0 -1
- package/dist/test/unit/lib/importInPTxBuilder.js +0 -501
- 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/lib/utils.d.ts +0 -2
- package/dist/test/unit/lib/utils.d.ts.map +0 -1
- package/dist/test/unit/lib/utils.js +0 -365
- package/dist/test/unit/permissionlessValidatorTxBuilder.test.d.ts +0 -2
- package/dist/test/unit/permissionlessValidatorTxBuilder.test.d.ts.map +0 -1
- package/dist/test/unit/permissionlessValidatorTxBuilder.test.js +0 -271
- 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/test/unit/transactionBuilder.test.d.ts +0 -2
- package/dist/test/unit/transactionBuilder.test.d.ts.map +0 -1
- package/dist/test/unit/transactionBuilder.test.js +0 -114
- package/dist/test/unit/validatorTxBuilder.test.d.ts +0 -2
- package/dist/test/unit/validatorTxBuilder.test.d.ts.map +0 -1
- package/dist/test/unit/validatorTxBuilder.test.js +0 -293
package/dist/src/lib/utils.js
CHANGED
|
@@ -3,61 +3,72 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
|
3
3
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
4
|
};
|
|
5
5
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
-
exports.Utils =
|
|
7
|
-
const
|
|
8
|
-
const bs58_1 = __importDefault(require("bs58"));
|
|
6
|
+
exports.Utils = void 0;
|
|
7
|
+
const flarejs_1 = require("@flarenetwork/flarejs");
|
|
9
8
|
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
10
|
-
const
|
|
9
|
+
const buffer_1 = require("buffer");
|
|
11
10
|
const crypto_1 = require("crypto");
|
|
12
|
-
const
|
|
13
|
-
|
|
14
|
-
const
|
|
15
|
-
|
|
16
|
-
return new RegExp(pattern);
|
|
17
|
-
};
|
|
18
|
-
exports.createHexRegex = createHexRegex;
|
|
19
|
-
const createFlexibleHexRegex = (requirePrefix = false) => {
|
|
20
|
-
const pattern = requirePrefix ? `^0x${constants_1.HEX_CHAR_PATTERN}+$` : constants_1.HEX_PATTERN_NO_PREFIX;
|
|
21
|
-
return new RegExp(pattern);
|
|
22
|
-
};
|
|
23
|
-
exports.createFlexibleHexRegex = createFlexibleHexRegex;
|
|
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");
|
|
24
15
|
class Utils {
|
|
25
16
|
constructor() {
|
|
17
|
+
/**
|
|
18
|
+
* Helper method to convert address components to string
|
|
19
|
+
*/
|
|
26
20
|
this.addressToString = (hrp, prefix, address) => {
|
|
27
21
|
// Convert the address bytes to 5-bit words for bech32 encoding
|
|
28
22
|
const words = bech32_1.bech32.toWords(address);
|
|
29
23
|
// Create the full bech32 address with format: P-{hrp}1{bech32_encoded_address}
|
|
30
24
|
return `${prefix}-${bech32_1.bech32.encode(hrp, words)}`;
|
|
31
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
|
+
//TODO: need check and validate this method
|
|
32
33
|
this.parseAddress = (address) => {
|
|
33
34
|
return this.stringToAddress(address);
|
|
34
35
|
};
|
|
35
36
|
this.stringToAddress = (address, hrp) => {
|
|
37
|
+
// Handle hex addresses
|
|
38
|
+
if (address.startsWith('0x')) {
|
|
39
|
+
return buffer_1.Buffer.from(address.slice(2), 'hex');
|
|
40
|
+
}
|
|
41
|
+
// Handle raw hex without 0x prefix
|
|
42
|
+
if (/^[0-9a-fA-F]{40}$/.test(address)) {
|
|
43
|
+
return buffer_1.Buffer.from(address, 'hex');
|
|
44
|
+
}
|
|
45
|
+
// Handle Bech32 addresses
|
|
36
46
|
const parts = address.trim().split('-');
|
|
37
47
|
if (parts.length < 2) {
|
|
38
48
|
throw new Error('Error - Valid address should include -');
|
|
39
49
|
}
|
|
40
50
|
const split = parts[1].lastIndexOf('1');
|
|
41
51
|
if (split < 0) {
|
|
42
|
-
throw new Error('Error - Valid address must include separator (1)');
|
|
52
|
+
throw new Error('Error - Valid bech32 address must include separator (1)');
|
|
43
53
|
}
|
|
44
54
|
const humanReadablePart = parts[1].slice(0, split);
|
|
45
55
|
if (humanReadablePart !== 'flare' && humanReadablePart !== 'costwo') {
|
|
46
56
|
throw new Error('Error - Invalid HRP');
|
|
47
57
|
}
|
|
48
|
-
return Buffer.from(bech32_1.bech32.fromWords(bech32_1.bech32.decode(parts[1]).words));
|
|
58
|
+
return buffer_1.Buffer.from(bech32_1.bech32.fromWords(bech32_1.bech32.decode(parts[1]).words));
|
|
49
59
|
};
|
|
50
60
|
}
|
|
61
|
+
/**
|
|
62
|
+
* Check if addresses in wallet match UTXO output addresses
|
|
63
|
+
*/
|
|
51
64
|
includeIn(walletAddresses, otxoOutputAddresses) {
|
|
52
65
|
return walletAddresses.map((a) => otxoOutputAddresses.includes(a)).reduce((a, b) => a && b, true);
|
|
53
66
|
}
|
|
54
67
|
/**
|
|
55
|
-
*
|
|
56
|
-
*
|
|
57
|
-
* @
|
|
58
|
-
* @returns {boolean} - the validation result
|
|
68
|
+
* Validates a Flare address or array of addresses
|
|
69
|
+
* @param {string | string[]} address - address(es) to validate
|
|
70
|
+
* @returns {boolean} - validation result
|
|
59
71
|
*/
|
|
60
|
-
/** @inheritdoc */
|
|
61
72
|
isValidAddress(address) {
|
|
62
73
|
const addressArr = Array.isArray(address) ? address : address.split('~');
|
|
63
74
|
for (const address of addressArr) {
|
|
@@ -67,48 +78,57 @@ class Utils {
|
|
|
67
78
|
}
|
|
68
79
|
return true;
|
|
69
80
|
}
|
|
81
|
+
// Regex patterns
|
|
82
|
+
// export const ADDRESS_REGEX = /^(^P||NodeID)-[a-zA-Z0-9]+$/;
|
|
83
|
+
// export const HEX_REGEX = /^(0x){0,1}([0-9a-f])+$/i;
|
|
70
84
|
isValidAddressRegex(address) {
|
|
71
|
-
return
|
|
85
|
+
return /^(^P||NodeID)-[a-zA-Z0-9]+$/.test(address);
|
|
72
86
|
}
|
|
73
87
|
/**
|
|
74
|
-
*
|
|
75
|
-
*
|
|
76
|
-
*
|
|
77
|
-
|
|
78
|
-
|
|
88
|
+
* Validates a block ID
|
|
89
|
+
* @param {string} hash - block ID to validate
|
|
90
|
+
* @returns {boolean} - validation result
|
|
91
|
+
*/
|
|
92
|
+
isValidBlockId(hash) {
|
|
93
|
+
try {
|
|
94
|
+
const decoded = buffer_1.Buffer.from(hash, 'hex');
|
|
95
|
+
return decoded.length === 32;
|
|
96
|
+
}
|
|
97
|
+
catch {
|
|
98
|
+
return false;
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Validates a public key
|
|
103
|
+
* @param {string} pub - public key to validate
|
|
104
|
+
* @returns {boolean} - validation result
|
|
79
105
|
*/
|
|
80
106
|
isValidPublicKey(pub) {
|
|
81
107
|
if ((0, sdk_core_1.isValidXpub)(pub))
|
|
82
108
|
return true;
|
|
83
109
|
let pubBuf;
|
|
84
|
-
if (pub.length ===
|
|
110
|
+
if (pub.length === 50) {
|
|
85
111
|
try {
|
|
86
|
-
pubBuf =
|
|
112
|
+
pubBuf = buffer_1.Buffer.from(pub, 'hex');
|
|
87
113
|
}
|
|
88
114
|
catch {
|
|
89
115
|
return false;
|
|
90
116
|
}
|
|
91
117
|
}
|
|
92
118
|
else {
|
|
93
|
-
if (pub.length !==
|
|
119
|
+
if (pub.length !== 66 && pub.length !== 130)
|
|
94
120
|
return false;
|
|
95
|
-
}
|
|
96
121
|
const firstByte = pub.slice(0, 2);
|
|
97
|
-
|
|
98
|
-
if (pub.length === constants_1.UNCOMPRESSED_PUBLIC_KEY_LENGTH && firstByte !== '04') {
|
|
122
|
+
if (pub.length === 130 && firstByte !== '04')
|
|
99
123
|
return false;
|
|
100
|
-
|
|
101
|
-
// compressed public key
|
|
102
|
-
if (pub.length === constants_1.COMPRESSED_PUBLIC_KEY_LENGTH && firstByte !== '02' && firstByte !== '03') {
|
|
124
|
+
if (pub.length === 66 && firstByte !== '02' && firstByte !== '03')
|
|
103
125
|
return false;
|
|
104
|
-
}
|
|
105
126
|
if (!this.allHexChars(pub))
|
|
106
127
|
return false;
|
|
107
|
-
pubBuf = Buffer.from(pub, 'hex');
|
|
128
|
+
pubBuf = buffer_1.Buffer.from(pub, 'hex');
|
|
108
129
|
}
|
|
109
|
-
// validate the public key using BitGo secp256k1
|
|
110
130
|
try {
|
|
111
|
-
secp256k1_1.ecc.isPoint(pubBuf);
|
|
131
|
+
secp256k1_1.ecc.isPoint(pubBuf);
|
|
112
132
|
return true;
|
|
113
133
|
}
|
|
114
134
|
catch (e) {
|
|
@@ -116,319 +136,87 @@ class Utils {
|
|
|
116
136
|
}
|
|
117
137
|
}
|
|
118
138
|
/**
|
|
119
|
-
*
|
|
120
|
-
* private key
|
|
121
|
-
*
|
|
122
|
-
* The protocol key format is described in the @stacks/transactions npm package, in the
|
|
123
|
-
* createStacksPrivateKey function:
|
|
124
|
-
* https://github.com/blockstack/stacks.js/blob/master/packages/transactions/src/keys.ts#L125
|
|
125
|
-
*
|
|
126
|
-
* @param {string} prv - the private key (or extended private key) to be validated
|
|
127
|
-
* @returns {boolean} - the validation result
|
|
139
|
+
* Validates a private key
|
|
140
|
+
* @param {string} prv - private key to validate
|
|
141
|
+
* @returns {boolean} - validation result
|
|
128
142
|
*/
|
|
129
143
|
isValidPrivateKey(prv) {
|
|
130
144
|
if ((0, sdk_core_1.isValidXprv)(prv))
|
|
131
145
|
return true;
|
|
132
|
-
if (prv.length !==
|
|
146
|
+
if (prv.length !== 64 && prv.length !== 66)
|
|
133
147
|
return false;
|
|
134
|
-
|
|
135
|
-
if (prv.length === constants_1.SUFFIXED_PRIVATE_KEY_LENGTH &&
|
|
136
|
-
prv.slice(constants_1.RAW_PRIVATE_KEY_LENGTH) !== constants_1.PRIVATE_KEY_COMPRESSED_SUFFIX) {
|
|
148
|
+
if (prv.length === 66 && prv.slice(64) !== '01')
|
|
137
149
|
return false;
|
|
138
|
-
}
|
|
139
150
|
return this.allHexChars(prv);
|
|
140
151
|
}
|
|
141
152
|
/**
|
|
142
|
-
*
|
|
143
|
-
*
|
|
144
|
-
* @param {string} maybe - the string to be validated
|
|
145
|
-
* @returns {boolean} - the validation result
|
|
153
|
+
* Checks if a string contains only hex characters
|
|
146
154
|
*/
|
|
147
|
-
allHexChars(
|
|
148
|
-
return
|
|
155
|
+
allHexChars(str) {
|
|
156
|
+
return /^(0x){0,1}([0-9a-f])+$/i.test(str);
|
|
149
157
|
}
|
|
150
158
|
/**
|
|
151
|
-
*
|
|
152
|
-
* Validates that an address is a 40-character hex string (optionally prefixed with 0x)
|
|
153
|
-
*
|
|
154
|
-
* @param {string} address - the Ethereum address to validate
|
|
155
|
-
* @returns {boolean} - true if valid Ethereum address format
|
|
156
|
-
*/
|
|
157
|
-
isValidEthereumAddress(address) {
|
|
158
|
-
if (!address || typeof address !== constants_1.STRING_TYPE) {
|
|
159
|
-
return false;
|
|
160
|
-
}
|
|
161
|
-
// Remove 0x prefix if present
|
|
162
|
-
const cleanAddress = address.startsWith('0x') ? address.slice(2) : address;
|
|
163
|
-
// Check if it's exactly 40 hex characters
|
|
164
|
-
return cleanAddress.length === 40 && /^[0-9a-fA-F]{40}$/.test(cleanAddress);
|
|
165
|
-
}
|
|
166
|
-
/**
|
|
167
|
-
* Pick specific properties from an object (replaces lodash.pick)
|
|
168
|
-
*
|
|
169
|
-
* @param {T} obj - the source object
|
|
170
|
-
* @param {K[]} keys - array of property keys to pick
|
|
171
|
-
* @returns {Pick<T, K>} - new object with only the specified properties
|
|
172
|
-
*/
|
|
173
|
-
pick(obj, keys) {
|
|
174
|
-
const result = {};
|
|
175
|
-
for (const key of keys) {
|
|
176
|
-
if (Object.prototype.hasOwnProperty.call(obj, key)) {
|
|
177
|
-
result[key] = obj[key];
|
|
178
|
-
}
|
|
179
|
-
}
|
|
180
|
-
return result;
|
|
181
|
-
}
|
|
182
|
-
/**
|
|
183
|
-
* Deep equality comparison (replaces lodash.isEqual)
|
|
184
|
-
*
|
|
185
|
-
* @param {unknown} a - first value to compare
|
|
186
|
-
* @param {unknown} b - second value to compare
|
|
187
|
-
* @returns {boolean} - true if values are deeply equal
|
|
188
|
-
*/
|
|
189
|
-
isEqual(a, b) {
|
|
190
|
-
if (a === b)
|
|
191
|
-
return true;
|
|
192
|
-
if (a === null || a === undefined || b === null || b === undefined)
|
|
193
|
-
return a === b;
|
|
194
|
-
if (typeof a !== typeof b)
|
|
195
|
-
return false;
|
|
196
|
-
if (typeof a === 'object') {
|
|
197
|
-
if (Array.isArray(a) !== Array.isArray(b))
|
|
198
|
-
return false;
|
|
199
|
-
if (Array.isArray(a)) {
|
|
200
|
-
const arrB = b;
|
|
201
|
-
if (a.length !== arrB.length)
|
|
202
|
-
return false;
|
|
203
|
-
for (let i = 0; i < a.length; i++) {
|
|
204
|
-
if (!this.isEqual(a[i], arrB[i]))
|
|
205
|
-
return false;
|
|
206
|
-
}
|
|
207
|
-
return true;
|
|
208
|
-
}
|
|
209
|
-
const objA = a;
|
|
210
|
-
const objB = b;
|
|
211
|
-
const keysA = Object.keys(objA);
|
|
212
|
-
const keysB = Object.keys(objB);
|
|
213
|
-
if (keysA.length !== keysB.length)
|
|
214
|
-
return false;
|
|
215
|
-
for (const key of keysA) {
|
|
216
|
-
if (!keysB.includes(key))
|
|
217
|
-
return false;
|
|
218
|
-
if (!this.isEqual(objA[key], objB[key]))
|
|
219
|
-
return false;
|
|
220
|
-
}
|
|
221
|
-
return true;
|
|
222
|
-
}
|
|
223
|
-
return false;
|
|
224
|
-
}
|
|
225
|
-
/** @inheritdoc */
|
|
226
|
-
isValidSignature(signature) {
|
|
227
|
-
throw new sdk_core_1.NotImplementedError('isValidSignature not implemented');
|
|
228
|
-
}
|
|
229
|
-
/** @inheritdoc */
|
|
230
|
-
isValidTransactionId(txId) {
|
|
231
|
-
return this.isValidId(txId);
|
|
232
|
-
}
|
|
233
|
-
/** @inheritdoc */
|
|
234
|
-
isValidBlockId(blockId) {
|
|
235
|
-
return this.isValidId(blockId);
|
|
236
|
-
}
|
|
237
|
-
/**
|
|
238
|
-
* FlareJS wrapper to create signature and return it for credentials
|
|
239
|
-
* @param network
|
|
240
|
-
* @param message
|
|
241
|
-
* @param prv
|
|
242
|
-
* @return signature
|
|
159
|
+
* Creates a signature using the Flare network parameters
|
|
243
160
|
*/
|
|
244
161
|
createSignature(network, message, prv) {
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
// It is essential that the same hashing (sha256 of the message) is applied during signature recovery,
|
|
249
|
-
// otherwise the recovered public key or signature verification will fail.
|
|
250
|
-
const messageHash = (0, crypto_1.createHash)('sha256').update(message).digest();
|
|
251
|
-
// Sign with recovery parameter
|
|
252
|
-
const signature = secp256k1_1.ecc.sign(messageHash, prv);
|
|
253
|
-
// Get recovery parameter by trying both values
|
|
254
|
-
let recoveryParam = -1;
|
|
255
|
-
const pubKey = secp256k1_1.ecc.pointFromScalar(prv, true);
|
|
256
|
-
if (!pubKey) {
|
|
257
|
-
throw new Error('Failed to derive public key from private key');
|
|
258
|
-
}
|
|
259
|
-
const recovered0 = secp256k1_1.ecc.recoverPublicKey(messageHash, signature, 0, true);
|
|
260
|
-
if (recovered0 && Buffer.from(recovered0).equals(Buffer.from(pubKey))) {
|
|
261
|
-
recoveryParam = 0;
|
|
262
|
-
}
|
|
263
|
-
else {
|
|
264
|
-
const recovered1 = secp256k1_1.ecc.recoverPublicKey(messageHash, signature, 1, true);
|
|
265
|
-
if (recovered1 && Buffer.from(recovered1).equals(Buffer.from(pubKey))) {
|
|
266
|
-
recoveryParam = 1;
|
|
267
|
-
}
|
|
268
|
-
else {
|
|
269
|
-
throw new Error('Could not determine correct recovery parameter for signature');
|
|
270
|
-
}
|
|
271
|
-
}
|
|
272
|
-
// Append recovery parameter to signature
|
|
273
|
-
const fullSig = Buffer.alloc(65); // 64 bytes signature + 1 byte recovery
|
|
274
|
-
fullSig.set(signature);
|
|
275
|
-
fullSig[64] = recoveryParam;
|
|
276
|
-
return fullSig;
|
|
277
|
-
}
|
|
278
|
-
catch (error) {
|
|
279
|
-
throw new Error(`Failed to create signature: ${error}`);
|
|
280
|
-
}
|
|
162
|
+
const messageHash = this.sha256(message);
|
|
163
|
+
const signature = secp256k1_1.ecc.sign(messageHash, prv);
|
|
164
|
+
return buffer_1.Buffer.from(signature);
|
|
281
165
|
}
|
|
282
166
|
/**
|
|
283
|
-
*
|
|
284
|
-
* @param network
|
|
285
|
-
* @param message
|
|
286
|
-
* @param signature
|
|
287
|
-
* @param publicKey - public key instead of private key for verification
|
|
288
|
-
* @return true if it's verify successful
|
|
167
|
+
* Verifies a signature
|
|
289
168
|
*/
|
|
290
169
|
verifySignature(network, message, signature, publicKey) {
|
|
291
170
|
try {
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
// Extract the actual signature without recovery parameter
|
|
295
|
-
if (signature.length !== 65) {
|
|
296
|
-
throw new Error('Invalid signature length - expected 65 bytes (64 bytes signature + 1 byte recovery)');
|
|
297
|
-
}
|
|
298
|
-
const sigOnly = signature.slice(0, 64);
|
|
299
|
-
return secp256k1_1.ecc.verify(messageHash, publicKey, sigOnly);
|
|
171
|
+
const messageHash = this.sha256(message);
|
|
172
|
+
return secp256k1_1.ecc.verify(signature, messageHash, publicKey);
|
|
300
173
|
}
|
|
301
|
-
catch (
|
|
174
|
+
catch (e) {
|
|
302
175
|
return false;
|
|
303
176
|
}
|
|
304
177
|
}
|
|
305
178
|
/**
|
|
306
|
-
*
|
|
307
|
-
* @param network
|
|
308
|
-
* @param message
|
|
309
|
-
* @param signature
|
|
310
|
-
* @return recovered public key
|
|
179
|
+
* Creates a new signature object
|
|
311
180
|
*/
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
const messageHash = (0, crypto_1.createHash)('sha256').update(message).digest();
|
|
316
|
-
// Extract recovery parameter and signature
|
|
317
|
-
if (signature.length !== 65) {
|
|
318
|
-
throw new Error('Invalid signature length - expected 65 bytes (64 bytes signature + 1 byte recovery)');
|
|
319
|
-
}
|
|
320
|
-
const recoveryParam = signature[64];
|
|
321
|
-
const sigOnly = signature.slice(0, 64);
|
|
322
|
-
// Recover public key using the provided recovery parameter
|
|
323
|
-
const recovered = secp256k1_1.ecc.recoverPublicKey(messageHash, sigOnly, recoveryParam, true);
|
|
324
|
-
if (!recovered) {
|
|
325
|
-
throw new Error('Failed to recover public key');
|
|
326
|
-
}
|
|
327
|
-
return Buffer.from(recovered);
|
|
328
|
-
}
|
|
329
|
-
catch (error) {
|
|
330
|
-
throw new Error(`Failed to recover signature: ${error}`);
|
|
331
|
-
}
|
|
181
|
+
createNewSig(sigHex) {
|
|
182
|
+
const buffer = buffer_1.Buffer.from(sigHex.padStart(130, '0'), 'hex');
|
|
183
|
+
return new flarejs_1.Signature(buffer);
|
|
332
184
|
}
|
|
185
|
+
/**
|
|
186
|
+
* Computes SHA256 hash
|
|
187
|
+
*/
|
|
333
188
|
sha256(buf) {
|
|
334
189
|
return (0, crypto_1.createHash)('sha256').update(buf).digest();
|
|
335
190
|
}
|
|
336
191
|
/**
|
|
337
|
-
*
|
|
338
|
-
* It's to reuse in TransactionBuilder and TransactionBuilderFactory
|
|
339
|
-
*
|
|
340
|
-
* @param rawTransaction Transaction as hex string
|
|
192
|
+
* Validates raw transaction format
|
|
341
193
|
*/
|
|
342
194
|
validateRawTransaction(rawTransaction) {
|
|
343
195
|
if (!rawTransaction) {
|
|
344
196
|
throw new sdk_core_1.InvalidTransactionError('Raw transaction is empty');
|
|
345
197
|
}
|
|
346
|
-
if (!
|
|
198
|
+
if (!this.allHexChars(rawTransaction)) {
|
|
347
199
|
throw new sdk_core_1.ParseTransactionError('Raw transaction is not hex string');
|
|
348
200
|
}
|
|
349
201
|
}
|
|
350
202
|
/**
|
|
351
|
-
*
|
|
352
|
-
*
|
|
353
|
-
* @param {DeprecatedTx} tx
|
|
354
|
-
* @param {string} blockchainId
|
|
355
|
-
* @returns true if tx is for blockchainId
|
|
356
|
-
*/
|
|
357
|
-
isTransactionOf(tx, blockchainId) {
|
|
358
|
-
// FlareJS equivalent - this would need proper CB58 encoding implementation
|
|
359
|
-
try {
|
|
360
|
-
const txRecord = tx;
|
|
361
|
-
const unsignedTx = txRecord.getUnsignedTx();
|
|
362
|
-
const transaction = unsignedTx.getTransaction();
|
|
363
|
-
const txBlockchainId = transaction.getBlockchainID();
|
|
364
|
-
return Buffer.from(txBlockchainId).toString(constants_1.HEX_ENCODING) === blockchainId;
|
|
365
|
-
}
|
|
366
|
-
catch (error) {
|
|
367
|
-
return false;
|
|
368
|
-
}
|
|
369
|
-
}
|
|
370
|
-
/**
|
|
371
|
-
* Check if Output is from PVM.
|
|
372
|
-
* Output could be EVM or PVM output.
|
|
373
|
-
* @param {DeprecatedOutput} output
|
|
374
|
-
* @returns {boolean} output has transferable output structure
|
|
375
|
-
*/
|
|
376
|
-
deprecatedIsTransferableOutput(output) {
|
|
377
|
-
return 'getOutput' in output;
|
|
378
|
-
}
|
|
379
|
-
/**
|
|
380
|
-
* Check if Output is from PVM.
|
|
381
|
-
* Output could be EVM or PVM output.
|
|
382
|
-
* @param {Output} output
|
|
383
|
-
* @returns {boolean} output is TransferableOutput
|
|
203
|
+
* Checks if output is TransferableOutput type
|
|
384
204
|
*/
|
|
385
205
|
isTransferableOutput(output) {
|
|
386
|
-
return
|
|
387
|
-
}
|
|
388
|
-
/**
|
|
389
|
-
* Return a mapper function to that network address representation.
|
|
390
|
-
* @param network required to stringify addresses
|
|
391
|
-
* @return mapper function
|
|
392
|
-
*/
|
|
393
|
-
deprecatedMapOutputToEntry(network) {
|
|
394
|
-
return (output) => {
|
|
395
|
-
if (this.deprecatedIsTransferableOutput(output)) {
|
|
396
|
-
// Simplified implementation for FlareJS
|
|
397
|
-
try {
|
|
398
|
-
const transferableOutput = output;
|
|
399
|
-
const amount = transferableOutput.amount();
|
|
400
|
-
// Simplified address handling - would need proper FlareJS address utilities
|
|
401
|
-
const address = constants_1.FLARE_ADDRESS_PLACEHOLDER; // TODO: implement proper address conversion
|
|
402
|
-
return {
|
|
403
|
-
value: amount.toString(),
|
|
404
|
-
address,
|
|
405
|
-
};
|
|
406
|
-
}
|
|
407
|
-
catch (error) {
|
|
408
|
-
throw new Error(`Failed to map output: ${error}`);
|
|
409
|
-
}
|
|
410
|
-
}
|
|
411
|
-
else {
|
|
412
|
-
// Handle EVM output case - simplified
|
|
413
|
-
return {
|
|
414
|
-
value: '0', // TODO: implement proper amount extraction
|
|
415
|
-
address: '0x0000000000000000000000000000000000000000', // TODO: implement proper address extraction
|
|
416
|
-
};
|
|
417
|
-
}
|
|
418
|
-
};
|
|
206
|
+
return output?._type === flarejs_1.TypeSymbols.TransferableOutput;
|
|
419
207
|
}
|
|
420
208
|
/**
|
|
421
|
-
*
|
|
422
|
-
* @param network required to stringify addresses
|
|
423
|
-
* @return mapper function
|
|
209
|
+
* Maps outputs to entry format
|
|
424
210
|
*/
|
|
425
211
|
mapOutputToEntry(network) {
|
|
426
212
|
return (output) => {
|
|
427
213
|
if (this.isTransferableOutput(output)) {
|
|
428
|
-
const
|
|
429
|
-
const
|
|
430
|
-
|
|
431
|
-
|
|
214
|
+
const outputAmount = output.amount();
|
|
215
|
+
const address = output.output
|
|
216
|
+
.getOwners()
|
|
217
|
+
.map((a) => this.addressToString(network.hrp, network.alias, buffer_1.Buffer.from(a)))
|
|
218
|
+
.sort()
|
|
219
|
+
.join(iface_1.ADDRESS_SEPARATOR);
|
|
432
220
|
return {
|
|
433
221
|
value: outputAmount.toString(),
|
|
434
222
|
address,
|
|
@@ -440,93 +228,39 @@ class Utils {
|
|
|
440
228
|
};
|
|
441
229
|
}
|
|
442
230
|
/**
|
|
443
|
-
*
|
|
444
|
-
* @param hex string
|
|
445
|
-
* @returns hex without 0x
|
|
231
|
+
* Removes 0x prefix from hex string
|
|
446
232
|
*/
|
|
447
233
|
removeHexPrefix(hex) {
|
|
448
|
-
|
|
449
|
-
return hex.substring(2);
|
|
450
|
-
}
|
|
451
|
-
return hex;
|
|
234
|
+
return hex.startsWith('0x') ? hex.substring(2) : hex;
|
|
452
235
|
}
|
|
453
236
|
/**
|
|
454
|
-
*
|
|
455
|
-
* @param {string} outputidx number
|
|
456
|
-
* @return {Buffer} buffer of size 4 with that number value
|
|
237
|
+
* Converts output index to buffer
|
|
457
238
|
*/
|
|
458
239
|
outputidxNumberToBuffer(outputidx) {
|
|
459
|
-
return Buffer.from(Number(outputidx).toString(
|
|
240
|
+
return buffer_1.Buffer.from(Number(outputidx).toString(16).padStart(8, '0'), 'hex');
|
|
460
241
|
}
|
|
461
242
|
/**
|
|
462
|
-
*
|
|
463
|
-
* @param {Buffer} outputidx
|
|
464
|
-
* @return {string} outputidx number
|
|
243
|
+
* Converts output index buffer to number string
|
|
465
244
|
*/
|
|
466
245
|
outputidxBufferToNumber(outputidx) {
|
|
467
|
-
return parseInt(outputidx.toString(
|
|
246
|
+
return parseInt(outputidx.toString('hex'), 16).toString();
|
|
468
247
|
}
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
* @param {string} text - Text to convert
|
|
473
|
-
* @returns {Uint8Array} Byte array
|
|
474
|
-
*/
|
|
475
|
-
stringToBytes(text) {
|
|
476
|
-
return new TextEncoder().encode(text);
|
|
477
|
-
}
|
|
478
|
-
/**
|
|
479
|
-
* Convert bytes to string from FlareJS memo
|
|
480
|
-
* @param {Uint8Array} bytes - Bytes to convert
|
|
481
|
-
* @returns {string} Decoded string
|
|
482
|
-
*/
|
|
483
|
-
bytesToString(bytes) {
|
|
484
|
-
return new TextDecoder().decode(bytes);
|
|
248
|
+
// Required by BaseUtils interface but not implemented
|
|
249
|
+
isValidSignature(signature) {
|
|
250
|
+
throw new sdk_core_1.NotImplementedError('isValidSignature not implemented');
|
|
485
251
|
}
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
* Supports string, JSON object, or raw bytes
|
|
489
|
-
* @param {string | Record<string, unknown> | Uint8Array} memo - Memo data
|
|
490
|
-
* @returns {Uint8Array} Memo bytes for FlareJS
|
|
491
|
-
*/
|
|
492
|
-
createMemoBytes(memo) {
|
|
493
|
-
if (memo instanceof Uint8Array) {
|
|
494
|
-
return memo;
|
|
495
|
-
}
|
|
496
|
-
if (typeof memo === constants_1.STRING_TYPE) {
|
|
497
|
-
return this.stringToBytes(memo);
|
|
498
|
-
}
|
|
499
|
-
if (typeof memo === 'object') {
|
|
500
|
-
return this.stringToBytes(JSON.stringify(memo));
|
|
501
|
-
}
|
|
502
|
-
throw new sdk_core_1.InvalidTransactionError('Invalid memo format');
|
|
252
|
+
isValidTransactionId(txId) {
|
|
253
|
+
throw new sdk_core_1.NotImplementedError('isValidTransactionId not implemented');
|
|
503
254
|
}
|
|
504
255
|
/**
|
|
505
|
-
*
|
|
506
|
-
* @param {Uint8Array} memoBytes - Memo bytes from FlareJS transaction
|
|
507
|
-
* @returns {string} Decoded memo string
|
|
256
|
+
* Decodes a base58 string with checksum to a Buffer
|
|
508
257
|
*/
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
258
|
+
cb58Decode(str) {
|
|
259
|
+
const decoded = bs58_1.default.decode(str);
|
|
260
|
+
if (!this.validateChecksum(buffer_1.Buffer.from(decoded))) {
|
|
261
|
+
throw new Error('Invalid checksum');
|
|
512
262
|
}
|
|
513
|
-
return
|
|
514
|
-
}
|
|
515
|
-
/**
|
|
516
|
-
* Validate memo size (FlareJS has transaction size limits)
|
|
517
|
-
* @param {Uint8Array} memoBytes - Memo bytes
|
|
518
|
-
* @param {number} maxSize - Maximum size in bytes (default 4KB)
|
|
519
|
-
* @returns {boolean} Whether memo is within size limits
|
|
520
|
-
*/
|
|
521
|
-
validateMemoSize(memoBytes, maxSize = 4096) {
|
|
522
|
-
return memoBytes.length <= maxSize;
|
|
523
|
-
}
|
|
524
|
-
/**
|
|
525
|
-
* Adds a checksum to a Buffer and returns the concatenated result
|
|
526
|
-
*/
|
|
527
|
-
addChecksum(buff) {
|
|
528
|
-
const hashSlice = (0, crypto_1.createHash)('sha256').update(buff).digest().slice(28);
|
|
529
|
-
return Buffer.concat([buff, hashSlice]);
|
|
263
|
+
return buffer_1.Buffer.from(decoded.slice(0, decoded.length - 4));
|
|
530
264
|
}
|
|
531
265
|
/**
|
|
532
266
|
* Validates a checksum on a Buffer and returns true if valid, false if not
|
|
@@ -547,37 +281,38 @@ class Utils {
|
|
|
547
281
|
return bs58_1.default.encode(withChecksum);
|
|
548
282
|
}
|
|
549
283
|
/**
|
|
550
|
-
*
|
|
284
|
+
* Adds a checksum to a Buffer and returns the concatenated result
|
|
551
285
|
*/
|
|
552
|
-
|
|
553
|
-
const
|
|
554
|
-
|
|
555
|
-
throw new Error('Invalid checksum');
|
|
556
|
-
}
|
|
557
|
-
return Buffer.from(decoded.slice(0, decoded.length - 4));
|
|
286
|
+
addChecksum(buff) {
|
|
287
|
+
const hashSlice = (0, crypto_1.createHash)('sha256').update(buff).digest().slice(28);
|
|
288
|
+
return buffer_1.Buffer.concat([buff, hashSlice]);
|
|
558
289
|
}
|
|
559
290
|
/**
|
|
560
|
-
*
|
|
291
|
+
* Check if tx is for the blockchainId
|
|
292
|
+
*
|
|
293
|
+
* @param {DeprecatedTx} tx
|
|
294
|
+
* @param {string} blockchainId
|
|
295
|
+
* @returns true if tx is for blockchainId
|
|
561
296
|
*/
|
|
562
|
-
|
|
297
|
+
// TODO: remove DeprecatedTx usage
|
|
298
|
+
isTransactionOf(tx, blockchainId) {
|
|
299
|
+
// FlareJS equivalent - this would need proper CB58 encoding implementation
|
|
563
300
|
try {
|
|
564
|
-
|
|
565
|
-
|
|
301
|
+
const txRecord = tx;
|
|
302
|
+
const unsignedTx = txRecord.getUnsignedTx();
|
|
303
|
+
const transaction = unsignedTx.getTransaction();
|
|
304
|
+
const txBlockchainId = transaction.getBlockchainID();
|
|
305
|
+
return buffer_1.Buffer.from(txBlockchainId).toString('hex') === blockchainId;
|
|
566
306
|
}
|
|
567
|
-
catch {
|
|
307
|
+
catch (error) {
|
|
568
308
|
return false;
|
|
569
309
|
}
|
|
570
310
|
}
|
|
571
|
-
|
|
572
|
-
|
|
573
|
-
return this.isCB58(id) && this.cb58Decode(id).length === constants_1.DECODED_BLOCK_ID_LENGTH;
|
|
574
|
-
}
|
|
575
|
-
catch {
|
|
576
|
-
return false;
|
|
577
|
-
}
|
|
311
|
+
flareIdString(value) {
|
|
312
|
+
return new flarejs_1.Id(buffer_1.Buffer.from(value, 'hex'));
|
|
578
313
|
}
|
|
579
314
|
}
|
|
580
315
|
exports.Utils = Utils;
|
|
581
316
|
const utils = new Utils();
|
|
582
317
|
exports.default = utils;
|
|
583
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { TransferableOutput } from '@flarenetwork/flarejs';\nimport { bech32 } from 'bech32';\nimport bs58 from 'bs58';\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  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  HEX_CHAR_PATTERN,\n  HEX_PATTERN_NO_PREFIX,\n  FLARE_ADDRESS_PLACEHOLDER,\n  HEX_ENCODING,\n  PADSTART_CHAR,\n  HEX_RADIX,\n  STRING_TYPE,\n  DECODED_BLOCK_ID_LENGTH,\n} from './constants';\n\n// Regex utility functions for hex validation\nexport const createHexRegex = (length: number, requirePrefix = false): RegExp => {\n  const pattern = requirePrefix ? `^0x${HEX_CHAR_PATTERN}{${length}}$` : `^${HEX_CHAR_PATTERN}{${length}}$`;\n  return new RegExp(pattern);\n};\n\nexport const createFlexibleHexRegex = (requirePrefix = false): RegExp => {\n  const pattern = requirePrefix ? `^0x${HEX_CHAR_PATTERN}+$` : HEX_PATTERN_NO_PREFIX;\n  return new RegExp(pattern);\n};\n\nexport class Utils implements BaseUtils {\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  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 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        pubBuf = this.cb58Decode(pub);\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 = (address: string): Buffer => {\n    return this.stringToAddress(address);\n  };\n\n  public stringToAddress = (address: string, hrp?: string): Buffer => {\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 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  /**\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_TYPE) {\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    return this.isValidId(txId);\n  }\n\n  /** @inheritdoc */\n  isValidBlockId(blockId: string): boolean {\n    return this.isValidId(blockId);\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    // Used BitGo secp256k1 since FlareJS may not expose KeyPair in the same way\n    try {\n      // Hash the message first: secp256k1 signing requires a 32-byte hash as input.\n      // It is essential that the same hashing (sha256 of the message) is applied during signature recovery,\n      // otherwise the recovered public key or signature verification will fail.\n      const messageHash = createHash('sha256').update(message).digest();\n\n      // Sign with recovery parameter\n      const signature = ecc.sign(messageHash, prv);\n\n      // Get recovery parameter by trying both values\n      let recoveryParam = -1;\n      const pubKey = ecc.pointFromScalar(prv, true);\n      if (!pubKey) {\n        throw new Error('Failed to derive public key from private key');\n      }\n      const recovered0 = ecc.recoverPublicKey(messageHash, signature, 0, true);\n      if (recovered0 && Buffer.from(recovered0).equals(Buffer.from(pubKey))) {\n        recoveryParam = 0;\n      } else {\n        const recovered1 = ecc.recoverPublicKey(messageHash, signature, 1, true);\n        if (recovered1 && Buffer.from(recovered1).equals(Buffer.from(pubKey))) {\n          recoveryParam = 1;\n        } else {\n          throw new Error('Could not determine correct recovery parameter for signature');\n        }\n      }\n\n      // Append recovery parameter to signature\n      const fullSig = Buffer.alloc(65); // 64 bytes signature + 1 byte recovery\n      fullSig.set(signature);\n      fullSig[64] = recoveryParam;\n\n      return fullSig;\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      // Hash the message first - must match the hash used in signing\n      const messageHash = createHash('sha256').update(message).digest();\n\n      // Extract the actual signature without recovery parameter\n      if (signature.length !== 65) {\n        throw new Error('Invalid signature length - expected 65 bytes (64 bytes signature + 1 byte recovery)');\n      }\n      const sigOnly = signature.slice(0, 64);\n\n      return ecc.verify(messageHash, publicKey, sigOnly);\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      // Hash the message first - must match the hash used in signing\n      const messageHash = createHash('sha256').update(message).digest();\n\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      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}`);\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_ENCODING) === 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(\n      Number(outputidx).toString(HEX_RADIX).padStart(OUTPUT_INDEX_HEX_LENGTH, PADSTART_CHAR),\n      HEX_ENCODING\n    );\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_ENCODING), HEX_RADIX).toString();\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_TYPE) {\n      return this.stringToBytes(memo as string);\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  /**\n   * Adds a checksum to a Buffer and returns the concatenated result\n   */\n  private addChecksum(buff: Buffer): Buffer {\n    const hashSlice = createHash('sha256').update(buff).digest().slice(28);\n    return Buffer.concat([buff, hashSlice]);\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   * 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   * Checks if a string is a valid CB58 (base58 with checksum) format\n   */\n  private isCB58(str: string): boolean {\n    try {\n      this.cb58Decode(str);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  isValidId(id: string): boolean {\n    try {\n      return this.isCB58(id) && this.cb58Decode(id).length === DECODED_BLOCK_ID_LENGTH;\n    } catch {\n      return false;\n    }\n  }\n}\n\nconst utils = new Utils();\n\nexport default utils;\n"]}
|
|
318
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":";;;;;;AAAA,mDAAuG;AACvG,mDAQ8B;AAE9B,mCAAgC;AAChC,mCAAoC;AACpC,qDAA4C;AAC5C,mCAAkE;AAClE,gDAAwB;AACxB,mCAAgC;AAEhC,MAAa,KAAK;IAAlB;QA+ME;;WAEG;QACI,oBAAe,GAAG,CAAC,GAAW,EAAE,MAAc,EAAE,OAAe,EAAU,EAAE;YAChF,+DAA+D;YAC/D,MAAM,KAAK,GAAG,eAAM,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;YACtC,+EAA+E;YAC/E,OAAO,GAAG,MAAM,IAAI,eAAM,CAAC,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,CAAC;QAClD,CAAC,CAAC;QAyCF,mDAAmD;QAEnD;;;;WAIG;QACH,2CAA2C;QACpC,iBAAY,GAAG,CAAC,OAAe,EAAU,EAAE;YAChD,OAAO,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC;QACvC,CAAC,CAAC;QAEK,oBAAe,GAAG,CAAC,OAAe,EAAE,GAAY,EAAU,EAAE;YACjE,uBAAuB;YACvB,IAAI,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC7B,OAAO,eAAM,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;YAC9C,CAAC;YAED,mCAAmC;YACnC,IAAI,mBAAmB,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;gBACtC,OAAO,eAAM,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;YACrC,CAAC;YAED,0BAA0B;YAC1B,MAAM,KAAK,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACxC,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACrB,MAAM,IAAI,KAAK,CAAC,wCAAwC,CAAC,CAAC;YAC5D,CAAC;YAED,MAAM,KAAK,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;YACxC,IAAI,KAAK,GAAG,CAAC,EAAE,CAAC;gBACd,MAAM,IAAI,KAAK,CAAC,yDAAyD,CAAC,CAAC;YAC7E,CAAC;YAED,MAAM,iBAAiB,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;YACnD,IAAI,iBAAiB,KAAK,OAAO,IAAI,iBAAiB,KAAK,QAAQ,EAAE,CAAC;gBACpE,MAAM,IAAI,KAAK,CAAC,qBAAqB,CAAC,CAAC;YACzC,CAAC;YAED,OAAO,eAAM,CAAC,IAAI,CAAC,eAAM,CAAC,SAAS,CAAC,eAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;QACtE,CAAC,CAAC;IA0BJ,CAAC;IAjUC;;OAEG;IACI,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;;;;OAIG;IACH,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;IAED,iBAAiB;IACjB,8DAA8D;IAC9D,sDAAsD;IAE9C,mBAAmB,CAAC,OAAe;QACzC,OAAO,6BAA6B,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACrD,CAAC;IAED;;;;OAIG;IACH,cAAc,CAAC,IAAY;QACzB,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,eAAM,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;YACzC,OAAO,OAAO,CAAC,MAAM,KAAK,EAAE,CAAC;QAC/B,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED;;;;OAIG;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,EAAE,EAAE,CAAC;YACtB,IAAI,CAAC;gBACH,MAAM,GAAG,eAAM,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;YACnC,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,KAAK,CAAC;YACf,CAAC;QACH,CAAC;aAAM,CAAC;YACN,IAAI,GAAG,CAAC,MAAM,KAAK,EAAE,IAAI,GAAG,CAAC,MAAM,KAAK,GAAG;gBAAE,OAAO,KAAK,CAAC;YAE1D,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;YAClC,IAAI,GAAG,CAAC,MAAM,KAAK,GAAG,IAAI,SAAS,KAAK,IAAI;gBAAE,OAAO,KAAK,CAAC;YAC3D,IAAI,GAAG,CAAC,MAAM,KAAK,EAAE,IAAI,SAAS,KAAK,IAAI,IAAI,SAAS,KAAK,IAAI;gBAAE,OAAO,KAAK,CAAC;YAChF,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC;gBAAE,OAAO,KAAK,CAAC;YAEzC,MAAM,GAAG,eAAM,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QACnC,CAAC;QAED,IAAI,CAAC;YACH,eAAG,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;YACpB,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED;;;;OAIG;IACH,iBAAiB,CAAC,GAAW;QAC3B,IAAI,IAAA,sBAAW,EAAC,GAAG,CAAC;YAAE,OAAO,IAAI,CAAC;QAClC,IAAI,GAAG,CAAC,MAAM,KAAK,EAAE,IAAI,GAAG,CAAC,MAAM,KAAK,EAAE;YAAE,OAAO,KAAK,CAAC;QACzD,IAAI,GAAG,CAAC,MAAM,KAAK,EAAE,IAAI,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,IAAI;YAAE,OAAO,KAAK,CAAC;QAC9D,OAAO,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;IAC/B,CAAC;IAED;;OAEG;IACH,WAAW,CAAC,GAAW;QACrB,OAAO,yBAAyB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAC7C,CAAC;IAED;;OAEG;IACH,eAAe,CAAC,OAAqB,EAAE,OAAe,EAAE,GAAW;QACjE,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QACzC,MAAM,SAAS,GAAG,eAAG,CAAC,IAAI,CAAC,WAAW,EAAE,GAAG,CAAC,CAAC;QAC7C,OAAO,eAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAChC,CAAC;IAED;;OAEG;IACH,eAAe,CAAC,OAAqB,EAAE,OAAe,EAAE,SAAiB,EAAE,SAAiB;QAC1F,IAAI,CAAC;YACH,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YACzC,OAAO,eAAG,CAAC,MAAM,CAAC,SAAS,EAAE,WAAW,EAAE,SAAS,CAAC,CAAC;QACvD,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED;;OAEG;IACH,YAAY,CAAC,MAAc;QACzB,MAAM,MAAM,GAAG,eAAM,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;QAC7D,OAAO,IAAI,mBAAS,CAAC,MAAM,CAAC,CAAC;IAC/B,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,GAAe;QACpB,OAAO,IAAA,mBAAU,EAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,CAAC;IACnD,CAAC;IAED;;OAEG;IACH,sBAAsB,CAAC,cAAsB;QAC3C,IAAI,CAAC,cAAc,EAAE,CAAC;YACpB,MAAM,IAAI,kCAAuB,CAAC,0BAA0B,CAAC,CAAC;QAChE,CAAC;QACD,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,EAAE,CAAC;YACtC,MAAM,IAAI,gCAAqB,CAAC,mCAAmC,CAAC,CAAC;QACvE,CAAC;IACH,CAAC;IAED;;OAEG;IACH,oBAAoB,CAAC,MAAc;QACjC,OAAO,MAAM,EAAE,KAAK,KAAK,qBAAW,CAAC,kBAAkB,CAAC;IAC1D,CAAC;IAED;;OAEG;IACH,gBAAgB,CAAC,OAAqB;QACpC,OAAO,CAAC,MAAc,EAAE,EAAE;YACxB,IAAI,IAAI,CAAC,oBAAoB,CAAC,MAAM,CAAC,EAAE,CAAC;gBACtC,MAAM,YAAY,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC;gBACrC,MAAM,OAAO,GAAI,MAAM,CAAC,MAAyB;qBAC9C,SAAS,EAAE;qBACX,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,GAAG,EAAE,OAAO,CAAC,KAAK,EAAE,eAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;qBAC5E,IAAI,EAAE;qBACN,IAAI,CAAC,yBAAiB,CAAC,CAAC;gBAC3B,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;;OAEG;IACH,eAAe,CAAC,GAAW;QACzB,OAAO,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;IACvD,CAAC;IAED;;OAEG;IACH,uBAAuB,CAAC,SAAiB;QACvC,OAAO,eAAM,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;IAC7E,CAAC;IAED;;OAEG;IACH,uBAAuB,CAAC,SAAiB;QACvC,OAAO,QAAQ,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC;IAC5D,CAAC;IAED,sDAAsD;IACtD,gBAAgB,CAAC,SAAiB;QAChC,MAAM,IAAI,8BAAmB,CAAC,kCAAkC,CAAC,CAAC;IACpE,CAAC;IAED,oBAAoB,CAAC,IAAY;QAC/B,MAAM,IAAI,8BAAmB,CAAC,sCAAsC,CAAC,CAAC;IACxE,CAAC;IAYD;;OAEG;IACI,UAAU,CAAC,GAAW;QAC3B,MAAM,OAAO,GAAG,cAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QACjC,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,eAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,EAAE,CAAC;YACjD,MAAM,IAAI,KAAK,CAAC,kBAAkB,CAAC,CAAC;QACtC,CAAC;QACD,OAAO,eAAM,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC;IAC3D,CAAC;IAED;;OAEG;IACK,gBAAgB,CAAC,IAAY;QACnC,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAC9C,MAAM,mBAAmB,GAAG,IAAA,mBAAU,EAAC,QAAQ,CAAC;aAC7C,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;aACtC,MAAM,EAAE;aACR,KAAK,CAAC,EAAE,CAAC,CAAC;QACb,OAAO,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,KAAK,mBAAmB,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IAC3E,CAAC;IAED;;OAEG;IACI,UAAU,CAAC,KAAa;QAC7B,MAAM,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;QAC7C,OAAO,cAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;IACnC,CAAC;IAED;;OAEG;IACK,WAAW,CAAC,IAAY;QAC9B,MAAM,SAAS,GAAG,IAAA,mBAAU,EAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QACvE,OAAO,eAAM,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC,CAAC;IAC1C,CAAC;IA4CD;;;;;;OAMG;IACH,kCAAkC;IAClC,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,eAAM,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,aAAa,CAAC,KAAa;QACzB,OAAO,IAAI,YAAE,CAAC,eAAM,CAAC,IAAI,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC;IAC3C,CAAC;CACF;AAlUD,sBAkUC;AAED,MAAM,KAAK,GAAG,IAAI,KAAK,EAAE,CAAC;AAC1B,kBAAe,KAAK,CAAC","sourcesContent":["import { Signature, TransferableOutput, TransferOutput, TypeSymbols, Id } 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 { Buffer } from 'buffer';\nimport { createHash } from 'crypto';\nimport { ecc } from '@bitgo-beta/secp256k1';\nimport { ADDRESS_SEPARATOR, Output, DeprecatedTx } from './iface';\nimport bs58 from 'bs58';\nimport { bech32 } from 'bech32';\n\nexport class Utils implements BaseUtils {\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  // Regex patterns\n  // export const ADDRESS_REGEX = /^(^P||NodeID)-[a-zA-Z0-9]+$/;\n  // export const HEX_REGEX = /^(0x){0,1}([0-9a-f])+$/i;\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   */\n  createSignature(network: FlareNetwork, message: Buffer, prv: Buffer): Buffer {\n    const messageHash = this.sha256(message);\n    const signature = ecc.sign(messageHash, prv);\n    return Buffer.from(signature);\n  }\n\n  /**\n   * Verifies a signature\n   */\n  verifySignature(network: FlareNetwork, message: Buffer, signature: Buffer, publicKey: Buffer): boolean {\n    try {\n      const messageHash = this.sha256(message);\n      return ecc.verify(signature, messageHash, publicKey);\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   * 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  // Required by BaseUtils interface but not implemented\n  isValidSignature(signature: string): boolean {\n    throw new NotImplementedError('isValidSignature not implemented');\n  }\n\n  isValidTransactionId(txId: string): boolean {\n    throw new NotImplementedError('isValidTransactionId not implemented');\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   */\n  private addChecksum(buff: Buffer): Buffer {\n    const hashSlice = createHash('sha256').update(buff).digest().slice(28);\n    return Buffer.concat([buff, 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  //TODO: need check and validate this method\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  /**\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  // TODO: remove DeprecatedTx usage\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  flareIdString(value: string): Id {\n    return new Id(Buffer.from(value, 'hex'));\n  }\n}\n\nconst utils = new Utils();\nexport default utils;\n"]}
|