@bitgo-beta/sdk-coin-flrp 1.0.1-beta.29 → 1.0.1-beta.290

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (118) hide show
  1. package/dist/src/flrp.d.ts +6 -75
  2. package/dist/src/flrp.d.ts.map +1 -1
  3. package/dist/src/flrp.js +13 -298
  4. package/dist/src/index.d.ts +0 -1
  5. package/dist/src/index.d.ts.map +1 -1
  6. package/dist/src/index.js +1 -2
  7. package/dist/src/lib/ExportInCTxBuilder.d.ts +51 -0
  8. package/dist/src/lib/ExportInCTxBuilder.d.ts.map +1 -0
  9. package/dist/src/lib/ExportInCTxBuilder.js +186 -0
  10. package/dist/src/lib/ExportInPTxBuilder.d.ts +47 -0
  11. package/dist/src/lib/ExportInPTxBuilder.d.ts.map +1 -0
  12. package/dist/src/lib/ExportInPTxBuilder.js +273 -0
  13. package/dist/src/lib/ImportInCTxBuilder.d.ts +48 -0
  14. package/dist/src/lib/ImportInCTxBuilder.d.ts.map +1 -0
  15. package/dist/src/lib/ImportInCTxBuilder.js +215 -0
  16. package/dist/src/lib/ImportInPTxBuilder.d.ts +33 -0
  17. package/dist/src/lib/ImportInPTxBuilder.d.ts.map +1 -0
  18. package/dist/src/lib/ImportInPTxBuilder.js +180 -0
  19. package/dist/src/lib/atomicInCTransactionBuilder.d.ts +18 -16
  20. package/dist/src/lib/atomicInCTransactionBuilder.d.ts.map +1 -1
  21. package/dist/src/lib/atomicInCTransactionBuilder.js +38 -36
  22. package/dist/src/lib/atomicTransactionBuilder.d.ts +42 -76
  23. package/dist/src/lib/atomicTransactionBuilder.d.ts.map +1 -1
  24. package/dist/src/lib/atomicTransactionBuilder.js +141 -218
  25. package/dist/src/lib/iface.d.ts +50 -51
  26. package/dist/src/lib/iface.d.ts.map +1 -1
  27. package/dist/src/lib/iface.js +22 -10
  28. package/dist/src/lib/index.d.ts +4 -0
  29. package/dist/src/lib/index.d.ts.map +1 -1
  30. package/dist/src/lib/index.js +10 -2
  31. package/dist/src/lib/keyPair.d.ts +5 -5
  32. package/dist/src/lib/keyPair.d.ts.map +1 -1
  33. package/dist/src/lib/keyPair.js +17 -9
  34. package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts +46 -0
  35. package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts.map +1 -0
  36. package/dist/src/lib/permissionlessValidatorTxBuilder.js +134 -0
  37. package/dist/src/lib/transaction.d.ts +8 -73
  38. package/dist/src/lib/transaction.d.ts.map +1 -1
  39. package/dist/src/lib/transaction.js +157 -207
  40. package/dist/src/lib/transactionBuilder.d.ts +107 -0
  41. package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
  42. package/dist/src/lib/transactionBuilder.js +210 -0
  43. package/dist/src/lib/transactionBuilderFactory.d.ts +35 -30
  44. package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
  45. package/dist/src/lib/transactionBuilderFactory.js +133 -74
  46. package/dist/src/lib/utils.d.ts +74 -147
  47. package/dist/src/lib/utils.d.ts.map +1 -1
  48. package/dist/src/lib/utils.js +226 -323
  49. package/dist/test/resources/account.d.ts +49 -0
  50. package/dist/test/resources/account.d.ts.map +1 -0
  51. package/dist/test/resources/account.js +52 -0
  52. package/dist/test/resources/transactionData/exportInC.d.ts +20 -0
  53. package/dist/test/resources/transactionData/exportInC.d.ts.map +1 -0
  54. package/dist/test/resources/transactionData/exportInC.js +34 -0
  55. package/dist/test/resources/transactionData/exportInP.d.ts +69 -0
  56. package/dist/test/resources/transactionData/exportInP.d.ts.map +1 -0
  57. package/dist/test/resources/transactionData/exportInP.js +140 -0
  58. package/dist/test/resources/transactionData/importInC.d.ts +27 -0
  59. package/dist/test/resources/transactionData/importInC.d.ts.map +1 -0
  60. package/dist/test/resources/transactionData/importInC.js +44 -0
  61. package/dist/test/resources/transactionData/importInP.d.ts +35 -0
  62. package/dist/test/resources/transactionData/importInP.d.ts.map +1 -0
  63. package/dist/test/resources/transactionData/importInP.js +58 -0
  64. package/dist/test/unit/lib/exportInCTxBuilder.d.ts +2 -0
  65. package/dist/test/unit/lib/exportInCTxBuilder.d.ts.map +1 -0
  66. package/dist/test/unit/lib/exportInCTxBuilder.js +166 -0
  67. package/dist/test/unit/lib/exportInPTxBuilder.d.ts +2 -0
  68. package/dist/test/unit/lib/exportInPTxBuilder.d.ts.map +1 -0
  69. package/dist/test/unit/lib/exportInPTxBuilder.js +121 -0
  70. package/dist/test/unit/lib/importInCTxBuilder.d.ts +2 -0
  71. package/dist/test/unit/lib/importInCTxBuilder.d.ts.map +1 -0
  72. package/dist/test/unit/lib/importInCTxBuilder.js +47 -0
  73. package/dist/test/unit/lib/importInPTxBuilder.d.ts +2 -0
  74. package/dist/test/unit/lib/importInPTxBuilder.d.ts.map +1 -0
  75. package/dist/test/unit/lib/importInPTxBuilder.js +73 -0
  76. package/dist/test/unit/lib/keyPair.d.ts +2 -0
  77. package/dist/test/unit/lib/keyPair.d.ts.map +1 -0
  78. package/dist/test/unit/lib/keyPair.js +158 -0
  79. package/dist/test/unit/lib/signFlowTestSuit.d.ts +20 -0
  80. package/dist/test/unit/lib/signFlowTestSuit.d.ts.map +1 -0
  81. package/dist/test/unit/lib/signFlowTestSuit.js +89 -0
  82. package/dist/test/unit/lib/utils.js +29 -223
  83. package/dist/tsconfig.tsbuildinfo +1 -1
  84. package/package.json +16 -13
  85. package/.eslintignore +0 -5
  86. package/.eslintrc.json +0 -7
  87. package/.mocharc.yml +0 -8
  88. package/CHANGELOG.md +0 -0
  89. package/dist/src/iface.d.ts +0 -25
  90. package/dist/src/iface.d.ts.map +0 -1
  91. package/dist/src/iface.js +0 -3
  92. package/dist/src/lib/constants.d.ts +0 -11
  93. package/dist/src/lib/constants.d.ts.map +0 -1
  94. package/dist/src/lib/constants.js +0 -17
  95. package/dist/src/lib/errors.d.ts +0 -8
  96. package/dist/src/lib/errors.d.ts.map +0 -1
  97. package/dist/src/lib/errors.js +0 -19
  98. package/dist/src/lib/exportInCTxBuilder.d.ts +0 -77
  99. package/dist/src/lib/exportInCTxBuilder.d.ts.map +0 -1
  100. package/dist/src/lib/exportInCTxBuilder.js +0 -170
  101. package/dist/src/lib/exportInPTxBuilder.d.ts +0 -30
  102. package/dist/src/lib/exportInPTxBuilder.d.ts.map +0 -1
  103. package/dist/src/lib/exportInPTxBuilder.js +0 -56
  104. package/dist/test/unit/flrp.d.ts +0 -2
  105. package/dist/test/unit/flrp.d.ts.map +0 -1
  106. package/dist/test/unit/flrp.js +0 -118
  107. package/dist/test/unit/lib/atomicTransactionBuilder.d.ts +0 -2
  108. package/dist/test/unit/lib/atomicTransactionBuilder.d.ts.map +0 -1
  109. package/dist/test/unit/lib/atomicTransactionBuilder.js +0 -222
  110. package/dist/test/unit/lib/exportTxBuilder.d.ts +0 -2
  111. package/dist/test/unit/lib/exportTxBuilder.d.ts.map +0 -1
  112. package/dist/test/unit/lib/exportTxBuilder.js +0 -45
  113. package/dist/test/unit/lib/transaction.d.ts +0 -2
  114. package/dist/test/unit/lib/transaction.d.ts.map +0 -1
  115. package/dist/test/unit/lib/transaction.js +0 -460
  116. package/dist/test/unit/smoke.d.ts +0 -2
  117. package/dist/test/unit/smoke.d.ts.map +0 -1
  118. package/dist/test/unit/smoke.js +0 -23
