@bitgo-beta/sdk-coin-flrp 1.0.0-alpha.72 → 1.0.0-alpha.74
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/lib/ExportInPTxBuilder.d.ts +19 -2
- package/dist/src/lib/ExportInPTxBuilder.d.ts.map +1 -1
- package/dist/src/lib/ExportInPTxBuilder.js +88 -22
- package/dist/src/lib/ImportInCTxBuilder.d.ts +5 -1
- package/dist/src/lib/ImportInCTxBuilder.d.ts.map +1 -1
- package/dist/src/lib/ImportInCTxBuilder.js +51 -41
- package/dist/src/lib/ImportInPTxBuilder.d.ts +5 -1
- package/dist/src/lib/ImportInPTxBuilder.d.ts.map +1 -1
- package/dist/src/lib/ImportInPTxBuilder.js +49 -25
- package/dist/src/lib/atomicTransactionBuilder.d.ts +39 -9
- package/dist/src/lib/atomicTransactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/atomicTransactionBuilder.js +142 -79
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +13 -4
- package/dist/src/lib/utils.d.ts +25 -0
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +42 -3
- package/dist/test/resources/account.d.ts +21 -0
- package/dist/test/resources/account.d.ts.map +1 -1
- package/dist/test/resources/account.js +23 -2
- package/dist/test/resources/transactionData/exportInP.d.ts.map +1 -1
- package/dist/test/resources/transactionData/exportInP.js +5 -6
- package/dist/test/resources/transactionData/importInC.js +5 -5
- package/dist/test/resources/transactionData/importInP.js +5 -5
- package/dist/test/unit/flrp.js +3 -3
- package/dist/test/unit/lib/exportInPTxBuilder.js +144 -34
- package/dist/test/unit/lib/importInCTxBuilder.js +51 -74
- package/dist/test/unit/lib/importInPTxBuilder.js +48 -183
- package/dist/test/unit/lib/utils.js +82 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +8 -8
|
@@ -4,6 +4,7 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
|
4
4
|
};
|
|
5
5
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
6
|
exports.AtomicTransactionBuilder = void 0;
|
|
7
|
+
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
7
8
|
const transactionBuilder_1 = require("./transactionBuilder");
|
|
8
9
|
const transaction_1 = require("./transaction");
|
|
9
10
|
const flarejs_1 = require("@flarenetwork/flarejs");
|
|
@@ -85,113 +86,175 @@ class AtomicTransactionBuilder extends transactionBuilder_1.TransactionBuilder {
|
|
|
85
86
|
this.transaction._amount = valueBigInt;
|
|
86
87
|
return this;
|
|
87
88
|
}
|
|
89
|
+
/*
|
|
90
|
+
* Key naming:
|
|
91
|
+
* A = user key, B = hsm key (BitGo), C = backup key
|
|
92
|
+
*
|
|
93
|
+
* _fromAddresses (BitGo convention) = [ A, B, C ] at indices [0, 1, 2]
|
|
94
|
+
*
|
|
95
|
+
* Signing key selection (which keys from _fromAddresses to use):
|
|
96
|
+
* - non-recovery: _fromAddresses[0] + _fromAddresses[1] (user + bitgo)
|
|
97
|
+
* - recovery: _fromAddresses[1] + _fromAddresses[2] (bitgo + backup)
|
|
98
|
+
*
|
|
99
|
+
* sigIndices in transaction (positions in sorted UTXO address list):
|
|
100
|
+
* - UTXO addresses are sorted by hex value on-chain
|
|
101
|
+
* - sigIndices = positions of the 2 signing keys in this sorted list
|
|
102
|
+
* - Example: if user sorts to position 2 and bitgo to position 0,
|
|
103
|
+
* then sigIndices = [0, 2] (even though we picked _fromAddresses[0, 1])
|
|
104
|
+
*/
|
|
105
|
+
/**
|
|
106
|
+
* Get the 2 signing addresses for FlareJS transaction building.
|
|
107
|
+
*
|
|
108
|
+
* FlareJS's matchOwners() selects signers in sorted UTXO address order,
|
|
109
|
+
* not based on which keys should actually sign. By passing only 2 addresses,
|
|
110
|
+
* we ensure the correct signers are selected.
|
|
111
|
+
*
|
|
112
|
+
* This mirrors AVAXP's approach in createInputOutput() where:
|
|
113
|
+
* - For non-recovery: use user (index 0) and bitgo (index 1)
|
|
114
|
+
* - For recovery: use bitgo (index 1) and recovery (index 2)
|
|
115
|
+
*
|
|
116
|
+
* @returns Array of 2 signing address buffers
|
|
117
|
+
* @protected
|
|
118
|
+
*/
|
|
119
|
+
getSigningAddresses() {
|
|
120
|
+
const firstIndex = this.recoverSigner ? 2 : 0;
|
|
121
|
+
const bitgoIndex = 1;
|
|
122
|
+
if (this.transaction._fromAddresses.length < Math.max(firstIndex, bitgoIndex) + 1) {
|
|
123
|
+
throw new sdk_core_1.BuildTransactionError(`Insufficient fromAddresses: need at least ${Math.max(firstIndex, bitgoIndex) + 1} addresses`);
|
|
124
|
+
}
|
|
125
|
+
const signingAddresses = [
|
|
126
|
+
Buffer.from(this.transaction._fromAddresses[firstIndex]),
|
|
127
|
+
Buffer.from(this.transaction._fromAddresses[bitgoIndex]),
|
|
128
|
+
];
|
|
129
|
+
const invalidAddr = signingAddresses.find((addr) => addr.length !== 20);
|
|
130
|
+
if (invalidAddr) {
|
|
131
|
+
throw new sdk_core_1.BuildTransactionError(`Invalid signing address length: expected 20 bytes, got ${invalidAddr.length}`);
|
|
132
|
+
}
|
|
133
|
+
return signingAddresses;
|
|
134
|
+
}
|
|
88
135
|
/**
|
|
89
|
-
*
|
|
90
|
-
*
|
|
136
|
+
* Compute addressesIndex for UTXOs.
|
|
137
|
+
* addressesIndex[senderIdx] = position of sender[senderIdx] in UTXO's sorted address list.
|
|
138
|
+
*
|
|
139
|
+
* UTXO addresses are sorted lexicographically by byte value to match on-chain storage order.
|
|
140
|
+
* @param forceRecompute - If true, recompute even if addressesIndex already exists
|
|
141
|
+
* @protected
|
|
142
|
+
*/
|
|
143
|
+
computeAddressesIndex(forceRecompute = false) {
|
|
144
|
+
const sender = this.transaction._fromAddresses;
|
|
145
|
+
if (!sender || sender.length === 0)
|
|
146
|
+
return;
|
|
147
|
+
this.transaction._utxos.forEach((utxo) => {
|
|
148
|
+
if (!forceRecompute && utxo.addressesIndex && utxo.addressesIndex.length > 0) {
|
|
149
|
+
return;
|
|
150
|
+
}
|
|
151
|
+
if (utxo.addresses && utxo.addresses.length > 0) {
|
|
152
|
+
const sortedAddresses = utils_1.default.sortAddressesByHex(utxo.addresses);
|
|
153
|
+
utxo.addresses = sortedAddresses;
|
|
154
|
+
const utxoAddresses = sortedAddresses.map((a) => utils_1.default.parseAddress(a));
|
|
155
|
+
utxo.addressesIndex = sender.map((a) => utxoAddresses.findIndex((u) => Buffer.compare(Buffer.from(u), Buffer.from(a)) === 0));
|
|
156
|
+
}
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
/**
|
|
160
|
+
* Validate UTXOs have consistent addresses.
|
|
161
|
+
* @protected
|
|
162
|
+
*/
|
|
163
|
+
validateUtxoAddresses() {
|
|
164
|
+
this.transaction._utxos.forEach((utxo) => {
|
|
165
|
+
if (!utxo) {
|
|
166
|
+
throw new sdk_core_1.BuildTransactionError('Utxo is undefined');
|
|
167
|
+
}
|
|
168
|
+
if (utxo.addressesIndex?.includes(-1)) {
|
|
169
|
+
throw new sdk_core_1.BuildTransactionError('Addresses are inconsistent: ' + utxo.txid);
|
|
170
|
+
}
|
|
171
|
+
});
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* Create credential for a UTXO following AVAX P approach.
|
|
175
|
+
* Embed user/recovery address, leave BitGo slot empty.
|
|
176
|
+
* Signing order is guaranteed: user signs first (address match), BitGo signs second (empty slot).
|
|
177
|
+
*
|
|
91
178
|
* @param utxo - The UTXO to create credential for
|
|
92
179
|
* @param threshold - Number of signatures required
|
|
93
|
-
* @
|
|
180
|
+
* @param sigIndices - Optional sigIndices from FlareJS (if not provided, derived from addressesIndex)
|
|
94
181
|
* @protected
|
|
95
182
|
*/
|
|
96
|
-
createCredentialForUtxo(utxo, threshold) {
|
|
183
|
+
createCredentialForUtxo(utxo, threshold, sigIndices) {
|
|
97
184
|
const sender = this.transaction._fromAddresses;
|
|
98
|
-
const
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
if (
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
for (let i = 0; i < threshold; i++) {
|
|
105
|
-
emptySignatures.push(utils_1.default.createNewSig(''));
|
|
185
|
+
const addressesIndex = utxo.addressesIndex ?? [];
|
|
186
|
+
const firstIndex = this.recoverSigner ? 2 : 0;
|
|
187
|
+
const bitgoIndex = 1;
|
|
188
|
+
if (threshold === 1) {
|
|
189
|
+
if (sender && sender.length > firstIndex) {
|
|
190
|
+
return new flarejs_1.Credential([utils_1.default.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex'))]);
|
|
106
191
|
}
|
|
107
|
-
return new flarejs_1.Credential(
|
|
192
|
+
return new flarejs_1.Credential([utils_1.default.createNewSig('')]);
|
|
108
193
|
}
|
|
109
|
-
if (
|
|
110
|
-
|
|
194
|
+
if (addressesIndex.length >= 2 && sender && sender.length >= threshold) {
|
|
195
|
+
const effectiveSigIndices = sigIndices && sigIndices.length >= 2
|
|
196
|
+
? sigIndices
|
|
197
|
+
: [addressesIndex[firstIndex], addressesIndex[bitgoIndex]].sort((a, b) => a - b);
|
|
111
198
|
const emptySignatures = [];
|
|
112
|
-
for (
|
|
113
|
-
|
|
199
|
+
for (const sigIdx of effectiveSigIndices) {
|
|
200
|
+
const senderIdx = addressesIndex.findIndex((utxoPos) => utxoPos === sigIdx);
|
|
201
|
+
if (senderIdx === firstIndex) {
|
|
202
|
+
emptySignatures.push(utils_1.default.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')));
|
|
203
|
+
}
|
|
204
|
+
else {
|
|
205
|
+
emptySignatures.push(utils_1.default.createNewSig(''));
|
|
206
|
+
}
|
|
114
207
|
}
|
|
115
208
|
return new flarejs_1.Credential(emptySignatures);
|
|
116
209
|
}
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
// either user (0) or recovery (2)
|
|
121
|
-
const firstIndex = this.recoverSigner ? 2 : 0;
|
|
122
|
-
const bitgoIndex = 1;
|
|
123
|
-
// Dynamic ordering based on addressesIndex
|
|
124
|
-
let emptySignatures;
|
|
125
|
-
if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {
|
|
126
|
-
// Bitgo comes first in signature order: [zeros, userAddress]
|
|
127
|
-
emptySignatures = [
|
|
128
|
-
utils_1.default.createNewSig(''),
|
|
129
|
-
utils_1.default.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')),
|
|
130
|
-
];
|
|
131
|
-
}
|
|
132
|
-
else {
|
|
133
|
-
// User comes first in signature order: [userAddress, zeros]
|
|
134
|
-
emptySignatures = [
|
|
135
|
-
utils_1.default.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')),
|
|
136
|
-
utils_1.default.createNewSig(''),
|
|
137
|
-
];
|
|
210
|
+
const emptySignatures = [];
|
|
211
|
+
for (let i = 0; i < threshold; i++) {
|
|
212
|
+
emptySignatures.push(utils_1.default.createNewSig(''));
|
|
138
213
|
}
|
|
139
214
|
return new flarejs_1.Credential(emptySignatures);
|
|
140
215
|
}
|
|
141
216
|
/**
|
|
142
|
-
* Create AddressMap
|
|
143
|
-
*
|
|
144
|
-
* AddressMaps should map addresses to signature slots in the same order as credentials
|
|
217
|
+
* Create AddressMap for a UTXO following AVAX P approach.
|
|
218
|
+
*
|
|
145
219
|
* @param utxo - The UTXO to create AddressMap for
|
|
146
220
|
* @param threshold - Number of signatures required
|
|
147
|
-
* @
|
|
221
|
+
* @param sigIndices - Optional sigIndices from FlareJS (if not provided, derived from addressesIndex)
|
|
148
222
|
* @protected
|
|
149
223
|
*/
|
|
150
|
-
createAddressMapForUtxo(utxo, threshold) {
|
|
224
|
+
createAddressMapForUtxo(utxo, threshold, sigIndices) {
|
|
151
225
|
const addressMap = new flarejs_1.utils.AddressMap();
|
|
152
226
|
const sender = this.transaction._fromAddresses;
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
addressMap.set(new flarejs_1.Address(sender[utxoAddrIdx]), slotIdx);
|
|
163
|
-
}
|
|
227
|
+
const addressesIndex = utxo.addressesIndex ?? [];
|
|
228
|
+
const firstIndex = this.recoverSigner ? 2 : 0;
|
|
229
|
+
const bitgoIndex = 1;
|
|
230
|
+
if (threshold === 1) {
|
|
231
|
+
if (sender && sender.length > firstIndex) {
|
|
232
|
+
addressMap.set(new flarejs_1.Address(sender[firstIndex]), 0);
|
|
233
|
+
}
|
|
234
|
+
else if (sender && sender.length > 0) {
|
|
235
|
+
addressMap.set(new flarejs_1.Address(sender[0]), 0);
|
|
164
236
|
}
|
|
165
237
|
return addressMap;
|
|
166
238
|
}
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
}
|
|
179
|
-
else {
|
|
180
|
-
// User/recovery comes first: slot 0 = firstIndex, slot 1 = bitgo
|
|
181
|
-
addressMap.set(new flarejs_1.Address(sender[firstIndex]), 0);
|
|
182
|
-
addressMap.set(new flarejs_1.Address(sender[bitgoIndex]), 1);
|
|
183
|
-
}
|
|
239
|
+
if (addressesIndex.length >= 2 && sender && sender.length >= threshold) {
|
|
240
|
+
const effectiveSigIndices = sigIndices && sigIndices.length >= 2
|
|
241
|
+
? sigIndices
|
|
242
|
+
: [addressesIndex[firstIndex], addressesIndex[bitgoIndex]].sort((a, b) => a - b);
|
|
243
|
+
effectiveSigIndices.forEach((sigIdx, slotIdx) => {
|
|
244
|
+
const senderIdx = addressesIndex.findIndex((utxoPos) => utxoPos === sigIdx);
|
|
245
|
+
if (senderIdx === bitgoIndex || senderIdx === firstIndex) {
|
|
246
|
+
addressMap.set(new flarejs_1.Address(sender[senderIdx]), slotIdx);
|
|
247
|
+
}
|
|
248
|
+
});
|
|
249
|
+
return addressMap;
|
|
184
250
|
}
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
addressMap.set(new flarejs_1.Address(addr), i);
|
|
190
|
-
});
|
|
191
|
-
}
|
|
251
|
+
if (sender && sender.length >= threshold) {
|
|
252
|
+
sender.slice(0, threshold).forEach((addr, i) => {
|
|
253
|
+
addressMap.set(new flarejs_1.Address(addr), i);
|
|
254
|
+
});
|
|
192
255
|
}
|
|
193
256
|
return addressMap;
|
|
194
257
|
}
|
|
195
258
|
}
|
|
196
259
|
exports.AtomicTransactionBuilder = AtomicTransactionBuilder;
|
|
197
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"atomicTransactionBuilder.js","sourceRoot":"","sources":["../../../src/lib/atomicTransactionBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { TransactionType } from '@bitgo-beta/sdk-core';\nimport { TransactionBuilder } from './transactionBuilder';\nimport { Transaction } from './transaction';\nimport { Credential, Address, utils as FlareUtils } from '@flarenetwork/flarejs';\nimport { DecodedUtxoObj } from './iface';\nimport { FlrpFeeState } from '@bitgo/public-types';\nimport utils from './utils';\n\nexport abstract class AtomicTransactionBuilder extends TransactionBuilder {\n  protected _externalChainId: Buffer;\n  protected recoverSigner = false;\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this.transaction = new Transaction(_coinConfig);\n    this.transaction._fee.fee = this.fixedFee;\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction> {\n    await this.buildFlareTransaction();\n    this.setTransactionType(this.transactionType);\n    if (this.hasSigner()) {\n      for (const keyPair of this._signer) {\n        await this.transaction.sign(keyPair);\n      }\n    }\n    return this.transaction;\n  }\n\n  /**\n   * Builds the Flare transaction. Transaction field is changed.\n   */\n  protected abstract buildFlareTransaction(): void | Promise<void>;\n\n  protected abstract get transactionType(): TransactionType;\n\n  /**\n   * Fee is fix for AVM atomic tx.\n   *\n   * @returns network.txFee\n   * @protected\n   */\n  protected get fixedFee(): string {\n    return this.transaction._network.txFee;\n  }\n\n  /**\n   * Set the transaction type\n   *\n   * @param {TransactionType} transactionType The transaction type to be set\n   */\n  setTransactionType(transactionType: TransactionType): void {\n    this.transaction._type = transactionType;\n  }\n\n  /**\n   * The internal chain is the one set for the coin in coinConfig.network. The external chain is the other chain involved.\n   * The external chain id is the source on import and the destination on export.\n   *\n   * @param {string} chainId - id of the external chain\n   */\n  externalChainId(chainId: string | Buffer): this {\n    const newTargetChainId = typeof chainId === 'string' ? utils.cb58Decode(chainId) : Buffer.from(chainId);\n    this.validateChainId(newTargetChainId);\n    this._externalChainId = newTargetChainId;\n    return this;\n  }\n\n  /**\n   * Set the transaction fee\n   *\n   * @param {string | bigint} feeValue - the fee value\n   */\n  fee(feeValue: string | bigint): this {\n    const fee = typeof feeValue === 'string' ? feeValue : feeValue.toString();\n    (this.transaction as Transaction)._fee.fee = fee;\n    return this;\n  }\n\n  /**\n   * Set the fee state for dynamic fee calculation (P-chain transactions)\n   *\n   * @param {FlrpFeeState} state - the fee state from the network\n   */\n  feeState(state: FlrpFeeState): this {\n    this.transaction._feeState = state;\n    return this;\n  }\n\n  /**\n   * Set the amount for the transaction\n   *\n   * @param {bigint | string} value - the amount to transfer\n   */\n  amount(value: bigint | string): this {\n    const valueBigInt = typeof value === 'string' ? BigInt(value) : value;\n    this.validateAmount(valueBigInt);\n    this.transaction._amount = valueBigInt;\n    return this;\n  }\n\n  /**\n   * Create credential with dynamic ordering based on addressesIndex from UTXO\n   * Matches avaxp behavior: signature order depends on UTXO address positions\n   * @param utxo - The UTXO to create credential for\n   * @param threshold - Number of signatures required\n   * @returns Credential with empty signatures ordered based on UTXO positions\n   * @protected\n   */\n  protected createCredentialForUtxo(utxo: DecodedUtxoObj, threshold: number): Credential {\n    const sender = (this.transaction as Transaction)._fromAddresses;\n    const hasAddresses = sender && sender.length >= threshold;\n\n    // If we have pre-computed addressesIndex (from parsing a transaction), use it directly\n    // This is the authoritative source for signature ordering from parsed transactions\n    if (utxo.addressesIndex && utxo.addressesIndex.length >= threshold) {\n      // Create credentials matching the sigIndicies order from the parsed transaction\n      const emptySignatures: ReturnType<typeof utils.createNewSig>[] = [];\n      for (let i = 0; i < threshold; i++) {\n        emptySignatures.push(utils.createNewSig(''));\n      }\n      return new Credential(emptySignatures);\n    }\n\n    if (!hasAddresses || !utxo.addresses || utxo.addresses.length === 0) {\n      // Fallback: use all zeros if no addresses available\n      const emptySignatures: ReturnType<typeof utils.createNewSig>[] = [];\n      for (let i = 0; i < threshold; i++) {\n        emptySignatures.push(utils.createNewSig(''));\n      }\n      return new Credential(emptySignatures);\n    }\n\n    // Compute addressesIndex: position of each _fromAddresses in UTXO's address list\n    const utxoAddresses = utxo.addresses.map((a) => utils.parseAddress(a));\n    const addressesIndex = sender.map((a) =>\n      utxoAddresses.findIndex((u) => Buffer.compare(Buffer.from(u), Buffer.from(a)) === 0)\n    );\n\n    // either user (0) or recovery (2)\n    const firstIndex = this.recoverSigner ? 2 : 0;\n    const bitgoIndex = 1;\n\n    // Dynamic ordering based on addressesIndex\n    let emptySignatures: ReturnType<typeof utils.createNewSig>[];\n    if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {\n      // Bitgo comes first in signature order: [zeros, userAddress]\n      emptySignatures = [\n        utils.createNewSig(''),\n        utils.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')),\n      ];\n    } else {\n      // User comes first in signature order: [userAddress, zeros]\n      emptySignatures = [\n        utils.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')),\n        utils.createNewSig(''),\n      ];\n    }\n    return new Credential(emptySignatures);\n  }\n\n  /**\n   * Create AddressMap based on signature slot order (matching credential order), not sorted addresses\n   * This matches the approach used in credentials: addressesIndex determines signature order\n   * AddressMaps should map addresses to signature slots in the same order as credentials\n   * @param utxo - The UTXO to create AddressMap for\n   * @param threshold - Number of signatures required\n   * @returns AddressMap that maps addresses to signature slots based on UTXO order\n   * @protected\n   */\n  protected createAddressMapForUtxo(utxo: DecodedUtxoObj, threshold: number): FlareUtils.AddressMap {\n    const addressMap = new FlareUtils.AddressMap();\n    const sender = (this.transaction as Transaction)._fromAddresses;\n\n    // If we have pre-computed addressesIndex (from parsing a transaction), use it directly\n    // addressesIndex from sigIndicies() tells us: addressesIndex[slotIdx] = utxoAddressIdx\n    // This means slot 'slotIdx' expects signature from UTXO address at index 'utxoAddressIdx'\n    // Assuming sender[i] corresponds to utxoAddress[i], we map sender[addressesIndex[slotIdx]] to slotIdx\n    if (utxo.addressesIndex && utxo.addressesIndex.length >= threshold && sender && sender.length >= threshold) {\n      for (let slotIdx = 0; slotIdx < threshold; slotIdx++) {\n        const utxoAddrIdx = utxo.addressesIndex[slotIdx];\n        // Map the sender that corresponds to this UTXO address index to this slot\n        if (utxoAddrIdx < sender.length) {\n          addressMap.set(new Address(sender[utxoAddrIdx]), slotIdx);\n        }\n      }\n      return addressMap;\n    }\n\n    // If UTXO has addresses, compute addressesIndex to determine signature order\n    if (utxo && utxo.addresses && utxo.addresses.length > 0 && sender && sender.length >= threshold) {\n      const utxoAddresses = utxo.addresses.map((a) => utils.parseAddress(a));\n      const addressesIndex = sender.map((a) =>\n        utxoAddresses.findIndex((u) => Buffer.compare(Buffer.from(u), Buffer.from(a)) === 0)\n      );\n\n      const firstIndex = this.recoverSigner ? 2 : 0;\n      const bitgoIndex = 1;\n\n      // Determine signature slot order based on addressesIndex (same logic as credentials)\n      if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {\n        // Bitgo comes first: slot 0 = bitgo, slot 1 = firstIndex\n        addressMap.set(new Address(sender[bitgoIndex]), 0);\n        addressMap.set(new Address(sender[firstIndex]), 1);\n      } else {\n        // User/recovery comes first: slot 0 = firstIndex, slot 1 = bitgo\n        addressMap.set(new Address(sender[firstIndex]), 0);\n        addressMap.set(new Address(sender[bitgoIndex]), 1);\n      }\n    } else {\n      // Fallback: map addresses sequentially if no UTXO addresses available\n      if (sender && sender.length >= threshold) {\n        sender.slice(0, threshold).forEach((addr, i) => {\n          addressMap.set(new Address(addr), i);\n        });\n      }\n    }\n\n    return addressMap;\n  }\n}\n"]}
|
|
260
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"atomicTransactionBuilder.js","sourceRoot":"","sources":["../../../src/lib/atomicTransactionBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { BuildTransactionError, TransactionType } from '@bitgo-beta/sdk-core';\nimport { TransactionBuilder } from './transactionBuilder';\nimport { Transaction } from './transaction';\nimport { Credential, Address, utils as FlareUtils } from '@flarenetwork/flarejs';\nimport { DecodedUtxoObj } from './iface';\nimport { FlrpFeeState } from '@bitgo/public-types';\nimport utils from './utils';\n\nexport abstract class AtomicTransactionBuilder extends TransactionBuilder {\n  protected _externalChainId: Buffer;\n  protected recoverSigner = false;\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this.transaction = new Transaction(_coinConfig);\n    this.transaction._fee.fee = this.fixedFee;\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction> {\n    await this.buildFlareTransaction();\n    this.setTransactionType(this.transactionType);\n    if (this.hasSigner()) {\n      for (const keyPair of this._signer) {\n        await this.transaction.sign(keyPair);\n      }\n    }\n    return this.transaction;\n  }\n\n  /**\n   * Builds the Flare transaction. Transaction field is changed.\n   */\n  protected abstract buildFlareTransaction(): void | Promise<void>;\n\n  protected abstract get transactionType(): TransactionType;\n\n  /**\n   * Fee is fix for AVM atomic tx.\n   *\n   * @returns network.txFee\n   * @protected\n   */\n  protected get fixedFee(): string {\n    return this.transaction._network.txFee;\n  }\n\n  /**\n   * Set the transaction type\n   *\n   * @param {TransactionType} transactionType The transaction type to be set\n   */\n  setTransactionType(transactionType: TransactionType): void {\n    this.transaction._type = transactionType;\n  }\n\n  /**\n   * The internal chain is the one set for the coin in coinConfig.network. The external chain is the other chain involved.\n   * The external chain id is the source on import and the destination on export.\n   *\n   * @param {string} chainId - id of the external chain\n   */\n  externalChainId(chainId: string | Buffer): this {\n    const newTargetChainId = typeof chainId === 'string' ? utils.cb58Decode(chainId) : Buffer.from(chainId);\n    this.validateChainId(newTargetChainId);\n    this._externalChainId = newTargetChainId;\n    return this;\n  }\n\n  /**\n   * Set the transaction fee\n   *\n   * @param {string | bigint} feeValue - the fee value\n   */\n  fee(feeValue: string | bigint): this {\n    const fee = typeof feeValue === 'string' ? feeValue : feeValue.toString();\n    (this.transaction as Transaction)._fee.fee = fee;\n    return this;\n  }\n\n  /**\n   * Set the fee state for dynamic fee calculation (P-chain transactions)\n   *\n   * @param {FlrpFeeState} state - the fee state from the network\n   */\n  feeState(state: FlrpFeeState): this {\n    this.transaction._feeState = state;\n    return this;\n  }\n\n  /**\n   * Set the amount for the transaction\n   *\n   * @param {bigint | string} value - the amount to transfer\n   */\n  amount(value: bigint | string): this {\n    const valueBigInt = typeof value === 'string' ? BigInt(value) : value;\n    this.validateAmount(valueBigInt);\n    this.transaction._amount = valueBigInt;\n    return this;\n  }\n\n  /*\n   * Key naming:\n   *   A = user key, B = hsm key (BitGo), C = backup key\n   *\n   * _fromAddresses (BitGo convention) = [ A, B, C ] at indices [0, 1, 2]\n   *\n   * Signing key selection (which keys from _fromAddresses to use):\n   *   - non-recovery: _fromAddresses[0] + _fromAddresses[1] (user + bitgo)\n   *   - recovery: _fromAddresses[1] + _fromAddresses[2] (bitgo + backup)\n   *\n   * sigIndices in transaction (positions in sorted UTXO address list):\n   *   - UTXO addresses are sorted by hex value on-chain\n   *   - sigIndices = positions of the 2 signing keys in this sorted list\n   *   - Example: if user sorts to position 2 and bitgo to position 0,\n   *     then sigIndices = [0, 2] (even though we picked _fromAddresses[0, 1])\n   */\n\n  /**\n   * Get the 2 signing addresses for FlareJS transaction building.\n   *\n   * FlareJS's matchOwners() selects signers in sorted UTXO address order,\n   * not based on which keys should actually sign. By passing only 2 addresses,\n   * we ensure the correct signers are selected.\n   *\n   * This mirrors AVAXP's approach in createInputOutput() where:\n   * - For non-recovery: use user (index 0) and bitgo (index 1)\n   * - For recovery: use bitgo (index 1) and recovery (index 2)\n   *\n   * @returns Array of 2 signing address buffers\n   * @protected\n   */\n  protected getSigningAddresses(): Buffer[] {\n    const firstIndex = this.recoverSigner ? 2 : 0;\n    const bitgoIndex = 1;\n\n    if (this.transaction._fromAddresses.length < Math.max(firstIndex, bitgoIndex) + 1) {\n      throw new BuildTransactionError(\n        `Insufficient fromAddresses: need at least ${Math.max(firstIndex, bitgoIndex) + 1} addresses`\n      );\n    }\n\n    const signingAddresses = [\n      Buffer.from(this.transaction._fromAddresses[firstIndex]),\n      Buffer.from(this.transaction._fromAddresses[bitgoIndex]),\n    ];\n\n    const invalidAddr = signingAddresses.find((addr) => addr.length !== 20);\n    if (invalidAddr) {\n      throw new BuildTransactionError(`Invalid signing address length: expected 20 bytes, got ${invalidAddr.length}`);\n    }\n\n    return signingAddresses;\n  }\n\n  /**\n   * Compute addressesIndex for UTXOs.\n   * addressesIndex[senderIdx] = position of sender[senderIdx] in UTXO's sorted address list.\n   *\n   * UTXO addresses are sorted lexicographically by byte value to match on-chain storage order.\n   * @param forceRecompute - If true, recompute even if addressesIndex already exists\n   * @protected\n   */\n  protected computeAddressesIndex(forceRecompute = false): void {\n    const sender = this.transaction._fromAddresses;\n    if (!sender || sender.length === 0) return;\n\n    this.transaction._utxos.forEach((utxo) => {\n      if (!forceRecompute && utxo.addressesIndex && utxo.addressesIndex.length > 0) {\n        return;\n      }\n\n      if (utxo.addresses && utxo.addresses.length > 0) {\n        const sortedAddresses = utils.sortAddressesByHex(utxo.addresses);\n        utxo.addresses = sortedAddresses;\n\n        const utxoAddresses = sortedAddresses.map((a) => utils.parseAddress(a));\n        utxo.addressesIndex = sender.map((a) =>\n          utxoAddresses.findIndex((u) => Buffer.compare(Buffer.from(u), Buffer.from(a)) === 0)\n        );\n      }\n    });\n  }\n\n  /**\n   * Validate UTXOs have consistent addresses.\n   * @protected\n   */\n  protected validateUtxoAddresses(): void {\n    this.transaction._utxos.forEach((utxo) => {\n      if (!utxo) {\n        throw new BuildTransactionError('Utxo is undefined');\n      }\n      if (utxo.addressesIndex?.includes(-1)) {\n        throw new BuildTransactionError('Addresses are inconsistent: ' + utxo.txid);\n      }\n    });\n  }\n\n  /**\n   * Create credential for a UTXO following AVAX P approach.\n   * Embed user/recovery address, leave BitGo slot empty.\n   * Signing order is guaranteed: user signs first (address match), BitGo signs second (empty slot).\n   *\n   * @param utxo - The UTXO to create credential for\n   * @param threshold - Number of signatures required\n   * @param sigIndices - Optional sigIndices from FlareJS (if not provided, derived from addressesIndex)\n   * @protected\n   */\n  protected createCredentialForUtxo(utxo: DecodedUtxoObj, threshold: number, sigIndices?: number[]): Credential {\n    const sender = this.transaction._fromAddresses;\n    const addressesIndex = utxo.addressesIndex ?? [];\n    const firstIndex = this.recoverSigner ? 2 : 0;\n    const bitgoIndex = 1;\n\n    if (threshold === 1) {\n      if (sender && sender.length > firstIndex) {\n        return new Credential([utils.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex'))]);\n      }\n      return new Credential([utils.createNewSig('')]);\n    }\n\n    if (addressesIndex.length >= 2 && sender && sender.length >= threshold) {\n      const effectiveSigIndices =\n        sigIndices && sigIndices.length >= 2\n          ? sigIndices\n          : [addressesIndex[firstIndex], addressesIndex[bitgoIndex]].sort((a, b) => a - b);\n\n      const emptySignatures: ReturnType<typeof utils.createNewSig>[] = [];\n      for (const sigIdx of effectiveSigIndices) {\n        const senderIdx = addressesIndex.findIndex((utxoPos) => utxoPos === sigIdx);\n        if (senderIdx === firstIndex) {\n          emptySignatures.push(utils.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')));\n        } else {\n          emptySignatures.push(utils.createNewSig(''));\n        }\n      }\n      return new Credential(emptySignatures);\n    }\n\n    const emptySignatures: ReturnType<typeof utils.createNewSig>[] = [];\n    for (let i = 0; i < threshold; i++) {\n      emptySignatures.push(utils.createNewSig(''));\n    }\n    return new Credential(emptySignatures);\n  }\n\n  /**\n   * Create AddressMap for a UTXO following AVAX P approach.\n   *\n   * @param utxo - The UTXO to create AddressMap for\n   * @param threshold - Number of signatures required\n   * @param sigIndices - Optional sigIndices from FlareJS (if not provided, derived from addressesIndex)\n   * @protected\n   */\n  protected createAddressMapForUtxo(\n    utxo: DecodedUtxoObj,\n    threshold: number,\n    sigIndices?: number[]\n  ): FlareUtils.AddressMap {\n    const addressMap = new FlareUtils.AddressMap();\n    const sender = this.transaction._fromAddresses;\n    const addressesIndex = utxo.addressesIndex ?? [];\n    const firstIndex = this.recoverSigner ? 2 : 0;\n    const bitgoIndex = 1;\n\n    if (threshold === 1) {\n      if (sender && sender.length > firstIndex) {\n        addressMap.set(new Address(sender[firstIndex]), 0);\n      } else if (sender && sender.length > 0) {\n        addressMap.set(new Address(sender[0]), 0);\n      }\n      return addressMap;\n    }\n\n    if (addressesIndex.length >= 2 && sender && sender.length >= threshold) {\n      const effectiveSigIndices =\n        sigIndices && sigIndices.length >= 2\n          ? sigIndices\n          : [addressesIndex[firstIndex], addressesIndex[bitgoIndex]].sort((a, b) => a - b);\n\n      effectiveSigIndices.forEach((sigIdx, slotIdx) => {\n        const senderIdx = addressesIndex.findIndex((utxoPos) => utxoPos === sigIdx);\n        if (senderIdx === bitgoIndex || senderIdx === firstIndex) {\n          addressMap.set(new Address(sender[senderIdx]), slotIdx);\n        }\n      });\n      return addressMap;\n    }\n\n    if (sender && sender.length >= threshold) {\n      sender.slice(0, threshold).forEach((addr, i) => {\n        addressMap.set(new Address(addr), i);\n      });\n    }\n    return addressMap;\n  }\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"transaction.d.ts","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC3E,OAAO,EACL,OAAO,EACP,eAAe,EACf,KAAK,EAGL,eAAe,EAChB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAEL,UAAU,EAOV,OAAO,EACR,MAAM,uBAAuB,CAAC;AAC/B,OAAO,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAEhC,OAAO,EACL,sBAAsB,EACtB,EAAE,EACF,MAAM,EAGN,kBAAkB,EAClB,cAAc,EACf,MAAM,SAAS,CAAC;AACjB,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AA+DpC,qBAAa,WAAY,SAAQ,eAAe;IAC9C,SAAS,CAAC,iBAAiB,EAAE,EAAE,CAAC;IACzB,KAAK,EAAE,eAAe,CAAC;IACvB,QAAQ,EAAE,YAAY,CAAC;IACvB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,YAAY,EAAE,MAAM,CAAC;IACrB,UAAU,SAAK;IACf,SAAS,SAAa;IACtB,cAAc,EAAE,UAAU,EAAE,CAAM;IAClC,GAAG,EAAE,UAAU,EAAE,CAAM;IACvB,gBAAgB,EAAE,UAAU,EAAE,CAAM;IACpC,MAAM,EAAE,cAAc,EAAE,CAAM;IAC9B,QAAQ,EAAE,OAAO,CAAC,OAAO,CAAC;IAC1B,IAAI,EAAE,kBAAkB,CAAgB;IACxC,SAAS,EAAE,YAAY,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,eAAe,EAAE,MAAM,GAAG,SAAS,CAAC;gBAE/B,UAAU,EAAE,QAAQ,CAAC,UAAU,CAAC;IAS5C,IAAI,SAAS,IAAI,MAAM,EAAE,CAKxB;IAED,IAAI,WAAW,IAAI,UAAU,EAAE,CAE9B;IAED,IAAI,cAAc,IAAI,OAAO,CAE5B;IAED,kBAAkB;IAClB,OAAO,CAAC,EAAE,GAAG,EAAE,EAAE,OAAO,GAAG,OAAO;IAI5B,IAAI,CAAC,OAAO,EAAE,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC;
|
|
1
|
+
{"version":3,"file":"transaction.d.ts","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC3E,OAAO,EACL,OAAO,EACP,eAAe,EACf,KAAK,EAGL,eAAe,EAChB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAEL,UAAU,EAOV,OAAO,EACR,MAAM,uBAAuB,CAAC;AAC/B,OAAO,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAEhC,OAAO,EACL,sBAAsB,EACtB,EAAE,EACF,MAAM,EAGN,kBAAkB,EAClB,cAAc,EACf,MAAM,SAAS,CAAC;AACjB,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AA+DpC,qBAAa,WAAY,SAAQ,eAAe;IAC9C,SAAS,CAAC,iBAAiB,EAAE,EAAE,CAAC;IACzB,KAAK,EAAE,eAAe,CAAC;IACvB,QAAQ,EAAE,YAAY,CAAC;IACvB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,YAAY,EAAE,MAAM,CAAC;IACrB,UAAU,SAAK;IACf,SAAS,SAAa;IACtB,cAAc,EAAE,UAAU,EAAE,CAAM;IAClC,GAAG,EAAE,UAAU,EAAE,CAAM;IACvB,gBAAgB,EAAE,UAAU,EAAE,CAAM;IACpC,MAAM,EAAE,cAAc,EAAE,CAAM;IAC9B,QAAQ,EAAE,OAAO,CAAC,OAAO,CAAC;IAC1B,IAAI,EAAE,kBAAkB,CAAgB;IACxC,SAAS,EAAE,YAAY,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,eAAe,EAAE,MAAM,GAAG,SAAS,CAAC;gBAE/B,UAAU,EAAE,QAAQ,CAAC,UAAU,CAAC;IAS5C,IAAI,SAAS,IAAI,MAAM,EAAE,CAKxB;IAED,IAAI,WAAW,IAAI,UAAU,EAAE,CAE9B;IAED,IAAI,cAAc,IAAI,OAAO,CAE5B;IAED,kBAAkB;IAClB,OAAO,CAAC,EAAE,GAAG,EAAE,EAAE,OAAO,GAAG,OAAO;IAI5B,IAAI,CAAC,OAAO,EAAE,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC;IAgG3C,iBAAiB,IAAI,MAAM;IAkB3B,MAAM,IAAI,MAAM;IAmBhB;;OAEG;IACH,IAAI,WAAW,IAAI,MAAM,GAAG,SAAS,CA6BpC;IAED;;OAEG;IACH,IAAI,gBAAgB,IAAI,MAAM,GAAG,SAAS,CA6BzC;IAED;;;;;OAKG;IACH,mBAAmB,CAAC,kBAAkB,EAAE,MAAM,GAAG,MAAM;IAWvD,cAAc,CAAC,EAAE,EAAE,EAAE,GAAG,IAAI;IAI5B;;;OAGG;IACH,mBAAmB,IAAI,EAAE;IAIzB,kBAAkB,CAAC,eAAe,EAAE,eAAe,GAAG,IAAI;IAO1D,IAAI,eAAe,IAAI,MAAM,CAE5B;IAED,IAAI,EAAE,IAAI,MAAM,CAKf;IAED,IAAI,aAAa,IAAI,MAAM,EAAE,CAE5B;IAED,IAAI,eAAe,IAAI,MAAM,EAAE,CAE9B;IAED,IAAI,GAAG,IAAI,kBAAkB,CAE5B;IAED;;OAEG;IACH,IAAI,sBAAsB,IAAI,OAAO,CAIpC;IAED,IAAI,OAAO,IAAI,KAAK,EAAE,CAwCrB;IAED,IAAI,aAAa,IAAI,KAAK,EAAE,CAmB3B;IAED,IAAI,MAAM,IAAI,KAAK,EAAE,CAsDpB;IAED,kBAAkB,IAAI,sBAAsB;CA0B7C"}
|