@bitgo-beta/sdk-coin-flrp 1.0.1-beta.40 → 1.0.1-beta.401
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/src/flrp.d.ts +10 -17
- package/dist/src/flrp.d.ts.map +1 -1
- package/dist/src/flrp.js +51 -77
- package/dist/src/index.d.ts +0 -1
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +1 -2
- package/dist/src/lib/ExportInCTxBuilder.d.ts +43 -0
- package/dist/src/lib/ExportInCTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ExportInCTxBuilder.js +150 -0
- package/dist/src/lib/ExportInPTxBuilder.d.ts +28 -0
- package/dist/src/lib/ExportInPTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ExportInPTxBuilder.js +190 -0
- package/dist/src/lib/ImportInCTxBuilder.d.ts +34 -0
- package/dist/src/lib/ImportInCTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ImportInCTxBuilder.js +191 -0
- package/dist/src/lib/ImportInPTxBuilder.d.ts +38 -0
- package/dist/src/lib/ImportInPTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/ImportInPTxBuilder.js +224 -0
- package/dist/src/lib/atomicInCTransactionBuilder.d.ts +12 -16
- package/dist/src/lib/atomicInCTransactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/atomicInCTransactionBuilder.js +30 -41
- package/dist/src/lib/atomicTransactionBuilder.d.ts +126 -69
- package/dist/src/lib/atomicTransactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/atomicTransactionBuilder.js +320 -211
- package/dist/src/lib/iface.d.ts +65 -57
- package/dist/src/lib/iface.d.ts.map +1 -1
- package/dist/src/lib/iface.js +20 -14
- package/dist/src/lib/index.d.ts +5 -0
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +12 -2
- package/dist/src/lib/keyPair.d.ts +5 -5
- package/dist/src/lib/keyPair.d.ts.map +1 -1
- package/dist/src/lib/keyPair.js +17 -9
- package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts +41 -0
- package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/permissionlessValidatorTxBuilder.js +126 -0
- package/dist/src/lib/transaction.d.ts +30 -66
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +347 -199
- package/dist/src/lib/transactionBuilder.d.ts +115 -0
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder.js +228 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts +50 -30
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilderFactory.js +129 -72
- package/dist/src/lib/utils.d.ts +131 -146
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +344 -321
- package/dist/test/resources/account.d.ts +81 -0
- package/dist/test/resources/account.d.ts.map +1 -0
- package/dist/test/resources/account.js +79 -0
- package/dist/test/resources/transactionData/exportInC.d.ts +50 -0
- package/dist/test/resources/transactionData/exportInC.d.ts.map +1 -0
- package/dist/test/resources/transactionData/exportInC.js +58 -0
- package/dist/test/resources/transactionData/exportInP.d.ts +60 -0
- package/dist/test/resources/transactionData/exportInP.d.ts.map +1 -0
- package/dist/test/resources/transactionData/exportInP.js +101 -0
- package/dist/test/resources/transactionData/importInC.d.ts +56 -0
- package/dist/test/resources/transactionData/importInC.d.ts.map +1 -0
- package/dist/test/resources/transactionData/importInC.js +120 -0
- package/dist/test/resources/transactionData/importInP.d.ts +66 -0
- package/dist/test/resources/transactionData/importInP.d.ts.map +1 -0
- package/dist/test/resources/transactionData/importInP.js +84 -0
- package/dist/test/unit/flrp.js +449 -68
- package/dist/test/unit/lib/exportInCTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/exportInCTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/exportInCTxBuilder.js +193 -0
- package/dist/test/unit/lib/exportInPTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/exportInPTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/exportInPTxBuilder.js +296 -0
- package/dist/test/unit/lib/importInCTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/importInCTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/importInCTxBuilder.js +309 -0
- package/dist/test/unit/lib/importInPTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/importInPTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/importInPTxBuilder.js +490 -0
- package/dist/test/unit/lib/keyPair.d.ts +2 -0
- package/dist/test/unit/lib/keyPair.d.ts.map +1 -0
- package/dist/test/unit/lib/keyPair.js +158 -0
- package/dist/test/unit/lib/signFlowTestSuit.d.ts +20 -0
- package/dist/test/unit/lib/signFlowTestSuit.d.ts.map +1 -0
- package/dist/test/unit/lib/signFlowTestSuit.js +83 -0
- package/dist/test/unit/lib/signatureIndex.d.ts +13 -0
- package/dist/test/unit/lib/signatureIndex.d.ts.map +1 -0
- package/dist/test/unit/lib/signatureIndex.js +1173 -0
- package/dist/test/unit/lib/transactionBuilderFactory.d.ts +2 -0
- package/dist/test/unit/lib/transactionBuilderFactory.d.ts.map +1 -0
- package/dist/test/unit/lib/transactionBuilderFactory.js +60 -0
- package/dist/test/unit/lib/utils.js +681 -206
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +20 -11
- package/.eslintignore +0 -5
- package/.eslintrc.json +0 -7
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -0
- package/dist/src/iface.d.ts +0 -25
- package/dist/src/iface.d.ts.map +0 -1
- package/dist/src/iface.js +0 -3
- package/dist/src/lib/constants.d.ts +0 -11
- package/dist/src/lib/constants.d.ts.map +0 -1
- package/dist/src/lib/constants.js +0 -17
- package/dist/src/lib/errors.d.ts +0 -8
- package/dist/src/lib/errors.d.ts.map +0 -1
- package/dist/src/lib/errors.js +0 -19
- package/dist/src/lib/exportInCTxBuilder.d.ts +0 -77
- package/dist/src/lib/exportInCTxBuilder.d.ts.map +0 -1
- package/dist/src/lib/exportInCTxBuilder.js +0 -170
- package/dist/src/lib/exportInPTxBuilder.d.ts +0 -30
- package/dist/src/lib/exportInPTxBuilder.d.ts.map +0 -1
- package/dist/src/lib/exportInPTxBuilder.js +0 -56
- package/dist/test/unit/lib/atomicTransactionBuilder.d.ts +0 -2
- package/dist/test/unit/lib/atomicTransactionBuilder.d.ts.map +0 -1
- package/dist/test/unit/lib/atomicTransactionBuilder.js +0 -222
- package/dist/test/unit/lib/exportTxBuilder.d.ts +0 -2
- package/dist/test/unit/lib/exportTxBuilder.d.ts.map +0 -1
- package/dist/test/unit/lib/exportTxBuilder.js +0 -45
- package/dist/test/unit/lib/transaction.d.ts +0 -2
- package/dist/test/unit/lib/transaction.d.ts.map +0 -1
- package/dist/test/unit/lib/transaction.js +0 -460
- package/dist/test/unit/smoke.d.ts +0 -2
- package/dist/test/unit/smoke.d.ts.map +0 -1
- package/dist/test/unit/smoke.js +0 -23
|
@@ -1,252 +1,361 @@
|
|
|
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.AtomicTransactionBuilder = void 0;
|
|
4
7
|
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
8
|
+
const transactionBuilder_1 = require("./transactionBuilder");
|
|
9
|
+
const transaction_1 = require("./transaction");
|
|
5
10
|
const flarejs_1 = require("@flarenetwork/flarejs");
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
class AtomicTransactionBuilder {
|
|
14
|
-
constructor(coinConfig) {
|
|
15
|
-
this._utxos = [];
|
|
16
|
-
this.transaction = {
|
|
17
|
-
_network: {},
|
|
18
|
-
_networkID: 0,
|
|
19
|
-
_blockchainID: Buffer.alloc(0),
|
|
20
|
-
_assetId: Buffer.alloc(0),
|
|
21
|
-
_fromAddresses: [],
|
|
22
|
-
_to: [],
|
|
23
|
-
_locktime: 0n,
|
|
24
|
-
_threshold: 1,
|
|
25
|
-
_fee: { fee: '0' },
|
|
26
|
-
hasCredentials: false,
|
|
27
|
-
setTransaction: function (_tx) {
|
|
28
|
-
this._tx = _tx;
|
|
29
|
-
},
|
|
30
|
-
};
|
|
31
|
-
this._coinConfig = coinConfig;
|
|
11
|
+
const utils_1 = __importDefault(require("./utils"));
|
|
12
|
+
class AtomicTransactionBuilder extends transactionBuilder_1.TransactionBuilder {
|
|
13
|
+
constructor(_coinConfig) {
|
|
14
|
+
super(_coinConfig);
|
|
15
|
+
this.recoverSigner = false;
|
|
16
|
+
this.transaction = new transaction_1.Transaction(_coinConfig);
|
|
17
|
+
this.transaction._fee.fee = this.fixedFee;
|
|
32
18
|
}
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
19
|
+
/** @inheritdoc */
|
|
20
|
+
async buildImplementation() {
|
|
21
|
+
await this.buildFlareTransaction();
|
|
22
|
+
this.setTransactionType(this.transactionType);
|
|
23
|
+
if (this.hasSigner()) {
|
|
24
|
+
for (const keyPair of this._signer) {
|
|
25
|
+
await this.transaction.sign(keyPair);
|
|
26
|
+
}
|
|
36
27
|
}
|
|
28
|
+
return this.transaction;
|
|
37
29
|
}
|
|
38
30
|
/**
|
|
39
|
-
*
|
|
40
|
-
*
|
|
31
|
+
* Fee is fix for AVM atomic tx.
|
|
32
|
+
*
|
|
33
|
+
* @returns network.txFee
|
|
34
|
+
* @protected
|
|
41
35
|
*/
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
throw new sdk_core_1.BuildTransactionError('Credentials must be an array');
|
|
45
|
-
}
|
|
46
|
-
credentials.forEach((credential, index) => {
|
|
47
|
-
if (!(credential instanceof flarejs_1.Credential)) {
|
|
48
|
-
throw new sdk_core_1.BuildTransactionError(`Invalid credential at index ${index}`);
|
|
49
|
-
}
|
|
50
|
-
});
|
|
36
|
+
get fixedFee() {
|
|
37
|
+
return this.transaction._network.txFee;
|
|
51
38
|
}
|
|
52
39
|
/**
|
|
53
|
-
*
|
|
54
|
-
* Based on AVAX P-chain implementation adapted for FlareJS.
|
|
40
|
+
* Set the transaction type
|
|
55
41
|
*
|
|
56
|
-
*
|
|
57
|
-
|
|
42
|
+
* @param {TransactionType} transactionType The transaction type to be set
|
|
43
|
+
*/
|
|
44
|
+
setTransactionType(transactionType) {
|
|
45
|
+
this.transaction._type = transactionType;
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* The internal chain is the one set for the coin in coinConfig.network. The external chain is the other chain involved.
|
|
49
|
+
* The external chain id is the source on import and the destination on export.
|
|
58
50
|
*
|
|
59
|
-
* @param
|
|
60
|
-
* @returns Object containing TransferableInput[], TransferableOutput[], and Credential[]
|
|
51
|
+
* @param {string} chainId - id of the external chain
|
|
61
52
|
*/
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
const outputs = [];
|
|
68
|
-
const credentials = [];
|
|
69
|
-
let inputSum = 0n;
|
|
70
|
-
const addressIndices = {};
|
|
71
|
-
let nextAddressIndex = 0;
|
|
72
|
-
// Sort UTXOs by amount in descending order for optimal coin selection
|
|
73
|
-
const sortedUtxos = [...this._utxos].sort((a, b) => {
|
|
74
|
-
const amountA = BigInt(a.amount);
|
|
75
|
-
const amountB = BigInt(b.amount);
|
|
76
|
-
if (amountA > amountB)
|
|
77
|
-
return -1;
|
|
78
|
-
if (amountA < amountB)
|
|
79
|
-
return 1;
|
|
80
|
-
return 0;
|
|
81
|
-
});
|
|
82
|
-
// Process UTXOs to create inputs and credentials
|
|
83
|
-
for (const utxo of sortedUtxos) {
|
|
84
|
-
const utxoAmount = BigInt(utxo.amount);
|
|
85
|
-
if (inputSum >= total) {
|
|
86
|
-
break; // We have enough inputs
|
|
87
|
-
}
|
|
88
|
-
// TODO: Create proper FlareJS TransferableInput once type issues are resolved
|
|
89
|
-
// For now, we create a placeholder that demonstrates the structure
|
|
90
|
-
// The actual FlareJS integration will need proper UTXOID handling
|
|
91
|
-
// Track input sum
|
|
92
|
-
inputSum += utxoAmount;
|
|
93
|
-
// Track address indices for signature ordering (mimics AVAX pattern)
|
|
94
|
-
const addressIndexArray = [];
|
|
95
|
-
for (const address of utxo.addresses) {
|
|
96
|
-
if (!(address in addressIndices)) {
|
|
97
|
-
addressIndices[address] = nextAddressIndex++;
|
|
98
|
-
}
|
|
99
|
-
addressIndexArray.push(addressIndices[address]);
|
|
100
|
-
}
|
|
101
|
-
// Store address indices on the UTXO for credential creation
|
|
102
|
-
utxo.addressesIndex = addressIndexArray;
|
|
103
|
-
// Create credential with placeholder signatures
|
|
104
|
-
// In a real implementation, these would be actual signatures
|
|
105
|
-
const signatures = Array.from({ length: utxo.threshold }, () => '');
|
|
106
|
-
const credential = this.createFlareCredential(0, signatures);
|
|
107
|
-
credentials.push(credential);
|
|
108
|
-
}
|
|
109
|
-
// Verify we have enough inputs
|
|
110
|
-
if (inputSum < total) {
|
|
111
|
-
throw new sdk_core_1.BuildTransactionError(`Insufficient funds: need ${total}, have ${inputSum}`);
|
|
112
|
-
}
|
|
113
|
-
// TODO: Create change output if we have excess input
|
|
114
|
-
// The TransferableOutput creation will be implemented once FlareJS types are resolved
|
|
115
|
-
return { inputs, outputs, credentials };
|
|
53
|
+
externalChainId(chainId) {
|
|
54
|
+
const newTargetChainId = typeof chainId === 'string' ? utils_1.default.cb58Decode(chainId) : Buffer.from(chainId);
|
|
55
|
+
this.validateChainId(newTargetChainId);
|
|
56
|
+
this._externalChainId = newTargetChainId;
|
|
57
|
+
return this;
|
|
116
58
|
}
|
|
117
59
|
/**
|
|
118
|
-
* Set
|
|
60
|
+
* Set the transaction fee
|
|
119
61
|
*
|
|
120
|
-
* @param
|
|
121
|
-
* @returns this builder instance for chaining
|
|
62
|
+
* @param {string | bigint} feeValue - the fee value
|
|
122
63
|
*/
|
|
123
|
-
|
|
124
|
-
|
|
64
|
+
fee(feeValue) {
|
|
65
|
+
const fee = typeof feeValue === 'string' ? feeValue : feeValue.toString();
|
|
66
|
+
this.transaction._fee.fee = fee;
|
|
125
67
|
return this;
|
|
126
68
|
}
|
|
127
69
|
/**
|
|
128
|
-
*
|
|
129
|
-
* Creates a credential with the provided signatures
|
|
70
|
+
* Set the fee state for dynamic fee calculation (P-chain transactions)
|
|
130
71
|
*
|
|
131
|
-
* @param
|
|
132
|
-
* @param signatures - Array of signature hex strings or empty strings for placeholders
|
|
133
|
-
* @returns Credential instance
|
|
72
|
+
* @param {FlrpFeeState} state - the fee state from the network
|
|
134
73
|
*/
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
74
|
+
feeState(state) {
|
|
75
|
+
this.transaction._feeState = state;
|
|
76
|
+
return this;
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Set the amount for the transaction
|
|
80
|
+
*
|
|
81
|
+
* @param {bigint | string} value - the amount to transfer
|
|
82
|
+
*/
|
|
83
|
+
amount(value) {
|
|
84
|
+
const valueBigInt = typeof value === 'string' ? BigInt(value) : value;
|
|
85
|
+
this.validateAmount(valueBigInt);
|
|
86
|
+
this.transaction._amount = valueBigInt;
|
|
87
|
+
return this;
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Compute addressesIndex for UTXOs following AVAX P approach.
|
|
91
|
+
* addressesIndex[senderIdx] = position of sender[senderIdx] in UTXO's address list
|
|
92
|
+
*
|
|
93
|
+
* IMPORTANT: UTXO addresses are sorted lexicographically by byte value to match
|
|
94
|
+
* on-chain storage order. The API may return addresses in arbitrary order, but
|
|
95
|
+
* on-chain UTXOs always store addresses in sorted order.
|
|
96
|
+
*
|
|
97
|
+
* Example:
|
|
98
|
+
* A = user key, B = hsm key, C = backup key
|
|
99
|
+
* sender (bitgoAddresses) = [ A, B, C ]
|
|
100
|
+
* utxo.addresses (from API) = [ B, C, A ]
|
|
101
|
+
* sorted utxo.addresses = [ A, B, C ] (sorted by hex value)
|
|
102
|
+
* addressesIndex = [ 0, 1, 2 ]
|
|
103
|
+
* (sender[0]=A is at position 0 in sorted UTXO, sender[1]=B is at position 1, etc.)
|
|
104
|
+
*
|
|
105
|
+
* @protected
|
|
106
|
+
*/
|
|
107
|
+
computeAddressesIndex() {
|
|
108
|
+
const sender = this.transaction._fromAddresses;
|
|
109
|
+
this.transaction._utxos.forEach((utxo) => {
|
|
110
|
+
if (utxo.addressesIndex && utxo.addressesIndex.length > 0) {
|
|
111
|
+
return;
|
|
112
|
+
}
|
|
113
|
+
if (utxo.addresses && utxo.addresses.length > 0) {
|
|
114
|
+
const sortedAddresses = utils_1.default.sortAddressesByHex(utxo.addresses);
|
|
115
|
+
utxo.addresses = sortedAddresses;
|
|
116
|
+
const utxoAddresses = sortedAddresses.map((a) => utils_1.default.parseAddress(a));
|
|
117
|
+
utxo.addressesIndex = sender.map((a) => utxoAddresses.findIndex((u) => Buffer.compare(Buffer.from(u), Buffer.from(a)) === 0));
|
|
165
118
|
}
|
|
166
119
|
});
|
|
167
|
-
try {
|
|
168
|
-
return new flarejs_1.Credential(sigs);
|
|
169
|
-
}
|
|
170
|
-
catch (error) {
|
|
171
|
-
throw new sdk_core_1.BuildTransactionError(`Failed to create credential: ${error instanceof Error ? error.message : 'unknown error'}`);
|
|
172
|
-
}
|
|
173
120
|
}
|
|
174
121
|
/**
|
|
175
|
-
*
|
|
122
|
+
* Compute addressesIndex from parsed transaction data.
|
|
123
|
+
* Similar to computeAddressesIndex() but used when parsing existing transactions
|
|
124
|
+
* via initBuilder().
|
|
125
|
+
*
|
|
126
|
+
* IMPORTANT: UTXO addresses are sorted lexicographically by byte value to match
|
|
127
|
+
* on-chain storage order, ensuring consistency with fresh builds.
|
|
128
|
+
*
|
|
129
|
+
* @protected
|
|
176
130
|
*/
|
|
177
|
-
|
|
178
|
-
|
|
131
|
+
computeAddressesIndexFromParsed() {
|
|
132
|
+
const sender = this.transaction._fromAddresses;
|
|
133
|
+
if (!sender || sender.length === 0)
|
|
134
|
+
return;
|
|
135
|
+
this.transaction._utxos.forEach((utxo) => {
|
|
136
|
+
if (utxo.addresses && utxo.addresses.length > 0) {
|
|
137
|
+
const sortedAddresses = utils_1.default.sortAddressesByHex(utxo.addresses);
|
|
138
|
+
utxo.addresses = sortedAddresses;
|
|
139
|
+
const utxoAddresses = sortedAddresses.map((a) => utils_1.default.parseAddress(a));
|
|
140
|
+
utxo.addressesIndex = sender.map((senderAddr) => utxoAddresses.findIndex((utxoAddr) => Buffer.compare(Buffer.from(utxoAddr), Buffer.from(senderAddr)) === 0));
|
|
141
|
+
}
|
|
142
|
+
});
|
|
179
143
|
}
|
|
180
144
|
/**
|
|
181
|
-
*
|
|
182
|
-
*
|
|
145
|
+
* Validate UTXOs have consistent addresses.
|
|
146
|
+
* Note: UTXO threshold can differ from transaction threshold - each UTXO has its own
|
|
147
|
+
* signature requirement based on how it was created (e.g., change outputs may have threshold=1).
|
|
148
|
+
* @protected
|
|
183
149
|
*/
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
150
|
+
validateUtxoAddresses() {
|
|
151
|
+
this.transaction._utxos.forEach((utxo) => {
|
|
152
|
+
if (!utxo) {
|
|
153
|
+
throw new sdk_core_1.BuildTransactionError('Utxo is undefined');
|
|
154
|
+
}
|
|
155
|
+
if (utxo.addressesIndex?.includes(-1)) {
|
|
156
|
+
throw new sdk_core_1.BuildTransactionError('Addresses are inconsistent: ' + utxo.txid);
|
|
157
|
+
}
|
|
158
|
+
if (utxo.threshold !== undefined && utxo.threshold <= 0) {
|
|
159
|
+
throw new sdk_core_1.BuildTransactionError('UTXO threshold must be positive: ' + utxo.txid);
|
|
160
|
+
}
|
|
161
|
+
});
|
|
189
162
|
}
|
|
190
163
|
/**
|
|
191
|
-
*
|
|
192
|
-
*
|
|
164
|
+
* Create credential with dynamic ordering based on addressesIndex from UTXO.
|
|
165
|
+
* Matches AVAX P behavior: signature order depends on UTXO address positions.
|
|
166
|
+
*
|
|
167
|
+
* addressesIndex[senderIdx] = utxoPosition tells us where each sender is in the UTXO.
|
|
168
|
+
* We create signature slots ordered by utxoPosition (smaller position = earlier slot).
|
|
169
|
+
*
|
|
170
|
+
* @param utxo - The UTXO to create credential for
|
|
171
|
+
* @param threshold - Number of signatures required for this specific UTXO
|
|
172
|
+
* @returns Credential with empty signatures ordered based on UTXO positions
|
|
173
|
+
* @protected
|
|
193
174
|
*/
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
}
|
|
229
|
-
return
|
|
175
|
+
createCredentialForUtxo(utxo, threshold) {
|
|
176
|
+
const sender = this.transaction._fromAddresses;
|
|
177
|
+
const addressesIndex = utxo.addressesIndex ?? [];
|
|
178
|
+
// either user (0) or recovery (2)
|
|
179
|
+
const firstIndex = this.recoverSigner ? 2 : 0;
|
|
180
|
+
const bitgoIndex = 1;
|
|
181
|
+
if (threshold === 1) {
|
|
182
|
+
if (sender && sender.length > firstIndex && addressesIndex[firstIndex] !== undefined) {
|
|
183
|
+
return new flarejs_1.Credential([utils_1.default.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex'))]);
|
|
184
|
+
}
|
|
185
|
+
return new flarejs_1.Credential([utils_1.default.createNewSig('')]);
|
|
186
|
+
}
|
|
187
|
+
// If we have valid addressesIndex, use it to determine signature order
|
|
188
|
+
// addressesIndex[senderIdx] = position in UTXO
|
|
189
|
+
// Smaller position = earlier slot in signature array
|
|
190
|
+
if (addressesIndex.length >= 2 && sender && sender.length >= threshold) {
|
|
191
|
+
let emptySignatures;
|
|
192
|
+
if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {
|
|
193
|
+
emptySignatures = [
|
|
194
|
+
utils_1.default.createNewSig(''),
|
|
195
|
+
utils_1.default.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')),
|
|
196
|
+
];
|
|
197
|
+
}
|
|
198
|
+
else {
|
|
199
|
+
emptySignatures = [
|
|
200
|
+
utils_1.default.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')),
|
|
201
|
+
utils_1.default.createNewSig(''),
|
|
202
|
+
];
|
|
203
|
+
}
|
|
204
|
+
return new flarejs_1.Credential(emptySignatures);
|
|
205
|
+
}
|
|
206
|
+
const emptySignatures = [];
|
|
207
|
+
for (let i = 0; i < threshold; i++) {
|
|
208
|
+
emptySignatures.push(utils_1.default.createNewSig(''));
|
|
209
|
+
}
|
|
210
|
+
return new flarejs_1.Credential(emptySignatures);
|
|
230
211
|
}
|
|
231
212
|
/**
|
|
232
|
-
*
|
|
233
|
-
*
|
|
213
|
+
* Create AddressMap based on addressesIndex following AVAX P approach.
|
|
214
|
+
* Maps each sender address to its signature slot based on UTXO position ordering.
|
|
215
|
+
*
|
|
216
|
+
* addressesIndex[senderIdx] = utxoPosition
|
|
217
|
+
* Signature slots are ordered by utxoPosition (smaller = earlier slot).
|
|
218
|
+
*
|
|
219
|
+
* @param utxo - The UTXO to create AddressMap for
|
|
220
|
+
* @param threshold - Number of signatures required for this specific UTXO
|
|
221
|
+
* @returns AddressMap that maps addresses to signature slots based on UTXO order
|
|
222
|
+
* @protected
|
|
234
223
|
*/
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
224
|
+
createAddressMapForUtxo(utxo, threshold) {
|
|
225
|
+
const addressMap = new flarejs_1.utils.AddressMap();
|
|
226
|
+
const sender = this.transaction._fromAddresses;
|
|
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);
|
|
236
|
+
}
|
|
237
|
+
return addressMap;
|
|
238
|
+
}
|
|
239
|
+
if (addressesIndex.length >= 2 && sender && sender.length >= threshold) {
|
|
240
|
+
if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {
|
|
241
|
+
addressMap.set(new flarejs_1.Address(sender[bitgoIndex]), 0);
|
|
242
|
+
addressMap.set(new flarejs_1.Address(sender[firstIndex]), 1);
|
|
243
|
+
}
|
|
244
|
+
else {
|
|
245
|
+
addressMap.set(new flarejs_1.Address(sender[firstIndex]), 0);
|
|
246
|
+
addressMap.set(new flarejs_1.Address(sender[bitgoIndex]), 1);
|
|
247
|
+
}
|
|
248
|
+
return addressMap;
|
|
249
|
+
}
|
|
250
|
+
if (sender && sender.length >= threshold) {
|
|
251
|
+
sender.slice(0, threshold).forEach((addr, i) => {
|
|
252
|
+
addressMap.set(new flarejs_1.Address(addr), i);
|
|
253
|
+
});
|
|
254
|
+
}
|
|
255
|
+
return addressMap;
|
|
256
|
+
}
|
|
257
|
+
/**
|
|
258
|
+
* Create credential using the ACTUAL sigIndices from FlareJS.
|
|
259
|
+
*
|
|
260
|
+
* This method determines which sender addresses correspond to which sigIndex positions,
|
|
261
|
+
* then creates the credential with signatures in the correct order matching the sigIndices.
|
|
262
|
+
*
|
|
263
|
+
* sigIndices tell us which positions in the UTXO's owner addresses need to sign.
|
|
264
|
+
* We need to figure out which sender addresses are at those positions and create
|
|
265
|
+
* signature slots in the same order as sigIndices.
|
|
266
|
+
*
|
|
267
|
+
* @param utxo - The UTXO to create credential for
|
|
268
|
+
* @param threshold - Number of signatures required
|
|
269
|
+
* @param actualSigIndices - The actual sigIndices from FlareJS's built input
|
|
270
|
+
* @returns Credential with signatures ordered to match sigIndices
|
|
271
|
+
* @protected
|
|
272
|
+
*/
|
|
273
|
+
createCredentialForUtxoWithSigIndices(utxo, threshold, actualSigIndices) {
|
|
274
|
+
const sender = this.transaction._fromAddresses;
|
|
275
|
+
const addressesIndex = utxo.addressesIndex ?? [];
|
|
276
|
+
// either user (0) or recovery (2)
|
|
277
|
+
const firstIndex = this.recoverSigner ? 2 : 0;
|
|
278
|
+
if (threshold === 1) {
|
|
279
|
+
if (sender && sender.length > firstIndex && addressesIndex[firstIndex] !== undefined) {
|
|
280
|
+
return new flarejs_1.Credential([utils_1.default.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex'))]);
|
|
281
|
+
}
|
|
282
|
+
return new flarejs_1.Credential([utils_1.default.createNewSig('')]);
|
|
283
|
+
}
|
|
284
|
+
// For threshold >= 2, use the actual sigIndices order from FlareJS
|
|
285
|
+
// sigIndices[i] = position in UTXO's owner addresses that needs to sign
|
|
286
|
+
// addressesIndex[senderIdx] = position in UTXO's owner addresses for that sender
|
|
287
|
+
//
|
|
288
|
+
// We need to find which sender corresponds to each sigIndex and create signatures
|
|
289
|
+
// in the sigIndices order.
|
|
290
|
+
if (actualSigIndices.length >= 2 && addressesIndex.length >= 2 && sender && sender.length >= threshold) {
|
|
291
|
+
const emptySignatures = [];
|
|
292
|
+
for (const sigIdx of actualSigIndices) {
|
|
293
|
+
// Find which sender address is at this UTXO position
|
|
294
|
+
// addressesIndex[senderIdx] tells us which UTXO position each sender is at
|
|
295
|
+
const senderIdx = addressesIndex.findIndex((utxoPos) => utxoPos === sigIdx);
|
|
296
|
+
if (senderIdx === firstIndex) {
|
|
297
|
+
// This sigIndex slot is for user/recovery - embed their address
|
|
298
|
+
emptySignatures.push(utils_1.default.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')));
|
|
299
|
+
}
|
|
300
|
+
else {
|
|
301
|
+
// BitGo (HSM) or unknown sender - empty signature
|
|
302
|
+
emptySignatures.push(utils_1.default.createNewSig(''));
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
return new flarejs_1.Credential(emptySignatures);
|
|
306
|
+
}
|
|
307
|
+
// Fallback: create threshold empty signatures
|
|
308
|
+
const emptySignatures = [];
|
|
309
|
+
for (let i = 0; i < threshold; i++) {
|
|
310
|
+
emptySignatures.push(utils_1.default.createNewSig(''));
|
|
311
|
+
}
|
|
312
|
+
return new flarejs_1.Credential(emptySignatures);
|
|
313
|
+
}
|
|
314
|
+
/**
|
|
315
|
+
* Create AddressMap using the ACTUAL sigIndices from FlareJS.
|
|
316
|
+
*
|
|
317
|
+
* Maps sender addresses to signature slots based on the actual sigIndices order.
|
|
318
|
+
*
|
|
319
|
+
* @param utxo - The UTXO to create AddressMap for
|
|
320
|
+
* @param threshold - Number of signatures required
|
|
321
|
+
* @param actualSigIndices - The actual sigIndices from FlareJS's built input
|
|
322
|
+
* @returns AddressMap that maps addresses to signature slots
|
|
323
|
+
* @protected
|
|
324
|
+
*/
|
|
325
|
+
createAddressMapForUtxoWithSigIndices(utxo, threshold, actualSigIndices) {
|
|
326
|
+
const addressMap = new flarejs_1.utils.AddressMap();
|
|
327
|
+
const sender = this.transaction._fromAddresses;
|
|
328
|
+
const addressesIndex = utxo.addressesIndex ?? [];
|
|
329
|
+
const firstIndex = this.recoverSigner ? 2 : 0;
|
|
330
|
+
const bitgoIndex = 1;
|
|
331
|
+
if (threshold === 1) {
|
|
332
|
+
if (sender && sender.length > firstIndex) {
|
|
333
|
+
addressMap.set(new flarejs_1.Address(sender[firstIndex]), 0);
|
|
334
|
+
}
|
|
335
|
+
else if (sender && sender.length > 0) {
|
|
336
|
+
addressMap.set(new flarejs_1.Address(sender[0]), 0);
|
|
337
|
+
}
|
|
338
|
+
return addressMap;
|
|
339
|
+
}
|
|
340
|
+
// For threshold >= 2, map addresses based on actual sigIndices order
|
|
341
|
+
if (actualSigIndices.length >= 2 && addressesIndex.length >= 2 && sender && sender.length >= threshold) {
|
|
342
|
+
actualSigIndices.forEach((sigIdx, slotIdx) => {
|
|
343
|
+
// Find which sender is at this UTXO position
|
|
344
|
+
const senderIdx = addressesIndex.findIndex((utxoPos) => utxoPos === sigIdx);
|
|
345
|
+
if (senderIdx === bitgoIndex || senderIdx === firstIndex) {
|
|
346
|
+
addressMap.set(new flarejs_1.Address(sender[senderIdx]), slotIdx);
|
|
347
|
+
}
|
|
348
|
+
});
|
|
349
|
+
return addressMap;
|
|
350
|
+
}
|
|
351
|
+
// Fallback
|
|
352
|
+
if (sender && sender.length >= threshold) {
|
|
353
|
+
sender.slice(0, threshold).forEach((addr, i) => {
|
|
354
|
+
addressMap.set(new flarejs_1.Address(addr), i);
|
|
355
|
+
});
|
|
356
|
+
}
|
|
357
|
+
return addressMap;
|
|
249
358
|
}
|
|
250
359
|
}
|
|
251
360
|
exports.AtomicTransactionBuilder = AtomicTransactionBuilder;
|
|
252
|
-
//# 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, BaseTransaction } from '@bitgo-beta/sdk-core';\nimport { Credential, Signature, TransferableInput, TransferableOutput } from '@flarenetwork/flarejs';\nimport { TransactionExplanation, DecodedUtxoObj } from './iface';\n\n// Constants for signature handling\nconst SECP256K1_SIGNATURE_LENGTH = 65;\n\n/**\n * Flare P-chain atomic transaction builder with FlareJS credential support.\n * This provides the foundation for building Flare P-chain transactions with proper\n * credential handling using FlareJS Credential and Signature classes.\n */\nexport abstract class AtomicTransactionBuilder {\n  protected readonly _coinConfig: Readonly<CoinConfig>;\n  // External chain id (destination) for export transactions\n  protected _externalChainId: Buffer | undefined;\n\n  protected _utxos: DecodedUtxoObj[] = [];\n\n  protected transaction: {\n    _network: Record<string, unknown>;\n    _networkID: number;\n    _blockchainID: Buffer;\n    _assetId: Buffer;\n    _fromAddresses: string[];\n    _to: string[];\n    _locktime: bigint;\n    _threshold: number;\n    _fee: { fee: string; feeRate?: string; size?: number };\n    hasCredentials: boolean;\n    _tx?: unknown;\n    setTransaction: (tx: unknown) => void;\n  } = {\n    _network: {},\n    _networkID: 0,\n    _blockchainID: Buffer.alloc(0),\n    _assetId: Buffer.alloc(0),\n    _fromAddresses: [],\n    _to: [],\n    _locktime: 0n,\n    _threshold: 1,\n    _fee: { fee: '0' },\n    hasCredentials: false,\n    setTransaction: function (_tx: unknown) {\n      this._tx = _tx;\n    },\n  };\n\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    this._coinConfig = coinConfig;\n  }\n\n  protected abstract get transactionType(): TransactionType;\n\n  validateAmount(amount: bigint): void {\n    if (amount <= 0n) {\n      throw new BuildTransactionError('Amount must be positive');\n    }\n  }\n\n  /**\n   * Validates that credentials array is properly formed\n   * @param credentials - Array of credentials to validate\n   */\n  protected validateCredentials(credentials: Credential[]): void {\n    if (!Array.isArray(credentials)) {\n      throw new BuildTransactionError('Credentials must be an array');\n    }\n\n    credentials.forEach((credential, index) => {\n      if (!(credential instanceof Credential)) {\n        throw new BuildTransactionError(`Invalid credential at index ${index}`);\n      }\n    });\n  }\n\n  /**\n   * Creates inputs, outputs, and credentials for Flare P-chain atomic transactions.\n   * Based on AVAX P-chain implementation adapted for FlareJS.\n   *\n   * Note: This is a simplified implementation that creates the core structure.\n   * The FlareJS type system integration will be refined in future iterations.\n   *\n   * @param total - Total amount needed including fees\n   * @returns Object containing TransferableInput[], TransferableOutput[], and Credential[]\n   */\n  protected createInputOutput(total: bigint): {\n    inputs: TransferableInput[];\n    outputs: TransferableOutput[];\n    credentials: Credential[];\n  } {\n    if (!this._utxos || this._utxos.length === 0) {\n      throw new BuildTransactionError('UTXOs are required for creating inputs and outputs');\n    }\n\n    const inputs: TransferableInput[] = [];\n    const outputs: TransferableOutput[] = [];\n    const credentials: Credential[] = [];\n\n    let inputSum = 0n;\n    const addressIndices: { [address: string]: number } = {};\n    let nextAddressIndex = 0;\n\n    // Sort UTXOs by amount in descending order for optimal coin selection\n    const sortedUtxos = [...this._utxos].sort((a, b) => {\n      const amountA = BigInt(a.amount);\n      const amountB = BigInt(b.amount);\n      if (amountA > amountB) return -1;\n      if (amountA < amountB) return 1;\n      return 0;\n    });\n\n    // Process UTXOs to create inputs and credentials\n    for (const utxo of sortedUtxos) {\n      const utxoAmount = BigInt(utxo.amount);\n\n      if (inputSum >= total) {\n        break; // We have enough inputs\n      }\n\n      // TODO: Create proper FlareJS TransferableInput once type issues are resolved\n      // For now, we create a placeholder that demonstrates the structure\n      // The actual FlareJS integration will need proper UTXOID handling\n\n      // Track input sum\n      inputSum += utxoAmount;\n\n      // Track address indices for signature ordering (mimics AVAX pattern)\n      const addressIndexArray: number[] = [];\n      for (const address of utxo.addresses) {\n        if (!(address in addressIndices)) {\n          addressIndices[address] = nextAddressIndex++;\n        }\n        addressIndexArray.push(addressIndices[address]);\n      }\n\n      // Store address indices on the UTXO for credential creation\n      utxo.addressesIndex = addressIndexArray;\n\n      // Create credential with placeholder signatures\n      // In a real implementation, these would be actual signatures\n      const signatures = Array.from({ length: utxo.threshold }, () => '');\n      const credential = this.createFlareCredential(0, signatures);\n      credentials.push(credential);\n    }\n\n    // Verify we have enough inputs\n    if (inputSum < total) {\n      throw new BuildTransactionError(`Insufficient funds: need ${total}, have ${inputSum}`);\n    }\n\n    // TODO: Create change output if we have excess input\n    // The TransferableOutput creation will be implemented once FlareJS types are resolved\n\n    return { inputs, outputs, credentials };\n  }\n\n  /**\n   * Set UTXOs for the transaction. This is required for creating inputs and outputs.\n   *\n   * @param utxos - Array of decoded UTXO objects\n   * @returns this builder instance for chaining\n   */\n  utxos(utxos: DecodedUtxoObj[]): this {\n    this._utxos = utxos;\n    return this;\n  }\n\n  /**\n   * Flare equivalent of Avalanche's SelectCredentialClass\n   * Creates a credential with the provided signatures\n   *\n   * @param credentialId - The credential ID (not used in FlareJS but kept for compatibility)\n   * @param signatures - Array of signature hex strings or empty strings for placeholders\n   * @returns Credential instance\n   */\n  protected createFlareCredential(_credentialId: number, signatures: string[]): Credential {\n    if (!Array.isArray(signatures)) {\n      throw new BuildTransactionError('Signatures must be an array');\n    }\n\n    if (signatures.length === 0) {\n      throw new BuildTransactionError('Signatures array cannot be empty');\n    }\n\n    const sigs = signatures.map((sig, index) => {\n      // Handle empty/placeholder signatures\n      if (!sig || sig.length === 0) {\n        return new Signature(new Uint8Array(SECP256K1_SIGNATURE_LENGTH));\n      }\n\n      // Validate hex string format\n      const cleanSig = sig.startsWith('0x') ? sig.slice(2) : sig;\n      if (!/^[0-9a-fA-F]*$/.test(cleanSig)) {\n        throw new BuildTransactionError(`Invalid hex signature at index ${index}: contains non-hex characters`);\n      }\n\n      // Convert to buffer and validate length\n      const sigBuffer = Buffer.from(cleanSig, 'hex');\n      if (sigBuffer.length > SECP256K1_SIGNATURE_LENGTH) {\n        throw new BuildTransactionError(\n          `Signature too long at index ${index}: ${sigBuffer.length} bytes (max ${SECP256K1_SIGNATURE_LENGTH})`\n        );\n      }\n\n      // Create fixed-length buffer and copy signature data\n      const fixedLengthBuffer = Buffer.alloc(SECP256K1_SIGNATURE_LENGTH);\n      sigBuffer.copy(fixedLengthBuffer);\n\n      try {\n        return new Signature(new Uint8Array(fixedLengthBuffer));\n      } catch (error) {\n        throw new BuildTransactionError(\n          `Failed to create signature at index ${index}: ${error instanceof Error ? error.message : 'unknown error'}`\n        );\n      }\n    });\n\n    try {\n      return new Credential(sigs);\n    } catch (error) {\n      throw new BuildTransactionError(\n        `Failed to create credential: ${error instanceof Error ? error.message : 'unknown error'}`\n      );\n    }\n  }\n\n  /**\n   * Base initBuilder used by concrete builders. For now just returns this so fluent API works.\n   */\n  initBuilder(_tx: unknown): this {\n    return this;\n  }\n\n  /**\n   * Sign transaction with private key (placeholder implementation)\n   * TODO: Implement proper FlareJS signing\n   */\n  sign(_params: { key: string }): this {\n    // TODO: Implement FlareJS signing\n    // For now, just mark as having credentials\n    this.transaction.hasCredentials = true;\n    return this;\n  }\n\n  /**\n   * Build the transaction (placeholder implementation)\n   * TODO: Implement proper FlareJS transaction building\n   */\n  async build(): Promise<BaseTransaction> {\n    // TODO: Create actual FlareJS UnsignedTx\n    // For now, return a mock transaction that satisfies the interface\n    const mockTransaction = {\n      _id: 'mock-transaction-id',\n      _inputs: [],\n      _outputs: [],\n      _type: this.transactionType,\n      signature: [] as string[],\n      toBroadcastFormat: () => 'mock-tx-hex',\n      toJson: () => ({}),\n      explainTransaction: (): TransactionExplanation => ({\n        type: this.transactionType,\n        inputs: [],\n        outputs: [],\n        outputAmount: '0',\n        rewardAddresses: [],\n        id: 'mock-transaction-id',\n        changeOutputs: [],\n        changeAmount: '0',\n        fee: { fee: '0' },\n      }),\n      isTransactionForCChain: false,\n      fromAddresses: [],\n      validationErrors: [],\n      loadInputsAndOutputs: () => {\n        /* placeholder */\n      },\n      inputs: () => [],\n      outputs: () => [],\n      fee: () => ({ fee: '0' }),\n      feeRate: () => 0,\n      id: () => 'mock-transaction-id',\n      type: this.transactionType,\n    } as unknown as BaseTransaction;\n\n    return mockTransaction;\n  }\n\n  /**\n   * Parse and explain a transaction from hex (placeholder implementation)\n   * TODO: Implement proper FlareJS transaction parsing\n   */\n  explainTransaction(): TransactionExplanation {\n    // TODO: Parse actual FlareJS transaction\n    // For now, return basic explanation\n    return {\n      type: this.transactionType,\n      inputs: [],\n      outputs: [],\n      outputAmount: '0',\n      rewardAddresses: [],\n      id: 'mock-transaction-id',\n      changeOutputs: [],\n      changeAmount: '0',\n      fee: { fee: '0' },\n    };\n  }\n}\n"]}
|
|
361
|
+
//# 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   * Compute addressesIndex for UTXOs following AVAX P approach.\n   * addressesIndex[senderIdx] = position of sender[senderIdx] in UTXO's address list\n   *\n   * IMPORTANT: UTXO addresses are sorted lexicographically by byte value to match\n   * on-chain storage order. The API may return addresses in arbitrary order, but\n   * on-chain UTXOs always store addresses in sorted order.\n   *\n   * Example:\n   *   A = user key, B = hsm key, C = backup key\n   *   sender (bitgoAddresses) = [ A, B, C ]\n   *   utxo.addresses (from API) = [ B, C, A ]\n   *   sorted utxo.addresses = [ A, B, C ] (sorted by hex value)\n   *   addressesIndex = [ 0, 1, 2 ]\n   *   (sender[0]=A is at position 0 in sorted UTXO, sender[1]=B is at position 1, etc.)\n   *\n   * @protected\n   */\n  protected computeAddressesIndex(): void {\n    const sender = this.transaction._fromAddresses;\n\n    this.transaction._utxos.forEach((utxo) => {\n      if (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   * Compute addressesIndex from parsed transaction data.\n   * Similar to computeAddressesIndex() but used when parsing existing transactions\n   * via initBuilder().\n   *\n   * IMPORTANT: UTXO addresses are sorted lexicographically by byte value to match\n   * on-chain storage order, ensuring consistency with fresh builds.\n   *\n   * @protected\n   */\n  protected computeAddressesIndexFromParsed(): void {\n    const sender = this.transaction._fromAddresses;\n    if (!sender || sender.length === 0) return;\n\n    this.transaction._utxos.forEach((utxo) => {\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((senderAddr) =>\n          utxoAddresses.findIndex((utxoAddr) => Buffer.compare(Buffer.from(utxoAddr), Buffer.from(senderAddr)) === 0)\n        );\n      }\n    });\n  }\n\n  /**\n   * Validate UTXOs have consistent addresses.\n   * Note: UTXO threshold can differ from transaction threshold - each UTXO has its own\n   * signature requirement based on how it was created (e.g., change outputs may have threshold=1).\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      if (utxo.threshold !== undefined && utxo.threshold <= 0) {\n        throw new BuildTransactionError('UTXO threshold must be positive: ' + utxo.txid);\n      }\n    });\n  }\n\n  /**\n   * Create credential with dynamic ordering based on addressesIndex from UTXO.\n   * Matches AVAX P behavior: signature order depends on UTXO address positions.\n   *\n   * addressesIndex[senderIdx] = utxoPosition tells us where each sender is in the UTXO.\n   * We create signature slots ordered by utxoPosition (smaller position = earlier slot).\n   *\n   * @param utxo - The UTXO to create credential for\n   * @param threshold - Number of signatures required for this specific UTXO\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._fromAddresses;\n    const addressesIndex = utxo.addressesIndex ?? [];\n\n    // either user (0) or recovery (2)\n    const firstIndex = this.recoverSigner ? 2 : 0;\n    const bitgoIndex = 1;\n\n    if (threshold === 1) {\n      if (sender && sender.length > firstIndex && addressesIndex[firstIndex] !== undefined) {\n        return new Credential([utils.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex'))]);\n      }\n      return new Credential([utils.createNewSig('')]);\n    }\n\n    // If we have valid addressesIndex, use it to determine signature order\n    // addressesIndex[senderIdx] = position in UTXO\n    // Smaller position = earlier slot in signature array\n    if (addressesIndex.length >= 2 && sender && sender.length >= threshold) {\n      let emptySignatures: ReturnType<typeof utils.createNewSig>[];\n\n      if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {\n        emptySignatures = [\n          utils.createNewSig(''),\n          utils.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')),\n        ];\n      } else {\n        emptySignatures = [\n          utils.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')),\n          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 based on addressesIndex following AVAX P approach.\n   * Maps each sender address to its signature slot based on UTXO position ordering.\n   *\n   * addressesIndex[senderIdx] = utxoPosition\n   * Signature slots are ordered by utxoPosition (smaller = earlier slot).\n   *\n   * @param utxo - The UTXO to create AddressMap for\n   * @param threshold - Number of signatures required for this specific UTXO\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._fromAddresses;\n    const addressesIndex = utxo.addressesIndex ?? [];\n\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      if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {\n        addressMap.set(new Address(sender[bitgoIndex]), 0);\n        addressMap.set(new Address(sender[firstIndex]), 1);\n      } else {\n        addressMap.set(new Address(sender[firstIndex]), 0);\n        addressMap.set(new Address(sender[bitgoIndex]), 1);\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\n    return addressMap;\n  }\n\n  /**\n   * Create credential using the ACTUAL sigIndices from FlareJS.\n   *\n   * This method determines which sender addresses correspond to which sigIndex positions,\n   * then creates the credential with signatures in the correct order matching the sigIndices.\n   *\n   * sigIndices tell us which positions in the UTXO's owner addresses need to sign.\n   * We need to figure out which sender addresses are at those positions and create\n   * signature slots in the same order as sigIndices.\n   *\n   * @param utxo - The UTXO to create credential for\n   * @param threshold - Number of signatures required\n   * @param actualSigIndices - The actual sigIndices from FlareJS's built input\n   * @returns Credential with signatures ordered to match sigIndices\n   * @protected\n   */\n  protected createCredentialForUtxoWithSigIndices(\n    utxo: DecodedUtxoObj,\n    threshold: number,\n    actualSigIndices: number[]\n  ): Credential {\n    const sender = this.transaction._fromAddresses;\n    const addressesIndex = utxo.addressesIndex ?? [];\n\n    // either user (0) or recovery (2)\n    const firstIndex = this.recoverSigner ? 2 : 0;\n\n    if (threshold === 1) {\n      if (sender && sender.length > firstIndex && addressesIndex[firstIndex] !== undefined) {\n        return new Credential([utils.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex'))]);\n      }\n      return new Credential([utils.createNewSig('')]);\n    }\n\n    // For threshold >= 2, use the actual sigIndices order from FlareJS\n    // sigIndices[i] = position in UTXO's owner addresses that needs to sign\n    // addressesIndex[senderIdx] = position in UTXO's owner addresses for that sender\n    //\n    // We need to find which sender corresponds to each sigIndex and create signatures\n    // in the sigIndices order.\n    if (actualSigIndices.length >= 2 && addressesIndex.length >= 2 && sender && sender.length >= threshold) {\n      const emptySignatures: ReturnType<typeof utils.createNewSig>[] = [];\n\n      for (const sigIdx of actualSigIndices) {\n        // Find which sender address is at this UTXO position\n        // addressesIndex[senderIdx] tells us which UTXO position each sender is at\n        const senderIdx = addressesIndex.findIndex((utxoPos) => utxoPos === sigIdx);\n\n        if (senderIdx === firstIndex) {\n          // This sigIndex slot is for user/recovery - embed their address\n          emptySignatures.push(utils.createEmptySigWithAddress(Buffer.from(sender[firstIndex]).toString('hex')));\n        } else {\n          // BitGo (HSM) or unknown sender - empty signature\n          emptySignatures.push(utils.createNewSig(''));\n        }\n      }\n\n      return new Credential(emptySignatures);\n    }\n\n    // Fallback: create threshold empty signatures\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 using the ACTUAL sigIndices from FlareJS.\n   *\n   * Maps sender addresses to signature slots based on the actual sigIndices order.\n   *\n   * @param utxo - The UTXO to create AddressMap for\n   * @param threshold - Number of signatures required\n   * @param actualSigIndices - The actual sigIndices from FlareJS's built input\n   * @returns AddressMap that maps addresses to signature slots\n   * @protected\n   */\n  protected createAddressMapForUtxoWithSigIndices(\n    utxo: DecodedUtxoObj,\n    threshold: number,\n    actualSigIndices: number[]\n  ): FlareUtils.AddressMap {\n    const addressMap = new FlareUtils.AddressMap();\n    const sender = this.transaction._fromAddresses;\n    const addressesIndex = utxo.addressesIndex ?? [];\n\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    // For threshold >= 2, map addresses based on actual sigIndices order\n    if (actualSigIndices.length >= 2 && addressesIndex.length >= 2 && sender && sender.length >= threshold) {\n      actualSigIndices.forEach((sigIdx, slotIdx) => {\n        // Find which sender is at this UTXO position\n        const senderIdx = addressesIndex.findIndex((utxoPos) => utxoPos === sigIdx);\n\n        if (senderIdx === bitgoIndex || senderIdx === firstIndex) {\n          addressMap.set(new Address(sender[senderIdx]), slotIdx);\n        }\n      });\n\n      return addressMap;\n    }\n\n    // Fallback\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    return addressMap;\n  }\n}\n"]}
|