@@ -1,27 +1,74 @@
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 secp256k1_1 = require("@bitgo-beta/secp256k1");
9
+ const buffer_1 = require("buffer");
6
10
  const crypto_1 = require("crypto");
7
- const constants_1 = require("./constants");
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
- this.parseAddress = (pub) => {
11
- // FlareJS equivalent for address parsing
12
- return Buffer.from(pub, 'hex'); // Simplified implementation
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
+ //TODO: need check and validate this method
33
+ this.parseAddress = (address) => {
34
+ return this.stringToAddress(address);
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
46
+ const parts = address.trim().split('-');
47
+ if (parts.length < 2) {
48
+ throw new Error('Error - Valid address should include -');
49
+ }
50
+ const split = parts[1].lastIndexOf('1');
51
+ if (split < 0) {
52
+ throw new Error('Error - Valid bech32 address must include separator (1)');
53
+ }
54
+ const humanReadablePart = parts[1].slice(0, split);
55
+ if (humanReadablePart !== 'flare' && humanReadablePart !== 'costwo') {
56
+ throw new Error('Error - Invalid HRP');
57
+ }
58
+ return buffer_1.Buffer.from(bech32_1.bech32.fromWords(bech32_1.bech32.decode(parts[1]).words));
13
59
  };
14
60
  }
