@bitgo-beta/sdk-coin-flrp 1.0.1-beta.277 → 1.0.1-beta.279
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 +15 -4
- package/dist/src/lib/ExportInPTxBuilder.d.ts.map +1 -1
- package/dist/src/lib/ExportInPTxBuilder.js +245 -43
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +68 -11
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilderFactory.js +6 -1
- package/dist/src/lib/utils.d.ts +12 -0
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +26 -1
- package/dist/test/resources/transactionData/exportInP.d.ts +69 -0
- package/dist/test/resources/transactionData/exportInP.d.ts.map +1 -0
- package/dist/test/resources/transactionData/exportInP.js +140 -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 +121 -0
- package/dist/test/unit/lib/signFlowTestSuit.d.ts +0 -1
- package/dist/test/unit/lib/signFlowTestSuit.d.ts.map +1 -1
- package/dist/test/unit/lib/signFlowTestSuit.js +6 -3
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +5 -5
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { BaseCoin as CoinConfig } from '@bitgo-beta/statics';
|
|
2
2
|
import { TransactionType } from '@bitgo-beta/sdk-core';
|
|
3
3
|
import { AtomicTransactionBuilder } from './atomicTransactionBuilder';
|
|
4
|
-
import { TransferableOutput } from '@flarenetwork/flarejs';
|
|
4
|
+
import { TransferableInput, TransferableOutput, Credential } from '@flarenetwork/flarejs';
|
|
5
5
|
import { Tx } from './iface';
|
|
6
6
|
export declare class ExportInPTxBuilder extends AtomicTransactionBuilder {
|
|
7
7
|
private _amount;
|
|
@@ -12,14 +12,25 @@ export declare class ExportInPTxBuilder extends AtomicTransactionBuilder {
|
|
|
12
12
|
* @param {bigint | string} amount The withdrawal amount
|
|
13
13
|
*/
|
|
14
14
|
amount(value: bigint | string): this;
|
|
15
|
-
initBuilder(tx: Tx): this;
|
|
15
|
+
initBuilder(tx: Tx, rawBytes?: Buffer): this;
|
|
16
16
|
static verifyTxType(txnType: string): boolean;
|
|
17
17
|
verifyTxType(txnType: string): boolean;
|
|
18
18
|
/**
|
|
19
|
-
* Build the export transaction
|
|
19
|
+
* Build the export transaction for P-chain
|
|
20
20
|
* @protected
|
|
21
21
|
*/
|
|
22
22
|
protected buildFlareTransaction(): void;
|
|
23
|
+
/**
|
|
24
|
+
* Create inputs from UTXOs for P-chain export
|
|
25
|
+
* Only selects enough UTXOs to cover the target amount (amount + fee)
|
|
26
|
+
* @returns inputs, change outputs, credentials, and total amount
|
|
27
|
+
*/
|
|
28
|
+
protected createExportInputs(): {
|
|
29
|
+
inputs: TransferableInput[];
|
|
30
|
+
changeOutputs: TransferableOutput[];
|
|
31
|
+
credentials: Credential[];
|
|
32
|
+
totalAmount: bigint;
|
|
33
|
+
};
|
|
23
34
|
/**
|
|
24
35
|
* Create the ExportedOutputs where the recipient address are the sender.
|
|
25
36
|
* Later an importTx should complete the operations signing with the same keys.
|
|
@@ -28,7 +39,7 @@ export declare class ExportInPTxBuilder extends AtomicTransactionBuilder {
|
|
|
28
39
|
protected exportedOutputs(): TransferableOutput[];
|
|
29
40
|
/**
|
|
30
41
|
* Recover UTXOs from inputs
|
|
31
|
-
* @param inputs Array of
|
|
42
|
+
* @param inputs Array of TransferableInput
|
|
32
43
|
* @returns Array of decoded UTXO objects
|
|
33
44
|
*/
|
|
34
45
|
private recoverUtxos;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ExportInPTxBuilder.d.ts","sourceRoot":"","sources":["../../../src/lib/ExportInPTxBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC7D,OAAO,EAAuC,eAAe,EAAE,MAAM,sBAAsB,CAAC;AAC5F,OAAO,EAAE,wBAAwB,EAAE,MAAM,4BAA4B,CAAC;AACtE,OAAO,
|
|
1
|
+
{"version":3,"file":"ExportInPTxBuilder.d.ts","sourceRoot":"","sources":["../../../src/lib/ExportInPTxBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC7D,OAAO,EAAuC,eAAe,EAAE,MAAM,sBAAsB,CAAC;AAC5F,OAAO,EAAE,wBAAwB,EAAE,MAAM,4BAA4B,CAAC;AACtE,OAAO,EAOL,iBAAiB,EACjB,kBAAkB,EAMlB,UAAU,EAEX,MAAM,uBAAuB,CAAC;AAE/B,OAAO,EAAmE,EAAE,EAAE,MAAM,SAAS,CAAC;AAE9F,qBAAa,kBAAmB,SAAQ,wBAAwB;IAC9D,OAAO,CAAC,OAAO,CAAS;gBAEZ,WAAW,EAAE,QAAQ,CAAC,UAAU,CAAC;IAY7C,SAAS,KAAK,eAAe,IAAI,eAAe,CAE/C;IAED;;;OAGG;IACH,MAAM,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI;IAOpC,WAAW,CAAC,EAAE,EAAE,EAAE,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,IAAI;IA6J5C,MAAM,CAAC,YAAY,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO;IAI7C,YAAY,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO;IAItC;;;OAGG;IACH,SAAS,CAAC,qBAAqB,IAAI,IAAI;IAoDvC;;;;OAIG;IACH,SAAS,CAAC,kBAAkB,IAAI;QAC9B,MAAM,EAAE,iBAAiB,EAAE,CAAC;QAC5B,aAAa,EAAE,kBAAkB,EAAE,CAAC;QACpC,WAAW,EAAE,UAAU,EAAE,CAAC;QAC1B,WAAW,EAAE,MAAM,CAAC;KACrB;IAwGD;;;;OAIG;IACH,SAAS,CAAC,eAAe,IAAI,kBAAkB,EAAE;IAgBjD;;;;OAIG;IACH,OAAO,CAAC,YAAY;CAkBrB"}
|
|
@@ -12,7 +12,12 @@ const iface_1 = require("./iface");
|
|
|
12
12
|
class ExportInPTxBuilder extends atomicTransactionBuilder_1.AtomicTransactionBuilder {
|
|
13
13
|
constructor(_coinConfig) {
|
|
14
14
|
super(_coinConfig);
|
|
15
|
+
// For Export FROM P-chain:
|
|
16
|
+
// - external chain (destination) is C-chain
|
|
17
|
+
// - blockchain ID (source) is P-chain
|
|
15
18
|
this._externalChainId = utils_1.default.cb58Decode(this.transaction._network.cChainBlockchainID);
|
|
19
|
+
// P-chain blockchain ID (from network config - decode from cb58 to hex)
|
|
20
|
+
this.transaction._blockchainID = Buffer.from(utils_1.default.cb58Decode(this.transaction._network.blockchainID)).toString('hex');
|
|
16
21
|
}
|
|
17
22
|
get transactionType() {
|
|
18
23
|
return sdk_core_1.TransactionType.Export;
|
|
@@ -27,36 +32,137 @@ class ExportInPTxBuilder extends atomicTransactionBuilder_1.AtomicTransactionBui
|
|
|
27
32
|
this._amount = valueBigInt;
|
|
28
33
|
return this;
|
|
29
34
|
}
|
|
30
|
-
initBuilder(tx) {
|
|
31
|
-
const
|
|
32
|
-
if (!this.verifyTxType(
|
|
35
|
+
initBuilder(tx, rawBytes) {
|
|
36
|
+
const exportTx = tx;
|
|
37
|
+
if (!this.verifyTxType(exportTx._type)) {
|
|
33
38
|
throw new sdk_core_1.NotSupported('Transaction cannot be parsed or has an unsupported transaction type');
|
|
34
39
|
}
|
|
35
40
|
// The exportedOutputs is a TransferableOutput array.
|
|
36
41
|
// It's expected to have only one output with the addresses of the sender.
|
|
37
|
-
const outputs =
|
|
42
|
+
const outputs = exportTx.outs;
|
|
38
43
|
if (outputs.length !== 1) {
|
|
39
44
|
throw new sdk_core_1.BuildTransactionError('Transaction can have one external output');
|
|
40
45
|
}
|
|
41
46
|
const output = outputs[0];
|
|
42
47
|
const outputTransfer = output.output;
|
|
43
48
|
const assetId = output.assetId.toBytes();
|
|
44
|
-
if (Buffer.compare(assetId, Buffer.from(this.transaction._assetId)) !== 0) {
|
|
49
|
+
if (Buffer.compare(Buffer.from(assetId), Buffer.from(this.transaction._assetId, 'hex')) !== 0) {
|
|
45
50
|
throw new Error('The Asset ID of the output does not match the transaction');
|
|
46
51
|
}
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
52
|
+
const outputOwners = outputTransfer.outputOwners;
|
|
53
|
+
// Set locktime from output
|
|
54
|
+
this.transaction._locktime = outputOwners.locktime.value();
|
|
55
|
+
// Set threshold from output
|
|
56
|
+
this.transaction._threshold = outputOwners.threshold.value();
|
|
57
|
+
// Convert output addresses to buffers and set as fromAddresses
|
|
53
58
|
this.transaction._fromAddresses = outputOwners.addrs.map((addr) => Buffer.from(addr.toBytes()));
|
|
54
59
|
// Set external chain ID from the destination chain
|
|
55
|
-
this._externalChainId = Buffer.from(
|
|
56
|
-
// Set amount from output
|
|
60
|
+
this._externalChainId = Buffer.from(exportTx.destination.toBytes());
|
|
61
|
+
// Set amount from exported output
|
|
57
62
|
this._amount = outputTransfer.amount();
|
|
58
|
-
// Recover UTXOs from inputs
|
|
59
|
-
this.transaction._utxos = this.recoverUtxos(baseTx.
|
|
63
|
+
// Recover UTXOs from base tx inputs
|
|
64
|
+
this.transaction._utxos = this.recoverUtxos([...exportTx.baseTx.inputs]);
|
|
65
|
+
// Calculate and set fee from input/output difference
|
|
66
|
+
const totalInputAmount = exportTx.baseTx.inputs.reduce((sum, input) => sum + input.amount(), BigInt(0));
|
|
67
|
+
const changeOutputAmount = exportTx.baseTx.outputs.reduce((sum, out) => {
|
|
68
|
+
const transferOut = out.output;
|
|
69
|
+
return sum + transferOut.amount();
|
|
70
|
+
}, BigInt(0));
|
|
71
|
+
const fee = totalInputAmount - changeOutputAmount - this._amount;
|
|
72
|
+
this.transaction._fee.fee = fee.toString();
|
|
73
|
+
// Extract credentials from raw bytes
|
|
74
|
+
let hasCredentials = false;
|
|
75
|
+
let credentials = [];
|
|
76
|
+
if (rawBytes) {
|
|
77
|
+
// Try standard extraction first
|
|
78
|
+
const result = utils_1.default.extractCredentialsFromRawBytes(rawBytes, exportTx, 'PVM');
|
|
79
|
+
hasCredentials = result.hasCredentials;
|
|
80
|
+
credentials = result.credentials;
|
|
81
|
+
// If extraction failed but raw bytes are longer, try parsing credentials at known offset
|
|
82
|
+
if ((!hasCredentials || credentials.length === 0) && rawBytes.length > 300) {
|
|
83
|
+
const codec = flarejs_1.utils.getManagerForVM('PVM').getDefaultCodec();
|
|
84
|
+
const txBytesLength = exportTx.toBytes(codec).length;
|
|
85
|
+
if (rawBytes.length > txBytesLength) {
|
|
86
|
+
hasCredentials = true;
|
|
87
|
+
const credResult = utils_1.default.parseCredentialsAtOffset(rawBytes, txBytesLength);
|
|
88
|
+
if (credResult.length > 0) {
|
|
89
|
+
credentials = credResult;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
// If we have parsed credentials with the correct number of credentials for the inputs,
|
|
95
|
+
// use them directly (preserves existing signatures)
|
|
96
|
+
const numInputs = exportTx.baseTx.inputs.length;
|
|
97
|
+
const useDirectCredentials = hasCredentials && credentials.length === numInputs;
|
|
98
|
+
// If there are credentials in raw bytes, store the original bytes to preserve exact format
|
|
99
|
+
if (rawBytes && hasCredentials) {
|
|
100
|
+
this.transaction._rawSignedBytes = rawBytes;
|
|
101
|
+
}
|
|
102
|
+
// Create proper UnsignedTx wrapper with credentials
|
|
103
|
+
const sortedAddresses = [...this.transaction._fromAddresses].sort((a, b) => Buffer.compare(a, b));
|
|
104
|
+
// Helper function to check if a signature is empty (contains no real signature data)
|
|
105
|
+
// A real ECDSA signature will never start with 45 bytes of zeros
|
|
106
|
+
const isSignatureEmpty = (sig) => {
|
|
107
|
+
if (!sig)
|
|
108
|
+
return true;
|
|
109
|
+
const cleanSig = utils_1.default.removeHexPrefix(sig);
|
|
110
|
+
if (cleanSig.length === 0)
|
|
111
|
+
return true;
|
|
112
|
+
// Check if the first 90 hex chars (45 bytes) are all zeros
|
|
113
|
+
// Real signatures from secp256k1 will never have this pattern
|
|
114
|
+
const first90Chars = cleanSig.substring(0, 90);
|
|
115
|
+
return first90Chars === '0'.repeat(90) || first90Chars === '0'.repeat(first90Chars.length);
|
|
116
|
+
};
|
|
117
|
+
// Build txCredentials - either use direct credentials or reconstruct with embedded addresses
|
|
118
|
+
let txCredentials;
|
|
119
|
+
if (useDirectCredentials) {
|
|
120
|
+
// Use the extracted credentials directly - they already have the correct signatures
|
|
121
|
+
// Just ensure empty slots have embedded addresses for signing identification
|
|
122
|
+
txCredentials = credentials;
|
|
123
|
+
}
|
|
124
|
+
else {
|
|
125
|
+
// Reconstruct credentials from scratch with embedded addresses
|
|
126
|
+
txCredentials = exportTx.baseTx.inputs.map((input, idx) => {
|
|
127
|
+
const transferInput = input.input;
|
|
128
|
+
const inputThreshold = transferInput.sigIndicies().length || this.transaction._threshold;
|
|
129
|
+
// Get existing signatures from parsed credentials if available
|
|
130
|
+
const existingSigs = [];
|
|
131
|
+
if (idx < credentials.length) {
|
|
132
|
+
const existingCred = credentials[idx];
|
|
133
|
+
existingSigs.push(...existingCred.getSignatures());
|
|
134
|
+
}
|
|
135
|
+
// Create credential with correct number of slots, preserving existing signatures
|
|
136
|
+
// Empty slots get embedded addresses for slot identification
|
|
137
|
+
const sigSlots = [];
|
|
138
|
+
for (let i = 0; i < inputThreshold; i++) {
|
|
139
|
+
const existingSig = i < existingSigs.length ? existingSigs[i] : null;
|
|
140
|
+
if (existingSig && !isSignatureEmpty(existingSig)) {
|
|
141
|
+
// Use existing non-empty signature (real signature from signing)
|
|
142
|
+
const sigHex = utils_1.default.removeHexPrefix(existingSig);
|
|
143
|
+
sigSlots.push(utils_1.default.createNewSig(sigHex));
|
|
144
|
+
}
|
|
145
|
+
else {
|
|
146
|
+
// Empty slot - create with embedded address for slot identification
|
|
147
|
+
const addrHex = Buffer.from(sortedAddresses[i]).toString('hex');
|
|
148
|
+
sigSlots.push(utils_1.default.createEmptySigWithAddress(addrHex));
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
return new flarejs_1.Credential(sigSlots);
|
|
152
|
+
});
|
|
153
|
+
}
|
|
154
|
+
// Create address maps for signing - one per input/credential
|
|
155
|
+
// Each address map contains all addresses mapped to their indices
|
|
156
|
+
const addressMaps = txCredentials.map(() => {
|
|
157
|
+
const addressMap = new flarejs_1.utils.AddressMap();
|
|
158
|
+
sortedAddresses.forEach((addr, i) => {
|
|
159
|
+
addressMap.set(new flarejs_1.Address(addr), i);
|
|
160
|
+
});
|
|
161
|
+
return addressMap;
|
|
162
|
+
});
|
|
163
|
+
// Always create a new UnsignedTx with properly structured credentials
|
|
164
|
+
const unsignedTx = new flarejs_1.UnsignedTx(exportTx, [], new flarejs_1.utils.AddressMaps(addressMaps), txCredentials);
|
|
165
|
+
this.transaction.setTransaction(unsignedTx);
|
|
60
166
|
return this;
|
|
61
167
|
}
|
|
62
168
|
static verifyTxType(txnType) {
|
|
@@ -66,30 +172,125 @@ class ExportInPTxBuilder extends atomicTransactionBuilder_1.AtomicTransactionBui
|
|
|
66
172
|
return ExportInPTxBuilder.verifyTxType(txnType);
|
|
67
173
|
}
|
|
68
174
|
/**
|
|
69
|
-
* Build the export transaction
|
|
175
|
+
* Build the export transaction for P-chain
|
|
70
176
|
* @protected
|
|
71
177
|
*/
|
|
72
178
|
buildFlareTransaction() {
|
|
73
179
|
// if tx has credentials, tx shouldn't change
|
|
74
180
|
if (this.transaction.hasCredentials)
|
|
75
181
|
return;
|
|
76
|
-
const { inputs, credentials } = this.
|
|
77
|
-
//
|
|
78
|
-
const
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
);
|
|
182
|
+
const { inputs, changeOutputs, credentials, totalAmount } = this.createExportInputs();
|
|
183
|
+
// Calculate fee from transaction fee settings
|
|
184
|
+
const fee = BigInt(this.transaction.fee.fee);
|
|
185
|
+
const targetAmount = this._amount + fee;
|
|
186
|
+
// Verify we have enough funds
|
|
187
|
+
if (totalAmount < targetAmount) {
|
|
188
|
+
throw new sdk_core_1.BuildTransactionError(`Insufficient funds: have ${totalAmount}, need ${targetAmount}`);
|
|
189
|
+
}
|
|
190
|
+
// Create the BaseTx for the P-chain export transaction
|
|
191
|
+
const baseTx = new flarejs_1.avaxSerial.BaseTx(new flarejs_1.Int(this.transaction._networkID), new flarejs_1.Id(Buffer.from(this.transaction._blockchainID, 'hex')), changeOutputs, // change outputs
|
|
192
|
+
inputs, // inputs
|
|
193
|
+
new flarejs_1.Bytes(new Uint8Array(0)) // empty memo
|
|
194
|
+
);
|
|
195
|
+
// Create the P-chain export transaction using pvmSerial.ExportTx
|
|
196
|
+
const exportTx = new flarejs_1.pvmSerial.ExportTx(baseTx, new flarejs_1.Id(this._externalChainId), // destinationChain (C-chain)
|
|
197
|
+
this.exportedOutputs() // exportedOutputs
|
|
198
|
+
);
|
|
199
|
+
// Create address maps for signing - one per input/credential
|
|
200
|
+
const sortedAddresses = [...this.transaction._fromAddresses].sort((a, b) => Buffer.compare(a, b));
|
|
201
|
+
const addressMaps = credentials.map(() => {
|
|
202
|
+
const addressMap = new flarejs_1.utils.AddressMap();
|
|
203
|
+
sortedAddresses.forEach((addr, i) => {
|
|
204
|
+
addressMap.set(new flarejs_1.Address(addr), i);
|
|
205
|
+
});
|
|
206
|
+
return addressMap;
|
|
83
207
|
});
|
|
84
|
-
// Create
|
|
85
|
-
const
|
|
86
|
-
|
|
87
|
-
const addressMap = new flarejs_1.utils.AddressMap([[new flarejs_1.Address(this.transaction._fromAddresses[0]), 0]]);
|
|
88
|
-
const addressMaps = new flarejs_1.utils.AddressMaps([addressMap]);
|
|
89
|
-
const unsignedTx = new flarejs_1.UnsignedTx(exportTx, [], // Empty UTXOs array, will be filled during processing
|
|
90
|
-
addressMaps, credentials);
|
|
208
|
+
// Create unsigned transaction
|
|
209
|
+
const unsignedTx = new flarejs_1.UnsignedTx(exportTx, [], // Empty UTXOs array
|
|
210
|
+
new flarejs_1.utils.AddressMaps(addressMaps), credentials);
|
|
91
211
|
this.transaction.setTransaction(unsignedTx);
|
|
92
212
|
}
|
|
213
|
+
/**
|
|
214
|
+
* Create inputs from UTXOs for P-chain export
|
|
215
|
+
* Only selects enough UTXOs to cover the target amount (amount + fee)
|
|
216
|
+
* @returns inputs, change outputs, credentials, and total amount
|
|
217
|
+
*/
|
|
218
|
+
createExportInputs() {
|
|
219
|
+
const sender = [...this.transaction._fromAddresses];
|
|
220
|
+
if (this.recoverSigner) {
|
|
221
|
+
// switch first and last signer
|
|
222
|
+
const tmp = sender.pop();
|
|
223
|
+
sender.push(sender[0]);
|
|
224
|
+
if (tmp) {
|
|
225
|
+
sender[0] = tmp;
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
const fee = BigInt(this.transaction.fee.fee);
|
|
229
|
+
const targetAmount = this._amount + fee;
|
|
230
|
+
let totalAmount = BigInt(0);
|
|
231
|
+
const inputs = [];
|
|
232
|
+
const credentials = [];
|
|
233
|
+
// Change output threshold is always 1 (matching Flare protocol behavior)
|
|
234
|
+
// This allows easier spending of change while maintaining security for export outputs
|
|
235
|
+
const changeOutputThreshold = 1;
|
|
236
|
+
// Only consume enough UTXOs to cover the target amount (in array order)
|
|
237
|
+
// Inputs will be sorted after selection
|
|
238
|
+
for (const utxo of this.transaction._utxos) {
|
|
239
|
+
// Stop if we already have enough
|
|
240
|
+
if (totalAmount >= targetAmount) {
|
|
241
|
+
break;
|
|
242
|
+
}
|
|
243
|
+
const amount = BigInt(utxo.amount);
|
|
244
|
+
totalAmount += amount;
|
|
245
|
+
// Use the UTXO's own threshold for signature indices
|
|
246
|
+
const utxoThreshold = utxo.threshold || this.transaction._threshold;
|
|
247
|
+
// Create signature indices for the UTXO's threshold
|
|
248
|
+
const sigIndices = [];
|
|
249
|
+
for (let i = 0; i < utxoThreshold; i++) {
|
|
250
|
+
sigIndices.push(i);
|
|
251
|
+
}
|
|
252
|
+
// Use fromNative to create TransferableInput
|
|
253
|
+
const txIdCb58 = utxo.txid; // Already cb58 encoded
|
|
254
|
+
const assetIdCb58 = utils_1.default.cb58Encode(Buffer.from(this.transaction._assetId, 'hex'));
|
|
255
|
+
const transferableInput = flarejs_1.TransferableInput.fromNative(txIdCb58, Number(utxo.outputidx), assetIdCb58, amount, sigIndices);
|
|
256
|
+
inputs.push(transferableInput);
|
|
257
|
+
// Create credential with empty signatures that have embedded addresses for slot identification
|
|
258
|
+
// This allows the signing logic to determine which slot belongs to which address
|
|
259
|
+
const sortedAddrs = [...this.transaction._fromAddresses].sort((a, b) => Buffer.compare(a, b));
|
|
260
|
+
const emptySignatures = sigIndices.map((idx) => {
|
|
261
|
+
const addrHex = Buffer.from(sortedAddrs[idx]).toString('hex');
|
|
262
|
+
return utils_1.default.createEmptySigWithAddress(addrHex);
|
|
263
|
+
});
|
|
264
|
+
credentials.push(new flarejs_1.Credential(emptySignatures));
|
|
265
|
+
}
|
|
266
|
+
// Create change output if there is remaining amount after export and fee
|
|
267
|
+
const changeOutputs = [];
|
|
268
|
+
const changeAmount = totalAmount - this._amount - fee;
|
|
269
|
+
if (changeAmount > BigInt(0)) {
|
|
270
|
+
const assetIdBytes = new Uint8Array(Buffer.from(this.transaction._assetId, 'hex'));
|
|
271
|
+
// Create OutputOwners with the P-chain addresses (sorted by byte value as per AVAX protocol)
|
|
272
|
+
// Use threshold=1 for change outputs (matching Flare protocol behavior)
|
|
273
|
+
const sortedAddresses = [...this.transaction._fromAddresses].sort((a, b) => Buffer.compare(a, b));
|
|
274
|
+
const outputOwners = new flarejs_1.OutputOwners(new flarejs_1.BigIntPr(this.transaction._locktime), new flarejs_1.Int(changeOutputThreshold), sortedAddresses.map((addr) => new flarejs_1.Address(addr)));
|
|
275
|
+
const transferOutput = new flarejs_1.TransferOutput(new flarejs_1.BigIntPr(changeAmount), outputOwners);
|
|
276
|
+
const changeOutput = new flarejs_1.TransferableOutput(new flarejs_1.Id(assetIdBytes), transferOutput);
|
|
277
|
+
changeOutputs.push(changeOutput);
|
|
278
|
+
}
|
|
279
|
+
// Sort inputs lexicographically by txid (Avalanche protocol requirement)
|
|
280
|
+
const sortedInputsWithCredentials = inputs
|
|
281
|
+
.map((input, i) => ({ input, credential: credentials[i] }))
|
|
282
|
+
.sort((a, b) => {
|
|
283
|
+
const aTxId = Buffer.from(a.input.utxoID.txID.toBytes());
|
|
284
|
+
const bTxId = Buffer.from(b.input.utxoID.txID.toBytes());
|
|
285
|
+
return Buffer.compare(aTxId, bTxId);
|
|
286
|
+
});
|
|
287
|
+
return {
|
|
288
|
+
inputs: sortedInputsWithCredentials.map((x) => x.input),
|
|
289
|
+
changeOutputs,
|
|
290
|
+
credentials: sortedInputsWithCredentials.map((x) => x.credential),
|
|
291
|
+
totalAmount,
|
|
292
|
+
};
|
|
293
|
+
}
|
|
93
294
|
/**
|
|
94
295
|
* Create the ExportedOutputs where the recipient address are the sender.
|
|
95
296
|
* Later an importTx should complete the operations signing with the same keys.
|
|
@@ -97,32 +298,33 @@ class ExportInPTxBuilder extends atomicTransactionBuilder_1.AtomicTransactionBui
|
|
|
97
298
|
*/
|
|
98
299
|
exportedOutputs() {
|
|
99
300
|
const assetIdBytes = new Uint8Array(Buffer.from(this.transaction._assetId, 'hex'));
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
301
|
+
// Create OutputOwners with sorted addresses
|
|
302
|
+
const sortedAddresses = [...this.transaction._fromAddresses].sort((a, b) => Buffer.compare(a, b));
|
|
303
|
+
const outputOwners = new flarejs_1.OutputOwners(new flarejs_1.BigIntPr(this.transaction._locktime), new flarejs_1.Int(this.transaction._threshold), sortedAddresses.map((addr) => new flarejs_1.Address(addr)));
|
|
103
304
|
const output = new flarejs_1.TransferOutput(new flarejs_1.BigIntPr(this._amount), outputOwners);
|
|
104
305
|
return [new flarejs_1.TransferableOutput(new flarejs_1.Id(assetIdBytes), output)];
|
|
105
306
|
}
|
|
106
307
|
/**
|
|
107
308
|
* Recover UTXOs from inputs
|
|
108
|
-
* @param inputs Array of
|
|
309
|
+
* @param inputs Array of TransferableInput
|
|
109
310
|
* @returns Array of decoded UTXO objects
|
|
110
311
|
*/
|
|
111
312
|
recoverUtxos(inputs) {
|
|
112
313
|
return inputs.map((input) => {
|
|
113
|
-
const
|
|
314
|
+
const utxoId = input.utxoID;
|
|
315
|
+
// Get the threshold from the input's sigIndices length
|
|
316
|
+
const transferInput = input.input;
|
|
317
|
+
const inputThreshold = transferInput.sigIndicies().length;
|
|
114
318
|
return {
|
|
115
319
|
outputID: iface_1.SECP256K1_Transfer_Output,
|
|
116
|
-
amount: input.amount.toString(),
|
|
117
|
-
txid: utils_1.default.cb58Encode(Buffer.from(
|
|
118
|
-
outputidx:
|
|
119
|
-
threshold: this.transaction._threshold,
|
|
120
|
-
addresses:
|
|
121
|
-
utils_1.default.addressToString(this.transaction._network.hrp, this.transaction._network.alias, Buffer.from(input.address.toBytes())),
|
|
122
|
-
],
|
|
320
|
+
amount: input.amount().toString(),
|
|
321
|
+
txid: utils_1.default.cb58Encode(Buffer.from(utxoId.txID.toBytes())),
|
|
322
|
+
outputidx: utxoId.outputIdx.value().toString(),
|
|
323
|
+
threshold: inputThreshold || this.transaction._threshold,
|
|
324
|
+
addresses: this.transaction._fromAddresses.map((addr) => utils_1.default.addressToString(this.transaction._network.hrp, this.transaction._network.alias, Buffer.from(addr))),
|
|
123
325
|
};
|
|
124
326
|
});
|
|
125
327
|
}
|
|
126
328
|
}
|
|
127
329
|
exports.ExportInPTxBuilder = ExportInPTxBuilder;
|
|
128
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ExportInPTxBuilder.js","sourceRoot":"","sources":["../../../src/lib/ExportInPTxBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { BuildTransactionError, NotSupported, TransactionType } from '@bitgo-beta/sdk-core';\nimport { AtomicTransactionBuilder } from './atomicTransactionBuilder';\nimport {\n  evmSerial,\n  UnsignedTx,\n  BigIntPr,\n  Int,\n  Id,\n  TransferableOutput,\n  Address,\n  utils as FlareUtils,\n  TransferOutput,\n  OutputOwners,\n} from '@flarenetwork/flarejs';\nimport utils from './utils';\nimport { DecodedUtxoObj, SECP256K1_Transfer_Output, FlareTransactionType, Tx } from './iface';\n\nexport class ExportInPTxBuilder extends AtomicTransactionBuilder {\n  private _amount: bigint;\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._externalChainId = utils.cb58Decode(this.transaction._network.cChainBlockchainID);\n  }\n\n  protected get transactionType(): TransactionType {\n    return TransactionType.Export;\n  }\n\n  /**\n   * Amount is a bigint that specifies the quantity of the asset that this output owns. Must be positive.\n   * @param {bigint | string} amount The withdrawal amount\n   */\n  amount(value: bigint | string): this {\n    const valueBigInt = typeof value === 'string' ? BigInt(value) : value;\n    this.validateAmount(valueBigInt);\n    this._amount = valueBigInt;\n    return this;\n  }\n\n  initBuilder(tx: Tx): this {\n    const baseTx = tx as evmSerial.ExportTx;\n    if (!this.verifyTxType(baseTx._type)) {\n      throw new NotSupported('Transaction cannot be parsed or has an unsupported transaction type');\n    }\n\n    // The exportedOutputs is a TransferableOutput array.\n    // It's expected to have only one output with the addresses of the sender.\n    const outputs = baseTx.exportedOutputs;\n    if (outputs.length !== 1) {\n      throw new BuildTransactionError('Transaction can have one external output');\n    }\n\n    const output = outputs[0];\n    const outputTransfer = output.output as TransferOutput;\n    const assetId = output.assetId.toBytes();\n    if (Buffer.compare(assetId, Buffer.from(this.transaction._assetId)) !== 0) {\n      throw new Error('The Asset ID of the output does not match the transaction');\n    }\n\n    // Set locktime to 0 since it's not used in EVM outputs\n    this.transaction._locktime = BigInt(0);\n\n    // Set threshold to 1 since EVM outputs only have one address\n    this.transaction._threshold = 1;\n\n    // Convert output address to buffer and set as fromAddress\n    const outputOwners = outputTransfer as unknown as OutputOwners;\n    this.transaction._fromAddresses = outputOwners.addrs.map((addr) => Buffer.from(addr.toBytes()));\n\n    // Set external chain ID from the destination chain\n    this._externalChainId = Buffer.from(baseTx.destinationChain.toString());\n\n    // Set amount from output\n    this._amount = outputTransfer.amount();\n\n    // Recover UTXOs from inputs\n    this.transaction._utxos = this.recoverUtxos(baseTx.ins);\n    return this;\n  }\n\n  static verifyTxType(txnType: string): boolean {\n    return txnType === FlareTransactionType.PvmExportTx;\n  }\n\n  verifyTxType(txnType: string): boolean {\n    return ExportInPTxBuilder.verifyTxType(txnType);\n  }\n\n  /**\n   * Build the export transaction\n   * @protected\n   */\n  protected buildFlareTransaction(): void {\n    // if tx has credentials, tx shouldn't change\n    if (this.transaction.hasCredentials) return;\n\n    const { inputs, credentials } = this.createInputOutput(this._amount + BigInt(this.transaction.fee.fee));\n\n    // Convert TransferableInputs to EVM Inputs\n    const transferableInputs: evmSerial.Input[] = inputs.map((input) => {\n      const assetIdBytes = input.assetId.toBytes();\n      const inputOwners = input as unknown as OutputOwners;\n      return new evmSerial.Input(\n        inputOwners.addrs[0],\n        new BigIntPr(input.amount()),\n        new Id(assetIdBytes),\n        new BigIntPr(BigInt(0)) // nonce is 0 for non-EVM inputs\n      );\n    });\n\n    // Create the export transaction\n    const exportTx = new evmSerial.ExportTx(\n      new Int(this.transaction._networkID),\n      new Id(new Uint8Array(Buffer.from(this.transaction._blockchainID, 'hex'))),\n      new Id(new Uint8Array(this._externalChainId)),\n      transferableInputs,\n      this.exportedOutputs()\n    );\n\n    // Create unsigned transaction with proper address maps\n    const addressMap = new FlareUtils.AddressMap([[new Address(this.transaction._fromAddresses[0]), 0]]);\n    const addressMaps = new FlareUtils.AddressMaps([addressMap]);\n\n    const unsignedTx = new UnsignedTx(\n      exportTx,\n      [], // Empty UTXOs array, will be filled during processing\n      addressMaps,\n      credentials\n    );\n\n    this.transaction.setTransaction(unsignedTx);\n  }\n\n  /**\n   * Create the ExportedOutputs where the recipient address are the sender.\n   * Later an importTx should complete the operations signing with the same keys.\n   * @protected\n   */\n  protected exportedOutputs(): TransferableOutput[] {\n    const assetIdBytes = new Uint8Array(Buffer.from(this.transaction._assetId, 'hex'));\n    const outputOwners = new OutputOwners(\n      new BigIntPr(BigInt(0)), // locktime\n      new Int(1), // threshold\n      [new Address(this.transaction._fromAddresses[0])]\n    );\n\n    const output = new TransferOutput(new BigIntPr(this._amount), outputOwners);\n\n    return [new TransferableOutput(new Id(assetIdBytes), output)];\n  }\n\n  /**\n   * Recover UTXOs from inputs\n   * @param inputs Array of inputs\n   * @returns Array of decoded UTXO objects\n   */\n  private recoverUtxos(inputs: evmSerial.Input[]): DecodedUtxoObj[] {\n    return inputs.map((input) => {\n      const txid = Buffer.from(input.assetId.toBytes()).toString('hex');\n      return {\n        outputID: SECP256K1_Transfer_Output,\n        amount: input.amount.toString(),\n        txid: utils.cb58Encode(Buffer.from(txid, 'hex')),\n        outputidx: '0', // Since EVM inputs don't have output indices\n        threshold: this.transaction._threshold,\n        addresses: [\n          utils.addressToString(\n            this.transaction._network.hrp,\n            this.transaction._network.alias,\n            Buffer.from(input.address.toBytes())\n          ),\n        ],\n      };\n    });\n  }\n}\n"]}
|
|
330
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ExportInPTxBuilder.js","sourceRoot":"","sources":["../../../src/lib/ExportInPTxBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { BuildTransactionError, NotSupported, TransactionType } from '@bitgo-beta/sdk-core';\nimport { AtomicTransactionBuilder } from './atomicTransactionBuilder';\nimport {\n  pvmSerial,\n  avaxSerial,\n  UnsignedTx,\n  BigIntPr,\n  Int,\n  Id,\n  TransferableInput,\n  TransferableOutput,\n  TransferInput,\n  Address,\n  utils as FlareUtils,\n  TransferOutput,\n  OutputOwners,\n  Credential,\n  Bytes,\n} from '@flarenetwork/flarejs';\nimport utils from './utils';\nimport { DecodedUtxoObj, SECP256K1_Transfer_Output, FlareTransactionType, Tx } from './iface';\n\nexport class ExportInPTxBuilder extends AtomicTransactionBuilder {\n  private _amount: bigint;\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    // For Export FROM P-chain:\n    // - external chain (destination) is C-chain\n    // - blockchain ID (source) is P-chain\n    this._externalChainId = utils.cb58Decode(this.transaction._network.cChainBlockchainID);\n    // P-chain blockchain ID (from network config - decode from cb58 to hex)\n    this.transaction._blockchainID = Buffer.from(utils.cb58Decode(this.transaction._network.blockchainID)).toString(\n      'hex'\n    );\n  }\n\n  protected get transactionType(): TransactionType {\n    return TransactionType.Export;\n  }\n\n  /**\n   * Amount is a bigint that specifies the quantity of the asset that this output owns. Must be positive.\n   * @param {bigint | string} amount The withdrawal amount\n   */\n  amount(value: bigint | string): this {\n    const valueBigInt = typeof value === 'string' ? BigInt(value) : value;\n    this.validateAmount(valueBigInt);\n    this._amount = valueBigInt;\n    return this;\n  }\n\n  initBuilder(tx: Tx, rawBytes?: Buffer): this {\n    const exportTx = tx as pvmSerial.ExportTx;\n\n    if (!this.verifyTxType(exportTx._type)) {\n      throw new NotSupported('Transaction cannot be parsed or has an unsupported transaction type');\n    }\n\n    // The exportedOutputs is a TransferableOutput array.\n    // It's expected to have only one output with the addresses of the sender.\n    const outputs = exportTx.outs;\n    if (outputs.length !== 1) {\n      throw new BuildTransactionError('Transaction can have one external output');\n    }\n\n    const output = outputs[0];\n    const outputTransfer = output.output as TransferOutput;\n    const assetId = output.assetId.toBytes();\n    if (Buffer.compare(Buffer.from(assetId), Buffer.from(this.transaction._assetId, 'hex')) !== 0) {\n      throw new Error('The Asset ID of the output does not match the transaction');\n    }\n\n    const outputOwners = outputTransfer.outputOwners;\n\n    // Set locktime from output\n    this.transaction._locktime = outputOwners.locktime.value();\n\n    // Set threshold from output\n    this.transaction._threshold = outputOwners.threshold.value();\n\n    // Convert output addresses to buffers and set as fromAddresses\n    this.transaction._fromAddresses = outputOwners.addrs.map((addr) => Buffer.from(addr.toBytes()));\n\n    // Set external chain ID from the destination chain\n    this._externalChainId = Buffer.from(exportTx.destination.toBytes());\n\n    // Set amount from exported output\n    this._amount = outputTransfer.amount();\n\n    // Recover UTXOs from base tx inputs\n    this.transaction._utxos = this.recoverUtxos([...exportTx.baseTx.inputs]);\n\n    // Calculate and set fee from input/output difference\n    const totalInputAmount = exportTx.baseTx.inputs.reduce((sum, input) => sum + input.amount(), BigInt(0));\n    const changeOutputAmount = exportTx.baseTx.outputs.reduce((sum, out) => {\n      const transferOut = out.output as TransferOutput;\n      return sum + transferOut.amount();\n    }, BigInt(0));\n    const fee = totalInputAmount - changeOutputAmount - this._amount;\n    this.transaction._fee.fee = fee.toString();\n\n    // Extract credentials from raw bytes\n    let hasCredentials = false;\n    let credentials: Credential[] = [];\n\n    if (rawBytes) {\n      // Try standard extraction first\n      const result = utils.extractCredentialsFromRawBytes(rawBytes, exportTx, 'PVM');\n      hasCredentials = result.hasCredentials;\n      credentials = result.credentials;\n\n      // If extraction failed but raw bytes are longer, try parsing credentials at known offset\n      if ((!hasCredentials || credentials.length === 0) && rawBytes.length > 300) {\n        const codec = FlareUtils.getManagerForVM('PVM').getDefaultCodec();\n        const txBytesLength = exportTx.toBytes(codec).length;\n\n        if (rawBytes.length > txBytesLength) {\n          hasCredentials = true;\n          const credResult = utils.parseCredentialsAtOffset(rawBytes, txBytesLength);\n          if (credResult.length > 0) {\n            credentials = credResult;\n          }\n        }\n      }\n    }\n\n    // If we have parsed credentials with the correct number of credentials for the inputs,\n    // use them directly (preserves existing signatures)\n    const numInputs = exportTx.baseTx.inputs.length;\n    const useDirectCredentials = hasCredentials && credentials.length === numInputs;\n\n    // If there are credentials in raw bytes, store the original bytes to preserve exact format\n    if (rawBytes && hasCredentials) {\n      this.transaction._rawSignedBytes = rawBytes;\n    }\n\n    // Create proper UnsignedTx wrapper with credentials\n    const sortedAddresses = [...this.transaction._fromAddresses].sort((a, b) => Buffer.compare(a, b));\n\n    // Helper function to check if a signature is empty (contains no real signature data)\n    // A real ECDSA signature will never start with 45 bytes of zeros\n    const isSignatureEmpty = (sig: string): boolean => {\n      if (!sig) return true;\n      const cleanSig = utils.removeHexPrefix(sig);\n      if (cleanSig.length === 0) return true;\n      // Check if the first 90 hex chars (45 bytes) are all zeros\n      // Real signatures from secp256k1 will never have this pattern\n      const first90Chars = cleanSig.substring(0, 90);\n      return first90Chars === '0'.repeat(90) || first90Chars === '0'.repeat(first90Chars.length);\n    };\n\n    // Build txCredentials - either use direct credentials or reconstruct with embedded addresses\n    let txCredentials: Credential[];\n\n    if (useDirectCredentials) {\n      // Use the extracted credentials directly - they already have the correct signatures\n      // Just ensure empty slots have embedded addresses for signing identification\n      txCredentials = credentials;\n    } else {\n      // Reconstruct credentials from scratch with embedded addresses\n      txCredentials = exportTx.baseTx.inputs.map((input, idx) => {\n        const transferInput = input.input as TransferInput;\n        const inputThreshold = transferInput.sigIndicies().length || this.transaction._threshold;\n\n        // Get existing signatures from parsed credentials if available\n        const existingSigs: string[] = [];\n        if (idx < credentials.length) {\n          const existingCred = credentials[idx];\n          existingSigs.push(...existingCred.getSignatures());\n        }\n\n        // Create credential with correct number of slots, preserving existing signatures\n        // Empty slots get embedded addresses for slot identification\n        const sigSlots: ReturnType<typeof utils.createNewSig>[] = [];\n        for (let i = 0; i < inputThreshold; i++) {\n          const existingSig = i < existingSigs.length ? existingSigs[i] : null;\n\n          if (existingSig && !isSignatureEmpty(existingSig)) {\n            // Use existing non-empty signature (real signature from signing)\n            const sigHex = utils.removeHexPrefix(existingSig);\n            sigSlots.push(utils.createNewSig(sigHex));\n          } else {\n            // Empty slot - create with embedded address for slot identification\n            const addrHex = Buffer.from(sortedAddresses[i]).toString('hex');\n            sigSlots.push(utils.createEmptySigWithAddress(addrHex));\n          }\n        }\n        return new Credential(sigSlots);\n      });\n    }\n\n    // Create address maps for signing - one per input/credential\n    // Each address map contains all addresses mapped to their indices\n    const addressMaps = txCredentials.map(() => {\n      const addressMap = new FlareUtils.AddressMap();\n      sortedAddresses.forEach((addr, i) => {\n        addressMap.set(new Address(addr), i);\n      });\n      return addressMap;\n    });\n\n    // Always create a new UnsignedTx with properly structured credentials\n    const unsignedTx = new UnsignedTx(exportTx, [], new FlareUtils.AddressMaps(addressMaps), txCredentials);\n\n    this.transaction.setTransaction(unsignedTx);\n    return this;\n  }\n\n  static verifyTxType(txnType: string): boolean {\n    return txnType === FlareTransactionType.PvmExportTx;\n  }\n\n  verifyTxType(txnType: string): boolean {\n    return ExportInPTxBuilder.verifyTxType(txnType);\n  }\n\n  /**\n   * Build the export transaction for P-chain\n   * @protected\n   */\n  protected buildFlareTransaction(): void {\n    // if tx has credentials, tx shouldn't change\n    if (this.transaction.hasCredentials) return;\n\n    const { inputs, changeOutputs, credentials, totalAmount } = this.createExportInputs();\n\n    // Calculate fee from transaction fee settings\n    const fee = BigInt(this.transaction.fee.fee);\n    const targetAmount = this._amount + fee;\n\n    // Verify we have enough funds\n    if (totalAmount < targetAmount) {\n      throw new BuildTransactionError(`Insufficient funds: have ${totalAmount}, need ${targetAmount}`);\n    }\n\n    // Create the BaseTx for the P-chain export transaction\n    const baseTx = new avaxSerial.BaseTx(\n      new Int(this.transaction._networkID),\n      new Id(Buffer.from(this.transaction._blockchainID, 'hex')),\n      changeOutputs, // change outputs\n      inputs, // inputs\n      new Bytes(new Uint8Array(0)) // empty memo\n    );\n\n    // Create the P-chain export transaction using pvmSerial.ExportTx\n    const exportTx = new pvmSerial.ExportTx(\n      baseTx,\n      new Id(this._externalChainId), // destinationChain (C-chain)\n      this.exportedOutputs() // exportedOutputs\n    );\n\n    // Create address maps for signing - one per input/credential\n    const sortedAddresses = [...this.transaction._fromAddresses].sort((a, b) => Buffer.compare(a, b));\n    const addressMaps = credentials.map(() => {\n      const addressMap = new FlareUtils.AddressMap();\n      sortedAddresses.forEach((addr, i) => {\n        addressMap.set(new Address(addr), i);\n      });\n      return addressMap;\n    });\n\n    // Create unsigned transaction\n    const unsignedTx = new UnsignedTx(\n      exportTx,\n      [], // Empty UTXOs array\n      new FlareUtils.AddressMaps(addressMaps),\n      credentials\n    );\n\n    this.transaction.setTransaction(unsignedTx);\n  }\n\n  /**\n   * Create inputs from UTXOs for P-chain export\n   * Only selects enough UTXOs to cover the target amount (amount + fee)\n   * @returns inputs, change outputs, credentials, and total amount\n   */\n  protected createExportInputs(): {\n    inputs: TransferableInput[];\n    changeOutputs: TransferableOutput[];\n    credentials: Credential[];\n    totalAmount: bigint;\n  } {\n    const sender = [...this.transaction._fromAddresses];\n    if (this.recoverSigner) {\n      // switch first and last signer\n      const tmp = sender.pop();\n      sender.push(sender[0]);\n      if (tmp) {\n        sender[0] = tmp;\n      }\n    }\n\n    const fee = BigInt(this.transaction.fee.fee);\n    const targetAmount = this._amount + fee;\n\n    let totalAmount = BigInt(0);\n    const inputs: TransferableInput[] = [];\n    const credentials: Credential[] = [];\n\n    // Change output threshold is always 1 (matching Flare protocol behavior)\n    // This allows easier spending of change while maintaining security for export outputs\n    const changeOutputThreshold = 1;\n\n    // Only consume enough UTXOs to cover the target amount (in array order)\n    // Inputs will be sorted after selection\n    for (const utxo of this.transaction._utxos) {\n      // Stop if we already have enough\n      if (totalAmount >= targetAmount) {\n        break;\n      }\n\n      const amount = BigInt(utxo.amount);\n      totalAmount += amount;\n\n      // Use the UTXO's own threshold for signature indices\n      const utxoThreshold = utxo.threshold || this.transaction._threshold;\n\n      // Create signature indices for the UTXO's threshold\n      const sigIndices: number[] = [];\n      for (let i = 0; i < utxoThreshold; i++) {\n        sigIndices.push(i);\n      }\n\n      // Use fromNative to create TransferableInput\n      const txIdCb58 = utxo.txid; // Already cb58 encoded\n      const assetIdCb58 = utils.cb58Encode(Buffer.from(this.transaction._assetId, 'hex'));\n\n      const transferableInput = TransferableInput.fromNative(\n        txIdCb58,\n        Number(utxo.outputidx),\n        assetIdCb58,\n        amount,\n        sigIndices\n      );\n\n      inputs.push(transferableInput);\n\n      // Create credential with empty signatures that have embedded addresses for slot identification\n      // This allows the signing logic to determine which slot belongs to which address\n      const sortedAddrs = [...this.transaction._fromAddresses].sort((a, b) => Buffer.compare(a, b));\n      const emptySignatures = sigIndices.map((idx) => {\n        const addrHex = Buffer.from(sortedAddrs[idx]).toString('hex');\n        return utils.createEmptySigWithAddress(addrHex);\n      });\n      credentials.push(new Credential(emptySignatures));\n    }\n\n    // Create change output if there is remaining amount after export and fee\n    const changeOutputs: TransferableOutput[] = [];\n    const changeAmount = totalAmount - this._amount - fee;\n\n    if (changeAmount > BigInt(0)) {\n      const assetIdBytes = new Uint8Array(Buffer.from(this.transaction._assetId, 'hex'));\n\n      // Create OutputOwners with the P-chain addresses (sorted by byte value as per AVAX protocol)\n      // Use threshold=1 for change outputs (matching Flare protocol behavior)\n      const sortedAddresses = [...this.transaction._fromAddresses].sort((a, b) => Buffer.compare(a, b));\n      const outputOwners = new OutputOwners(\n        new BigIntPr(this.transaction._locktime),\n        new Int(changeOutputThreshold),\n        sortedAddresses.map((addr) => new Address(addr))\n      );\n\n      const transferOutput = new TransferOutput(new BigIntPr(changeAmount), outputOwners);\n      const changeOutput = new TransferableOutput(new Id(assetIdBytes), transferOutput);\n      changeOutputs.push(changeOutput);\n    }\n\n    // Sort inputs lexicographically by txid (Avalanche protocol requirement)\n    const sortedInputsWithCredentials = inputs\n      .map((input, i) => ({ input, credential: credentials[i] }))\n      .sort((a, b) => {\n        const aTxId = Buffer.from(a.input.utxoID.txID.toBytes());\n        const bTxId = Buffer.from(b.input.utxoID.txID.toBytes());\n        return Buffer.compare(aTxId, bTxId);\n      });\n\n    return {\n      inputs: sortedInputsWithCredentials.map((x) => x.input),\n      changeOutputs,\n      credentials: sortedInputsWithCredentials.map((x) => x.credential),\n      totalAmount,\n    };\n  }\n\n  /**\n   * Create the ExportedOutputs where the recipient address are the sender.\n   * Later an importTx should complete the operations signing with the same keys.\n   * @protected\n   */\n  protected exportedOutputs(): TransferableOutput[] {\n    const assetIdBytes = new Uint8Array(Buffer.from(this.transaction._assetId, 'hex'));\n\n    // Create OutputOwners with sorted addresses\n    const sortedAddresses = [...this.transaction._fromAddresses].sort((a, b) => Buffer.compare(a, b));\n    const outputOwners = new OutputOwners(\n      new BigIntPr(this.transaction._locktime),\n      new Int(this.transaction._threshold),\n      sortedAddresses.map((addr) => new Address(addr))\n    );\n\n    const output = new TransferOutput(new BigIntPr(this._amount), outputOwners);\n\n    return [new TransferableOutput(new Id(assetIdBytes), output)];\n  }\n\n  /**\n   * Recover UTXOs from inputs\n   * @param inputs Array of TransferableInput\n   * @returns Array of decoded UTXO objects\n   */\n  private recoverUtxos(inputs: TransferableInput[]): DecodedUtxoObj[] {\n    return inputs.map((input) => {\n      const utxoId = input.utxoID;\n      // Get the threshold from the input's sigIndices length\n      const transferInput = input.input as TransferInput;\n      const inputThreshold = transferInput.sigIndicies().length;\n      return {\n        outputID: SECP256K1_Transfer_Output,\n        amount: input.amount().toString(),\n        txid: utils.cb58Encode(Buffer.from(utxoId.txID.toBytes())),\n        outputidx: utxoId.outputIdx.value().toString(),\n        threshold: inputThreshold || this.transaction._threshold,\n        addresses: this.transaction._fromAddresses.map((addr) =>\n          utils.addressToString(this.transaction._network.hrp, this.transaction._network.alias, Buffer.from(addr))\n        ),\n      };\n    });\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,EACf,cAAc,EACf,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAEL,UAAU,EAMX,MAAM,uBAAuB,CAAC;AAC/B,OAAO,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAEhC,OAAO,EAAE,cAAc,EAAE,sBAAsB,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAC7E,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,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,EACf,cAAc,EACf,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAEL,UAAU,EAMX,MAAM,uBAAuB,CAAC;AAC/B,OAAO,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAEhC,OAAO,EAAE,cAAc,EAAE,sBAAsB,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAC7E,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAgEpC,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,IAAI,EAAE,OAAO,CAAC,cAAc,CAAC,CAAM;IAEnC,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;IAqE3C,iBAAiB,IAAI,MAAM;IAkB3B,MAAM,IAAI,MAAM;IAiBhB,cAAc,CAAC,EAAE,EAAE,EAAE,GAAG,IAAI;IAI5B,kBAAkB,CAAC,eAAe,EAAE,eAAe,GAAG,IAAI;IAO1D,IAAI,eAAe,IAAI,MAAM,CAE5B;IAED,IAAI,EAAE,IAAI,MAAM,CAGf;IAED,IAAI,aAAa,IAAI,MAAM,EAAE,CAE5B;IAED,IAAI,eAAe,IAAI,MAAM,EAAE,CAE9B;IAED,IAAI,GAAG,IAAI,cAAc,CAExB;IAED,IAAI,OAAO,IAAI,KAAK,EAAE,CAgBrB;IAED,IAAI,aAAa,IAAI,KAAK,EAAE,CAI3B;IAED,kBAAkB,IAAI,sBAAsB;CA0B7C"}
|