61
+ /**
62
+ * Check if addresses in wallet match UTXO output addresses
63
+ */
15
64
  includeIn(walletAddresses, otxoOutputAddresses) {
16
65
  return walletAddresses.map((a) => otxoOutputAddresses.includes(a)).reduce((a, b) => a && b, true);
17
66
  }
18
67
  /**
19
- * Checks if it is a valid address no illegal characters
20
- *
21
- * @param {string} address - address to be validated
22
- * @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
23
71
  */
24
- /** @inheritdoc */
25
72
  isValidAddress(address) {
26
73
  const addressArr = Array.isArray(address) ? address : address.split('~');
27
74
  for (const address of addressArr) {
@@ -31,66 +78,57 @@ class Utils {
31
78
  }
32
79
  return true;
33
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;
34
84
  isValidAddressRegex(address) {
35
- return constants_1.ADDRESS_REGEX.test(address);
85
+ return /^(^P||NodeID)-[a-zA-Z0-9]+$/.test(address);
36
86
  }
37
87
  /**
38
- * Checks if it is a valid blockId with length 66 including 0x
39
- *
40
- * @param {string} hash - blockId to be validated
41
- * @returns {boolean} - the validation result
88
+ * Validates a block ID
89
+ * @param {string} hash - block ID to validate
90
+ * @returns {boolean} - validation result
42
91
  */
43
- /** @inheritdoc */
44
92
  isValidBlockId(hash) {
45
- // FlareJS equivalent - check if it's a valid CB58 hash with correct length
46
93
  try {
47
- const decoded = Buffer.from(hash); // FlareJS should provide CB58 utilities
48
- return decoded.length === constants_1.DECODED_BLOCK_ID_LENGTH;
94
+ const decoded = buffer_1.Buffer.from(hash, 'hex');
95
+ return decoded.length === 32;
49
96
  }
50
97
  catch {
51
98
  return false;
52
99
  }
53
100
  }
54
101
  /**
55
- * Checks if the string is a valid protocol public key or
56
- * extended public key.
57
- *
58
- * @param {string} pub - the public key to be validated
59
- * @returns {boolean} - the validation result
102
+ * Validates a public key
103
+ * @param {string} pub - public key to validate
104
+ * @returns {boolean} - validation result
60
105
  */
61
106
  isValidPublicKey(pub) {
62
107
  if ((0, sdk_core_1.isValidXpub)(pub))
63
108
  return true;
64
109
  let pubBuf;
65
- if (pub.length === constants_1.SHORT_PUB_KEY_LENGTH) {
110
+ if (pub.length === 50) {
66
111
  try {
67
- // For FlareJS, we'll need to implement CB58 decode functionality
68
- pubBuf = Buffer.from(pub, 'hex'); // Temporary placeholder
112
+ pubBuf = buffer_1.Buffer.from(pub, 'hex');
69
113
  }
70
114
  catch {
71
115
  return false;
72
116
  }
73
117
  }
74
118
  else {
75
- if (pub.length !== constants_1.COMPRESSED_PUBLIC_KEY_LENGTH && pub.length !== constants_1.UNCOMPRESSED_PUBLIC_KEY_LENGTH) {
119
+ if (pub.length !== 66 && pub.length !== 130)
76
120
  return false;
77
- }
78
121
  const firstByte = pub.slice(0, 2);
79
- // uncompressed public key
80
- if (pub.length === constants_1.UNCOMPRESSED_PUBLIC_KEY_LENGTH && firstByte !== '04') {
122
+ if (pub.length === 130 && firstByte !== '04')
81
123
  return false;
82
- }
83
- // compressed public key
84
- if (pub.length === constants_1.COMPRESSED_PUBLIC_KEY_LENGTH && firstByte !== '02' && firstByte !== '03') {
124
+ if (pub.length === 66 && firstByte !== '02' && firstByte !== '03')
85
125
  return false;
86
- }
87
126
  if (!this.allHexChars(pub))
88
127
  return false;
89
- pubBuf = Buffer.from(pub, 'hex');
128
+ pubBuf = buffer_1.Buffer.from(pub, 'hex');
90
129
  }
91
- // validate the public key using BitGo secp256k1
92
130
  try {
93
- secp256k1_1.ecc.isPoint(pubBuf); // Check if it's a valid point
131
+ secp256k1_1.ecc.isPoint(pubBuf);
94
132
  return true;
95
133
  }
96
134
  catch (e) {
@@ -98,269 +136,131 @@ class Utils {
98
136
  }
99
137
  }
100
138
  /**
101
- * Returns whether or not the string is a valid protocol private key, or extended
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
139
+ * Validates a private key
140
+ * @param {string} prv - private key to validate
141
+ * @returns {boolean} - validation result
110
142
  */
111
143
  isValidPrivateKey(prv) {
112
144
  if ((0, sdk_core_1.isValidXprv)(prv))
113
145
  return true;
114
- if (prv.length !== constants_1.RAW_PRIVATE_KEY_LENGTH && prv.length !== constants_1.SUFFIXED_PRIVATE_KEY_LENGTH) {
146
+ if (prv.length !== 64 && prv.length !== 66)
115
147
  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) {
148
+ if (prv.length === 66 && prv.slice(64) !== '01')
119
149
  return false;
120
- }
121
150
  return this.allHexChars(prv);
122
151
  }
123
152
  /**
124
- * Returns whether or not the string is a composed of hex chars only
125
- *
126
- * @param {string} maybe - the string to be validated
127
- * @returns {boolean} - the validation result
153
+ * Checks if a string contains only hex characters
128
154
  */
129
- allHexChars(maybe) {
130
- return constants_1.HEX_REGEX.test(maybe);
155
+ allHexChars(str) {
156
+ return /^(0x){0,1}([0-9a-f])+$/i.test(str);
131
157
  }
132
158
  /**
133
- * Lightweight Ethereum address validation
134
- * Validates that an address is a 40-character hex string (optionally prefixed with 0x)
135
- *
136
- * @param {string} address - the Ethereum address to validate
137
- * @returns {boolean} - true if valid Ethereum address format
159
+ * Creates a signature using the Flare network parameters
160
+ * Returns a 65-byte signature (64 bytes signature + 1 byte recovery parameter)
138
161
  */
139
- isValidEthereumAddress(address) {
140
- if (!address || typeof address !== 'string') {
141
- return false;
162
+ createSignature(network, message, prv) {
163
+ const messageHash = this.sha256(message);
164
+ const signature = secp256k1_1.ecc.sign(messageHash, prv);
165
+ // Get the public key from the private key for recovery parameter determination
166
+ const publicKey = secp256k1_1.ecc.pointFromScalar(prv, true);
167
+ if (!publicKey) {
168
+ throw new Error('Failed to derive public key from private key');
142
169
  }
143
- // Remove 0x prefix if present
144
- const cleanAddress = address.startsWith('0x') ? address.slice(2) : address;
145
- // Check if it's exactly 40 hex characters
146
- return cleanAddress.length === 40 && /^[0-9a-fA-F]{40}$/.test(cleanAddress);
147
- }
148
- /**
149
- * Pick specific properties from an object (replaces lodash.pick)
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];
170
+ // Try recovery with param 0 and 1 to find the correct one
171
+ let recoveryParam = 0;
172
+ for (let i = 0; i <= 1; i++) {
173
+ const recovered = secp256k1_1.ecc.recoverPublicKey(messageHash, signature, i, true);
174
+ if (recovered && buffer_1.Buffer.from(recovered).equals(buffer_1.Buffer.from(publicKey))) {
175
+ recoveryParam = i;
176
+ break;
160
177
  }
161
178
  }
162
- return result;
179
+ // Append recovery parameter to create 65-byte signature
180
+ const sigWithRecovery = buffer_1.Buffer.alloc(65);
181
+ buffer_1.Buffer.from(signature).copy(sigWithRecovery, 0);
182
+ sigWithRecovery[64] = recoveryParam;
183
+ return sigWithRecovery;
163
184
  }
164
185
  /**
165
- * Deep equality comparison (replaces lodash.isEqual)
166
- *
167
- * @param {unknown} a - first value to compare
168
- * @param {unknown} b - second value to compare
169
- * @returns {boolean} - true if values are deeply equal
186
+ * Verifies a signature
170
187
  */
171
- isEqual(a, b) {
172
- if (a === b)
173
- return true;
174
- if (a === null || a === undefined || b === null || b === undefined)
175
- return a === b;
176
- if (typeof a !== typeof b)
188
+ verifySignature(network, message, signature, publicKey) {
189
+ try {
190
+ const messageHash = this.sha256(message);
191
+ return secp256k1_1.ecc.verify(signature, messageHash, publicKey);
192
+ }
193
+ catch (e) {
177
194
  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
195
  }
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
196
  }
215
197
  /**
216
- * FlareJS wrapper to create signature and return it for credentials
217
- * @param network
218
- * @param message
219
- * @param prv
220
- * @return signature
198
+ * Creates a new signature object
221
199
  */
222
- createSignature(network, message, prv) {
223
- // Use BitGo secp256k1 since FlareJS may not expose KeyPair in the same way
224
- try {
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
- }
200
+ createNewSig(sigHex) {
201
+ const buffer = buffer_1.Buffer.from(sigHex.padStart(130, '0'), 'hex');
202
+ return new flarejs_1.Signature(buffer);
231
203
  }
232
204
  /**
233
- * FlareJS wrapper to verify signature
234
- * @param network
235
- * @param message
236
- * @param signature
237
- * @param publicKey - public key instead of private key for verification
238
- * @return true if it's verify successful
205
+ * Creates an empty signature with embedded address for signature slot identification.
206
+ * The address is embedded at position 90 (after the first 45 zero bytes).
207
+ * This allows the signing logic to determine which slot belongs to which address.
208
+ * @param addressHex The 20-byte address in hex format (40 chars, without 0x prefix)
239
209
  */
240
- verifySignature(network, message, signature, publicKey) {
241
- try {
242
- return secp256k1_1.ecc.verify(message, publicKey, signature);
243
- }
244
- catch (error) {
245
- return false;
246
- }
210
+ createEmptySigWithAddress(addressHex) {
211
+ // First 45 bytes (90 hex chars) are zeros, followed by 20-byte address (40 hex chars)
212
+ const cleanAddr = this.removeHexPrefix(addressHex).toLowerCase();
213
+ const sigHex = '0'.repeat(90) + cleanAddr.padStart(40, '0');
214
+ const buffer = buffer_1.Buffer.from(sigHex, 'hex');
215
+ return new flarejs_1.Signature(buffer);
247
216
  }
248
217
  /**
249
- * FlareJS wrapper to recover signature
250
- * @param network
251
- * @param message
252
- * @param signature
253
- * @return recovered public key
218
+ * Extracts the embedded address from an empty signature.
219
+ * Returns the address hex string (40 chars) or empty string if not found.
254
220
  */
255
- recoverySignature(network, message, signature) {
256
- try {
257
- // This would need to be implemented with secp256k1 recovery
258
- // For now, throwing error since recovery logic would need to be adapted
259
- throw new sdk_core_1.NotImplementedError('recoverySignature not fully implemented for FlareJS');
260
- }
261
- catch (error) {
262
- throw new Error(`Failed to recover signature: ${error}`);
221
+ getAddressFromEmptySig(sig) {
222
+ const cleanSig = this.removeHexPrefix(sig);
223
+ if (cleanSig.length >= 130) {
224
+ // Address is at position 90-130 (last 40 hex chars = 20 bytes)
225
+ return cleanSig.substring(90, 130).toLowerCase();
263
226
  }
227
+ return '';
264
228
  }
229
+ /**
230
+ * Computes SHA256 hash
231
+ */
265
232
  sha256(buf) {
266
233
  return (0, crypto_1.createHash)('sha256').update(buf).digest();
267
234
  }
268
235
  /**
269
- * Check the raw transaction has a valid format in the blockchain context, throw otherwise.
270
- * It's to reuse in TransactionBuilder and TransactionBuilderFactory
271
- *
272
- * @param rawTransaction Transaction as hex string
236
+ * Validates raw transaction format
273
237
  */
274
238
  validateRawTransaction(rawTransaction) {
275
239
  if (!rawTransaction) {
276
240
  throw new sdk_core_1.InvalidTransactionError('Raw transaction is empty');
277
241
  }
278
- if (!utils.allHexChars(rawTransaction)) {
242
+ if (!this.allHexChars(rawTransaction)) {
279
243
  throw new sdk_core_1.ParseTransactionError('Raw transaction is not hex string');
280
244
  }
281
245
  }
282
246
  /**
283
- * Check if tx is for the blockchainId
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
247
+ * Checks if output is TransferableOutput type
316
248
  */
317
249
  isTransferableOutput(output) {
318
- return typeof output.getOutput === 'function';
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
- };
250
+ return output?._type === flarejs_1.TypeSymbols.TransferableOutput;
351
251
  }
352
252
  /**
353
- * Return a mapper function to that network address representation.
354
- * @param network required to stringify addresses
355
- * @return mapper function
253
+ * Maps outputs to entry format
356
254
  */
357
255
  mapOutputToEntry(network) {
358
256
  return (output) => {
359
257
  if (this.isTransferableOutput(output)) {
360
- const transferableOutput = output;
361
- const outputAmount = transferableOutput.amount();
362
- // Simplified address handling for FlareJS
363
- const address = 'flare-address-placeholder'; // TODO: implement proper address conversion
258
+ const outputAmount = output.amount();
259
+ const address = output.output
260
+ .getOwners()
261
+ .map((a) => this.addressToString(network.hrp, network.alias, buffer_1.Buffer.from(a)))
262
+ .sort()
263
+ .join(iface_1.ADDRESS_SEPARATOR);
364
264
  return {
365
265
  value: outputAmount.toString(),
366
266
  address,
@@ -372,116 +272,119 @@ class Utils {
372
272
  };
373
273
  }
374
274
  /**
375
- * remove hex prefix (0x)
376
- * @param hex string
377
- * @returns hex without 0x
275
+ * Removes 0x prefix from hex string
378
276
  */
379
277
  removeHexPrefix(hex) {
380
- if (hex.startsWith('0x')) {
381
- return hex.substring(2);
382
- }
383
- return hex;
278
+ return hex.startsWith('0x') ? hex.substring(2) : hex;
384
279
  }
385
280
  /**
386
- * Outputidx convert from number (as string) to buffer.
387
- * @param {string} outputidx number
388
- * @return {Buffer} buffer of size 4 with that number value
281
+ * Converts output index to buffer
389
282
  */
390
283
  outputidxNumberToBuffer(outputidx) {
391
- return Buffer.from(Number(outputidx).toString(16).padStart(constants_1.OUTPUT_INDEX_HEX_LENGTH, '0'), 'hex');
284
+ return buffer_1.Buffer.from(Number(outputidx).toString(16).padStart(8, '0'), 'hex');
392
285
  }
393
286
  /**
394
- * Outputidx buffer to number (as string)
395
- * @param {Buffer} outputidx
396
- * @return {string} outputidx number
287
+ * Converts output index buffer to number string
397
288
  */
398
289
  outputidxBufferToNumber(outputidx) {
399
290
  return parseInt(outputidx.toString('hex'), 16).toString();
400
291
  }
292
+ // Required by BaseUtils interface but not implemented
293
+ isValidSignature(signature) {
294
+ throw new sdk_core_1.NotImplementedError('isValidSignature not implemented');
295
+ }
296
+ isValidTransactionId(txId) {
297
+ throw new sdk_core_1.NotImplementedError('isValidTransactionId not implemented');
298
+ }
401
299
  /**
402
- * CB58 decode function - simple Base58 decode implementation
403
- * @param {string} data - CB58 encoded string
404
- * @returns {Buffer} decoded buffer
300
+ * Decodes a base58 string with checksum to a Buffer
405
301
  */
406
- cb58Decode(data) {
407
- // For now, use a simple hex decode as placeholder
408
- // In a full implementation, this would be proper CB58 decoding
409
- try {
410
- return Buffer.from(data, 'hex');
411
- }
412
- catch {
413
- // Fallback to buffer from string
414
- return Buffer.from(data);
302
+ cb58Decode(str) {
303
+ const decoded = bs58_1.default.decode(str);
304
+ if (!this.validateChecksum(buffer_1.Buffer.from(decoded))) {
305
+ throw new Error('Invalid checksum');
415
306
  }
307
+ return buffer_1.Buffer.from(decoded.slice(0, decoded.length - 4));
416
308
  }
417
309
  /**
418
- * Convert address buffer to bech32 string
419
- * @param {string} hrp - Human readable part
420
- * @param {string} chainid - Chain identifier
421
- * @param {Buffer} addressBuffer - Address buffer
422
- * @returns {string} Address string
310
+ * Validates a checksum on a Buffer and returns true if valid, false if not
423
311
  */
424
- addressToString(hrp, chainid, addressBuffer) {
425
- // Simple implementation - in practice this would use bech32 encoding
426
- return `${chainid}-${addressBuffer.toString('hex')}`;
312
+ validateChecksum(buff) {
313
+ const hashSlice = buff.slice(buff.length - 4);
314
+ const calculatedHashSlice = (0, crypto_1.createHash)('sha256')
315
+ .update(buff.slice(0, buff.length - 4))
316
+ .digest()
317
+ .slice(28);
318
+ return hashSlice.toString('hex') === calculatedHashSlice.toString('hex');
427
319
  }
428
320
  /**
429
- * Convert string to bytes for FlareJS memo
430
- * Follows FlareJS utils.stringToBytes pattern
431
- * @param {string} text - Text to convert
432
- * @returns {Uint8Array} Byte array
321
+ * Encodes a Buffer as a base58 string with checksum
433
322
  */
434
- stringToBytes(text) {
435
- return new TextEncoder().encode(text);
323
+ cb58Encode(bytes) {
324
+ const withChecksum = this.addChecksum(bytes);
325
+ return bs58_1.default.encode(withChecksum);
436
326
  }
437
327
  /**
438
- * Convert bytes to string from FlareJS memo
439
- * @param {Uint8Array} bytes - Bytes to convert
440
- * @returns {string} Decoded string
328
+ * Adds a checksum to a Buffer and returns the concatenated result
441
329
  */
442
- bytesToString(bytes) {
443
- return new TextDecoder().decode(bytes);
330
+ addChecksum(buff) {
331
+ const hashSlice = (0, crypto_1.createHash)('sha256').update(buff).digest().slice(28);
332
+ return buffer_1.Buffer.concat([buff, hashSlice]);
444
333
  }
445
334
  /**
446
- * Create memo bytes from various input formats
447
- * Supports string, JSON object, or raw bytes
448
- * @param {string | Record<string, unknown> | Uint8Array} memo - Memo data
449
- * @returns {Uint8Array} Memo bytes for FlareJS
335
+ * Check if tx is for the blockchainId
336
+ *
337
+ * @param {DeprecatedTx} tx
338
+ * @param {string} blockchainId
339
+ * @returns true if tx is for blockchainId
450
340
  */
451
- createMemoBytes(memo) {
452
- if (memo instanceof Uint8Array) {
453
- return memo;
454
- }
455
- if (typeof memo === 'string') {
456
- return this.stringToBytes(memo);
341
+ // TODO: remove DeprecatedTx usage
342
+ isTransactionOf(tx, blockchainId) {
343
+ // FlareJS equivalent - this would need proper CB58 encoding implementation
344
+ try {
345
+ const txRecord = tx;
346
+ const unsignedTx = txRecord.getUnsignedTx();
347
+ const transaction = unsignedTx.getTransaction();
348
+ const txBlockchainId = transaction.getBlockchainID();
349
+ return buffer_1.Buffer.from(txBlockchainId).toString('hex') === blockchainId;
457
350
  }
458
- if (typeof memo === 'object') {
459
- return this.stringToBytes(JSON.stringify(memo));
351
+ catch (error) {
352
+ return false;
460
353
  }
461
- throw new sdk_core_1.InvalidTransactionError('Invalid memo format');
462
354
  }
463
- /**
464
- * Parse memo bytes to string
465
- * @param {Uint8Array} memoBytes - Memo bytes from FlareJS transaction
466
- * @returns {string} Decoded memo string
467
- */
468
- parseMemoBytes(memoBytes) {
469
- if (memoBytes.length === 0) {
470
- return '';
471
- }
472
- return this.bytesToString(memoBytes);
355
+ flareIdString(value) {
356
+ return new flarejs_1.Id(buffer_1.Buffer.from(value, 'hex'));
473
357
  }
474
358
  /**
475
- * Validate memo size (FlareJS has transaction size limits)
476
- * @param {Uint8Array} memoBytes - Memo bytes
477
- * @param {number} maxSize - Maximum size in bytes (default 4KB)
478
- * @returns {boolean} Whether memo is within size limits
359
+ * FlareJS wrapper to recover signature
360
+ * @param network
361
+ * @param message
362
+ * @param signature
363
+ * @return recovered public key
479
364
  */
480
- validateMemoSize(memoBytes, maxSize = 4096) {
481
- return memoBytes.length <= maxSize;
365
+ recoverySignature(network, message, signature) {
366
+ try {
367
+ // Hash the message first - must match the hash used in signing
368
+ const messageHash = (0, crypto_1.createHash)('sha256').update(message).digest();
369
+ // Extract recovery parameter and signature
370
+ if (signature.length !== 65) {
371
+ throw new Error('Invalid signature length - expected 65 bytes (64 bytes signature + 1 byte recovery)');
372
+ }
373
+ const recoveryParam = signature[64];
374
+ const sigOnly = signature.slice(0, 64);
375
+ // Recover public key using the provided recovery parameter
376
+ const recovered = secp256k1_1.ecc.recoverPublicKey(messageHash, sigOnly, recoveryParam, true);
377
+ if (!recovered) {
378
+ throw new Error('Failed to recover public key');
379
+ }
380
+ return buffer_1.Buffer.from(recovered);
381
+ }
382
+ catch (error) {
383
+ throw new Error(`Failed to recover signature: ${error}`);
384
+ }
482
385
  }
483
386
  }
484
387
  exports.Utils = Utils;
485
388
  const utils = new Utils();
486
389
  exports.default = utils;
487
- //# 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 {\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"]}
390
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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","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   * 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   */\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   * 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  // 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  /**\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\nconst utils = new Utils();\nexport default utils;\n"]}