@bitgo-beta/sdk-coin-flrp 1.0.1-beta.4 → 1.0.1-beta.400
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 +75 -61
- package/dist/src/flrp.d.ts.map +1 -1
- package/dist/src/flrp.js +276 -134
- 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 +174 -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 +175 -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 +208 -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 +112 -35
- package/dist/src/lib/atomicTransactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/atomicTransactionBuilder.js +244 -34
- package/dist/src/lib/iface.d.ts +81 -22
- 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 +7 -0
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +16 -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 +15 -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 +75 -0
- package/dist/src/lib/transaction.d.ts.map +1 -0
- package/dist/src/lib/transaction.js +460 -0
- 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 +57 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilderFactory.js +148 -0
- package/dist/src/lib/utils.d.ts +138 -102
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +359 -245
- 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 +490 -20
- 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 +843 -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.d.ts +2 -0
- package/dist/test/unit/lib/utils.d.ts.map +1 -0
- package/dist/test/unit/lib/utils.js +761 -0
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +18 -12
- 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 -164
- 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/smoke.d.ts +0 -2
- package/dist/test/unit/smoke.d.ts.map +0 -1
- package/dist/test/unit/smoke.js +0 -9
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import { FlareNetwork, BaseCoin as CoinConfig } from '@bitgo-beta/statics';
|
|
2
|
+
import { BaseKey, BaseTransaction, Entry, TransactionType } from '@bitgo-beta/sdk-core';
|
|
3
|
+
import { Credential, Context } from '@flarenetwork/flarejs';
|
|
4
|
+
import { Buffer } from 'buffer';
|
|
5
|
+
import { TransactionExplanation, Tx, TxData, FlrpTransactionFee, DecodedUtxoObj } from './iface';
|
|
6
|
+
import { FlrpFeeState } from '@bitgo/public-types';
|
|
7
|
+
import { KeyPair } from './keyPair';
|
|
8
|
+
export declare class Transaction extends BaseTransaction {
|
|
9
|
+
protected _flareTransaction: Tx;
|
|
10
|
+
_type: TransactionType;
|
|
11
|
+
_network: FlareNetwork;
|
|
12
|
+
_networkID: number;
|
|
13
|
+
_assetId: string;
|
|
14
|
+
_blockchainID: string;
|
|
15
|
+
_nodeID: string;
|
|
16
|
+
_startTime: bigint;
|
|
17
|
+
_endTime: bigint;
|
|
18
|
+
_stakeAmount: bigint;
|
|
19
|
+
_threshold: number;
|
|
20
|
+
_locktime: bigint;
|
|
21
|
+
_fromAddresses: Uint8Array[];
|
|
22
|
+
_to: Uint8Array[];
|
|
23
|
+
_rewardAddresses: Uint8Array[];
|
|
24
|
+
_utxos: DecodedUtxoObj[];
|
|
25
|
+
_context: Context.Context;
|
|
26
|
+
_fee: FlrpTransactionFee;
|
|
27
|
+
_feeState: FlrpFeeState;
|
|
28
|
+
_amount: bigint;
|
|
29
|
+
_rawSignedBytes: Buffer | undefined;
|
|
30
|
+
constructor(coinConfig: Readonly<CoinConfig>);
|
|
31
|
+
get signature(): string[];
|
|
32
|
+
get credentials(): Credential[];
|
|
33
|
+
get hasCredentials(): boolean;
|
|
34
|
+
/** @inheritdoc */
|
|
35
|
+
canSign({ key }: BaseKey): boolean;
|
|
36
|
+
sign(keyPair: KeyPair): Promise<void>;
|
|
37
|
+
toBroadcastFormat(): string;
|
|
38
|
+
toJson(): TxData;
|
|
39
|
+
/**
|
|
40
|
+
* Get the source chain id or undefined if it's not a cross chain transfer.
|
|
41
|
+
*/
|
|
42
|
+
get sourceChain(): string | undefined;
|
|
43
|
+
/**
|
|
44
|
+
* Get the destination chain id or undefined if it's not a cross chain transfer.
|
|
45
|
+
*/
|
|
46
|
+
get destinationChain(): string | undefined;
|
|
47
|
+
/**
|
|
48
|
+
* Convert a blockchainId buffer to string and return P or C alias if it matches any of those chains.
|
|
49
|
+
* @param {Buffer} blockchainIDBuffer
|
|
50
|
+
* @return {string} blockchainID or alias if exists.
|
|
51
|
+
* @private
|
|
52
|
+
*/
|
|
53
|
+
blockchainIDtoAlias(blockchainIDBuffer: Buffer): string;
|
|
54
|
+
setTransaction(tx: Tx): void;
|
|
55
|
+
/**
|
|
56
|
+
* Get the underlying Flare transaction
|
|
57
|
+
* @returns The Flare transaction object
|
|
58
|
+
*/
|
|
59
|
+
getFlareTransaction(): Tx;
|
|
60
|
+
setTransactionType(transactionType: TransactionType): void;
|
|
61
|
+
get signablePayload(): Buffer;
|
|
62
|
+
get id(): string;
|
|
63
|
+
get fromAddresses(): string[];
|
|
64
|
+
get rewardAddresses(): string[];
|
|
65
|
+
get fee(): FlrpTransactionFee;
|
|
66
|
+
/**
|
|
67
|
+
* Check if this transaction is for C-chain (EVM transactions)
|
|
68
|
+
*/
|
|
69
|
+
get isTransactionForCChain(): boolean;
|
|
70
|
+
get outputs(): Entry[];
|
|
71
|
+
get changeOutputs(): Entry[];
|
|
72
|
+
get inputs(): Entry[];
|
|
73
|
+
explainTransaction(): TransactionExplanation;
|
|
74
|
+
}
|
|
75
|
+
//# sourceMappingURL=transaction.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transaction.d.ts","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC3E,OAAO,EACL,OAAO,EACP,eAAe,EACf,KAAK,EAGL,eAAe,EAChB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAEL,UAAU,EAOV,OAAO,EACR,MAAM,uBAAuB,CAAC;AAC/B,OAAO,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAEhC,OAAO,EACL,sBAAsB,EACtB,EAAE,EACF,MAAM,EAGN,kBAAkB,EAClB,cAAc,EACf,MAAM,SAAS,CAAC;AACjB,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AA+DpC,qBAAa,WAAY,SAAQ,eAAe;IAC9C,SAAS,CAAC,iBAAiB,EAAE,EAAE,CAAC;IACzB,KAAK,EAAE,eAAe,CAAC;IACvB,QAAQ,EAAE,YAAY,CAAC;IACvB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,YAAY,EAAE,MAAM,CAAC;IACrB,UAAU,SAAK;IACf,SAAS,SAAa;IACtB,cAAc,EAAE,UAAU,EAAE,CAAM;IAClC,GAAG,EAAE,UAAU,EAAE,CAAM;IACvB,gBAAgB,EAAE,UAAU,EAAE,CAAM;IACpC,MAAM,EAAE,cAAc,EAAE,CAAM;IAC9B,QAAQ,EAAE,OAAO,CAAC,OAAO,CAAC;IAC1B,IAAI,EAAE,kBAAkB,CAAgB;IACxC,SAAS,EAAE,YAAY,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,eAAe,EAAE,MAAM,GAAG,SAAS,CAAC;gBAE/B,UAAU,EAAE,QAAQ,CAAC,UAAU,CAAC;IAS5C,IAAI,SAAS,IAAI,MAAM,EAAE,CAKxB;IAED,IAAI,WAAW,IAAI,UAAU,EAAE,CAE9B;IAED,IAAI,cAAc,IAAI,OAAO,CAE5B;IAED,kBAAkB;IAClB,OAAO,CAAC,EAAE,GAAG,EAAE,EAAE,OAAO,GAAG,OAAO;IAI5B,IAAI,CAAC,OAAO,EAAE,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC;IAsF3C,iBAAiB,IAAI,MAAM;IAkB3B,MAAM,IAAI,MAAM;IAmBhB;;OAEG;IACH,IAAI,WAAW,IAAI,MAAM,GAAG,SAAS,CA6BpC;IAED;;OAEG;IACH,IAAI,gBAAgB,IAAI,MAAM,GAAG,SAAS,CA6BzC;IAED;;;;;OAKG;IACH,mBAAmB,CAAC,kBAAkB,EAAE,MAAM,GAAG,MAAM;IAWvD,cAAc,CAAC,EAAE,EAAE,EAAE,GAAG,IAAI;IAI5B;;;OAGG;IACH,mBAAmB,IAAI,EAAE;IAIzB,kBAAkB,CAAC,eAAe,EAAE,eAAe,GAAG,IAAI;IAO1D,IAAI,eAAe,IAAI,MAAM,CAE5B;IAED,IAAI,EAAE,IAAI,MAAM,CAKf;IAED,IAAI,aAAa,IAAI,MAAM,EAAE,CAE5B;IAED,IAAI,eAAe,IAAI,MAAM,EAAE,CAE9B;IAED,IAAI,GAAG,IAAI,kBAAkB,CAE5B;IAED;;OAEG;IACH,IAAI,sBAAsB,IAAI,OAAO,CAIpC;IAED,IAAI,OAAO,IAAI,KAAK,EAAE,CAwCrB;IAED,IAAI,aAAa,IAAI,KAAK,EAAE,CAmB3B;IAED,IAAI,MAAM,IAAI,KAAK,EAAE,CAsDpB;IAED,kBAAkB,IAAI,sBAAsB;CA0B7C"}
|
|
@@ -0,0 +1,460 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.Transaction = void 0;
|
|
7
|
+
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
8
|
+
const flarejs_1 = require("@flarenetwork/flarejs");
|
|
9
|
+
const buffer_1 = require("buffer");
|
|
10
|
+
const crypto_1 = require("crypto");
|
|
11
|
+
const iface_1 = require("./iface");
|
|
12
|
+
const utils_1 = __importDefault(require("./utils"));
|
|
13
|
+
/**
|
|
14
|
+
* Checks if a signature is empty (first 90 hex chars are zeros)
|
|
15
|
+
* @param signature
|
|
16
|
+
* @returns {boolean}
|
|
17
|
+
*/
|
|
18
|
+
function isEmptySignature(signature) {
|
|
19
|
+
return !!signature && utils_1.default.removeHexPrefix(signature).startsWith(''.padStart(90, '0'));
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Checks if an empty signature has an embedded address (non-zero bytes after position 90)
|
|
23
|
+
* @param signature Hex string of the signature
|
|
24
|
+
*/
|
|
25
|
+
function hasEmbeddedAddress(signature) {
|
|
26
|
+
if (!isEmptySignature(signature))
|
|
27
|
+
return false;
|
|
28
|
+
const cleanSig = utils_1.default.removeHexPrefix(signature);
|
|
29
|
+
if (cleanSig.length < 130)
|
|
30
|
+
return false;
|
|
31
|
+
const embeddedPart = cleanSig.substring(90, 130);
|
|
32
|
+
return embeddedPart !== '0'.repeat(40);
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Generates a function to check if a signature slot matches a given address.
|
|
36
|
+
* If signatures have embedded addresses, it matches by address.
|
|
37
|
+
* Otherwise, it just finds empty slots.
|
|
38
|
+
* @param signatures Array of signature hex strings
|
|
39
|
+
*/
|
|
40
|
+
function generateSelectorSignature(signatures) {
|
|
41
|
+
// Check if any empty signature has an embedded address
|
|
42
|
+
const hasEmbeddedAddresses = signatures.some((sig) => isEmptySignature(sig) && hasEmbeddedAddress(sig));
|
|
43
|
+
if (hasEmbeddedAddresses) {
|
|
44
|
+
// Look for address embedded in the empty signature (after position 90)
|
|
45
|
+
return function (sig, address) {
|
|
46
|
+
try {
|
|
47
|
+
if (!isEmptySignature(sig)) {
|
|
48
|
+
return false;
|
|
49
|
+
}
|
|
50
|
+
const cleanSig = utils_1.default.removeHexPrefix(sig);
|
|
51
|
+
const embeddedAddr = cleanSig.substring(90, 130).toLowerCase();
|
|
52
|
+
return embeddedAddr === address.toLowerCase();
|
|
53
|
+
}
|
|
54
|
+
catch (e) {
|
|
55
|
+
return false;
|
|
56
|
+
}
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
else {
|
|
60
|
+
// Look for any empty slot (no embedded addresses)
|
|
61
|
+
return function (sig, address) {
|
|
62
|
+
return isEmptySignature(sig);
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
class Transaction extends sdk_core_1.BaseTransaction {
|
|
67
|
+
constructor(coinConfig) {
|
|
68
|
+
super(coinConfig);
|
|
69
|
+
this._threshold = 2;
|
|
70
|
+
this._locktime = BigInt(0);
|
|
71
|
+
this._fromAddresses = [];
|
|
72
|
+
this._to = [];
|
|
73
|
+
this._rewardAddresses = [];
|
|
74
|
+
this._utxos = [];
|
|
75
|
+
this._fee = { fee: '0' };
|
|
76
|
+
this._network = coinConfig.network;
|
|
77
|
+
// Decode cb58-encoded asset ID to hex for use in transaction serialization
|
|
78
|
+
this._assetId = utils_1.default.cb58Decode(this._network.assetId).toString('hex');
|
|
79
|
+
this._blockchainID = this._network.blockchainID;
|
|
80
|
+
this._networkID = this._network.networkID;
|
|
81
|
+
}
|
|
82
|
+
get signature() {
|
|
83
|
+
if (!this.hasCredentials) {
|
|
84
|
+
return [];
|
|
85
|
+
}
|
|
86
|
+
return this.credentials[0].getSignatures().filter((s) => !isEmptySignature(s));
|
|
87
|
+
}
|
|
88
|
+
get credentials() {
|
|
89
|
+
return this._flareTransaction?.credentials;
|
|
90
|
+
}
|
|
91
|
+
get hasCredentials() {
|
|
92
|
+
return this.credentials !== undefined && this.credentials.length > 0;
|
|
93
|
+
}
|
|
94
|
+
/** @inheritdoc */
|
|
95
|
+
canSign({ key }) {
|
|
96
|
+
return true;
|
|
97
|
+
}
|
|
98
|
+
async sign(keyPair) {
|
|
99
|
+
const prv = keyPair.getPrivateKey();
|
|
100
|
+
if (!prv) {
|
|
101
|
+
throw new sdk_core_1.SigningError('Missing private key');
|
|
102
|
+
}
|
|
103
|
+
if (!this._flareTransaction) {
|
|
104
|
+
throw new sdk_core_1.InvalidTransactionError('empty transaction to sign');
|
|
105
|
+
}
|
|
106
|
+
if (!this.hasCredentials) {
|
|
107
|
+
throw new sdk_core_1.InvalidTransactionError('empty credentials to sign');
|
|
108
|
+
}
|
|
109
|
+
const unsignedTx = this._flareTransaction;
|
|
110
|
+
const unsignedBytes = unsignedTx.toBytes();
|
|
111
|
+
const publicKey = flarejs_1.secp256k1.getPublicKey(prv);
|
|
112
|
+
// Derive both EVM and P-chain addresses from the public key
|
|
113
|
+
const evmAddressHex = new flarejs_1.Address(flarejs_1.secp256k1.publicKeyToEthAddress(publicKey)).toHex();
|
|
114
|
+
// P-chain address derivation: ripemd160(sha256(publicKey))
|
|
115
|
+
const sha256Hash = (0, crypto_1.createHash)('sha256').update(buffer_1.Buffer.from(publicKey)).digest();
|
|
116
|
+
const pChainAddressBuffer = (0, crypto_1.createHash)('ripemd160').update(sha256Hash).digest();
|
|
117
|
+
const pChainAddressHex = pChainAddressBuffer.toString('hex');
|
|
118
|
+
const addressMap = unsignedTx.getAddresses();
|
|
119
|
+
// Check for both EVM and P-chain address formats
|
|
120
|
+
const hasMatchingAddress = addressMap.some((addr) => {
|
|
121
|
+
const addrHex = buffer_1.Buffer.from(addr).toString('hex').toLowerCase();
|
|
122
|
+
return (addrHex === utils_1.default.removeHexPrefix(evmAddressHex).toLowerCase() || addrHex === pChainAddressHex.toLowerCase());
|
|
123
|
+
});
|
|
124
|
+
const signature = await flarejs_1.secp256k1.sign(unsignedBytes, prv);
|
|
125
|
+
let signatureSet = false;
|
|
126
|
+
if (hasMatchingAddress) {
|
|
127
|
+
// Use address-based slot matching (like AVAX-P)
|
|
128
|
+
let checkSign = undefined;
|
|
129
|
+
for (const credential of unsignedTx.credentials) {
|
|
130
|
+
const signatures = credential.getSignatures();
|
|
131
|
+
if (checkSign === undefined) {
|
|
132
|
+
checkSign = generateSelectorSignature(signatures);
|
|
133
|
+
}
|
|
134
|
+
// Find the slot that matches this address
|
|
135
|
+
for (let i = 0; i < signatures.length; i++) {
|
|
136
|
+
const sig = signatures[i];
|
|
137
|
+
// Try matching with P-chain address first, then EVM address
|
|
138
|
+
if (checkSign(sig, pChainAddressHex) || checkSign(sig, utils_1.default.removeHexPrefix(evmAddressHex).toLowerCase())) {
|
|
139
|
+
credential.setSignature(i, signature);
|
|
140
|
+
signatureSet = true;
|
|
141
|
+
// Clear raw signed bytes since we've modified the transaction
|
|
142
|
+
this._rawSignedBytes = undefined;
|
|
143
|
+
break; // Break inner loop, but continue to sign other credentials
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
// Don't break outer loop - continue signing ALL credentials that have a matching slot
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
// Fallback: If address-based matching didn't work (e.g., ImportInC loaded from unsigned tx
|
|
150
|
+
// where P-chain addresses aren't in addressMaps), sign ALL empty slots across ALL credentials.
|
|
151
|
+
// This handles multisig where each UTXO needs a credential signed by the same key.
|
|
152
|
+
if (!signatureSet) {
|
|
153
|
+
for (const credential of unsignedTx.credentials) {
|
|
154
|
+
const signatures = credential.getSignatures();
|
|
155
|
+
for (let i = 0; i < signatures.length; i++) {
|
|
156
|
+
if (isEmptySignature(signatures[i])) {
|
|
157
|
+
credential.setSignature(i, signature);
|
|
158
|
+
signatureSet = true;
|
|
159
|
+
this._rawSignedBytes = undefined;
|
|
160
|
+
break; // Break inner loop, but continue to sign other credentials
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
// Don't break outer loop - continue signing ALL credentials with empty slots
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
if (!signatureSet) {
|
|
167
|
+
throw new sdk_core_1.SigningError('No matching signature slot found for this private key');
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
toBroadcastFormat() {
|
|
171
|
+
if (!this._flareTransaction) {
|
|
172
|
+
throw new sdk_core_1.InvalidTransactionError('Empty transaction data');
|
|
173
|
+
}
|
|
174
|
+
// If we have the original raw signed bytes, use them directly to preserve exact format
|
|
175
|
+
if (this._rawSignedBytes) {
|
|
176
|
+
return flarejs_1.utils.bufferToHex(this._rawSignedBytes);
|
|
177
|
+
}
|
|
178
|
+
const unsignedTx = this._flareTransaction;
|
|
179
|
+
const signedTxBytes = unsignedTx.getSignedTx().toBytes();
|
|
180
|
+
// Both P-chain and C-chain transactions include checksum (matching avaxp behavior)
|
|
181
|
+
// avaxp P-chain: transaction.ts uses addChecksum() explicitly
|
|
182
|
+
// avaxp C-chain: deprecatedTransaction.ts uses Tx.toStringHex() which internally adds checksum
|
|
183
|
+
const rawTx = flarejs_1.utils.bufferToHex(utils_1.default.addChecksum(signedTxBytes));
|
|
184
|
+
return rawTx;
|
|
185
|
+
}
|
|
186
|
+
toJson() {
|
|
187
|
+
if (!this._flareTransaction) {
|
|
188
|
+
throw new sdk_core_1.InvalidTransactionError('Empty transaction data');
|
|
189
|
+
}
|
|
190
|
+
return {
|
|
191
|
+
id: this.id,
|
|
192
|
+
inputs: this.inputs,
|
|
193
|
+
fromAddresses: this.fromAddresses,
|
|
194
|
+
threshold: this._threshold,
|
|
195
|
+
locktime: this._locktime.toString(),
|
|
196
|
+
type: this.type,
|
|
197
|
+
signatures: this.signature,
|
|
198
|
+
outputs: this.outputs,
|
|
199
|
+
changeOutputs: this.changeOutputs,
|
|
200
|
+
sourceChain: this.sourceChain,
|
|
201
|
+
destinationChain: this.destinationChain,
|
|
202
|
+
};
|
|
203
|
+
}
|
|
204
|
+
/**
|
|
205
|
+
* Get the source chain id or undefined if it's not a cross chain transfer.
|
|
206
|
+
*/
|
|
207
|
+
get sourceChain() {
|
|
208
|
+
const tx = this._flareTransaction.getTx();
|
|
209
|
+
switch (this.type) {
|
|
210
|
+
case sdk_core_1.TransactionType.Import:
|
|
211
|
+
if (this.isTransactionForCChain) {
|
|
212
|
+
// C-chain Import: source is the chain we're importing FROM (P-chain)
|
|
213
|
+
const importTx = tx;
|
|
214
|
+
return this.blockchainIDtoAlias(buffer_1.Buffer.from(importTx.sourceChain.toBytes()));
|
|
215
|
+
}
|
|
216
|
+
else {
|
|
217
|
+
// P-chain Import: source is the chain we're importing FROM (C-chain)
|
|
218
|
+
const pvmImportTx = tx;
|
|
219
|
+
return this.blockchainIDtoAlias(buffer_1.Buffer.from(pvmImportTx.sourceChain.toBytes()));
|
|
220
|
+
}
|
|
221
|
+
case sdk_core_1.TransactionType.Export:
|
|
222
|
+
if (this.isTransactionForCChain) {
|
|
223
|
+
// C-chain Export: source is C-chain (the blockchain ID)
|
|
224
|
+
const exportTx = tx;
|
|
225
|
+
return this.blockchainIDtoAlias(buffer_1.Buffer.from(exportTx.blockchainId.toBytes()));
|
|
226
|
+
}
|
|
227
|
+
else {
|
|
228
|
+
// P-chain Export: source is P-chain (the blockchain ID from baseTx)
|
|
229
|
+
const pvmExportTx = tx;
|
|
230
|
+
return this.blockchainIDtoAlias(buffer_1.Buffer.from(pvmExportTx.baseTx.BlockchainId.toBytes()));
|
|
231
|
+
}
|
|
232
|
+
default:
|
|
233
|
+
return undefined;
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
/**
|
|
237
|
+
* Get the destination chain id or undefined if it's not a cross chain transfer.
|
|
238
|
+
*/
|
|
239
|
+
get destinationChain() {
|
|
240
|
+
const tx = this._flareTransaction.getTx();
|
|
241
|
+
switch (this.type) {
|
|
242
|
+
case sdk_core_1.TransactionType.Import:
|
|
243
|
+
if (this.isTransactionForCChain) {
|
|
244
|
+
// C-chain Import: destination is C-chain (the blockchain ID)
|
|
245
|
+
const importTx = tx;
|
|
246
|
+
return this.blockchainIDtoAlias(buffer_1.Buffer.from(importTx.blockchainId.toBytes()));
|
|
247
|
+
}
|
|
248
|
+
else {
|
|
249
|
+
// P-chain Import: destination is P-chain (the blockchain ID from baseTx)
|
|
250
|
+
const pvmImportTx = tx;
|
|
251
|
+
return this.blockchainIDtoAlias(buffer_1.Buffer.from(pvmImportTx.baseTx.BlockchainId.toBytes()));
|
|
252
|
+
}
|
|
253
|
+
case sdk_core_1.TransactionType.Export:
|
|
254
|
+
if (this.isTransactionForCChain) {
|
|
255
|
+
// C-chain Export: destination is P-chain (the destination chain)
|
|
256
|
+
const exportTx = tx;
|
|
257
|
+
return this.blockchainIDtoAlias(buffer_1.Buffer.from(exportTx.destinationChain.toBytes()));
|
|
258
|
+
}
|
|
259
|
+
else {
|
|
260
|
+
// P-chain Export: destination is C-chain (the destination chain)
|
|
261
|
+
const pvmExportTx = tx;
|
|
262
|
+
return this.blockchainIDtoAlias(buffer_1.Buffer.from(pvmExportTx.destination.toBytes()));
|
|
263
|
+
}
|
|
264
|
+
default:
|
|
265
|
+
return undefined;
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
/**
|
|
269
|
+
* Convert a blockchainId buffer to string and return P or C alias if it matches any of those chains.
|
|
270
|
+
* @param {Buffer} blockchainIDBuffer
|
|
271
|
+
* @return {string} blockchainID or alias if exists.
|
|
272
|
+
* @private
|
|
273
|
+
*/
|
|
274
|
+
blockchainIDtoAlias(blockchainIDBuffer) {
|
|
275
|
+
const blockchainId = utils_1.default.cb58Encode(blockchainIDBuffer);
|
|
276
|
+
if (blockchainId === this._network.cChainBlockchainID) {
|
|
277
|
+
return 'C';
|
|
278
|
+
}
|
|
279
|
+
if (blockchainId === this._network.blockchainID) {
|
|
280
|
+
return 'P';
|
|
281
|
+
}
|
|
282
|
+
return blockchainId;
|
|
283
|
+
}
|
|
284
|
+
setTransaction(tx) {
|
|
285
|
+
this._flareTransaction = tx;
|
|
286
|
+
}
|
|
287
|
+
/**
|
|
288
|
+
* Get the underlying Flare transaction
|
|
289
|
+
* @returns The Flare transaction object
|
|
290
|
+
*/
|
|
291
|
+
getFlareTransaction() {
|
|
292
|
+
return this._flareTransaction;
|
|
293
|
+
}
|
|
294
|
+
setTransactionType(transactionType) {
|
|
295
|
+
if (![sdk_core_1.TransactionType.AddPermissionlessValidator].includes(transactionType)) {
|
|
296
|
+
throw new Error(`Transaction type ${transactionType} is not supported`);
|
|
297
|
+
}
|
|
298
|
+
this._type = transactionType;
|
|
299
|
+
}
|
|
300
|
+
get signablePayload() {
|
|
301
|
+
return utils_1.default.sha256(this._flareTransaction.toBytes());
|
|
302
|
+
}
|
|
303
|
+
get id() {
|
|
304
|
+
const unsignedTx = this._flareTransaction;
|
|
305
|
+
const txBytes = unsignedTx.getSignedTx().toBytes();
|
|
306
|
+
const bufferArray = utils_1.default.sha256(txBytes);
|
|
307
|
+
return utils_1.default.cb58Encode(buffer_1.Buffer.from(bufferArray));
|
|
308
|
+
}
|
|
309
|
+
get fromAddresses() {
|
|
310
|
+
return this._fromAddresses.map((a) => flarejs_1.utils.format(this._network.alias, this._network.hrp, a));
|
|
311
|
+
}
|
|
312
|
+
get rewardAddresses() {
|
|
313
|
+
return this._rewardAddresses.map((a) => flarejs_1.utils.format(this._network.alias, this._network.hrp, a));
|
|
314
|
+
}
|
|
315
|
+
get fee() {
|
|
316
|
+
return this._fee;
|
|
317
|
+
}
|
|
318
|
+
/**
|
|
319
|
+
* Check if this transaction is for C-chain (EVM transactions)
|
|
320
|
+
*/
|
|
321
|
+
get isTransactionForCChain() {
|
|
322
|
+
const tx = this._flareTransaction.getTx();
|
|
323
|
+
const txType = tx._type;
|
|
324
|
+
return txType === iface_1.FlareTransactionType.EvmExportTx || txType === iface_1.FlareTransactionType.EvmImportTx;
|
|
325
|
+
}
|
|
326
|
+
get outputs() {
|
|
327
|
+
const tx = this._flareTransaction.getTx();
|
|
328
|
+
switch (this.type) {
|
|
329
|
+
case sdk_core_1.TransactionType.Import:
|
|
330
|
+
if (this.isTransactionForCChain) {
|
|
331
|
+
// C-chain Import: output is to a C-chain address
|
|
332
|
+
const importTx = tx;
|
|
333
|
+
return importTx.Outs.map((out) => ({
|
|
334
|
+
address: '0x' + buffer_1.Buffer.from(out.address.toBytes()).toString('hex'),
|
|
335
|
+
value: out.amount.value().toString(),
|
|
336
|
+
}));
|
|
337
|
+
}
|
|
338
|
+
else {
|
|
339
|
+
// P-chain Import: outputs are the baseTx.outputs (destination on P-chain)
|
|
340
|
+
const pvmImportTx = tx;
|
|
341
|
+
return pvmImportTx.baseTx.outputs.map(utils_1.default.mapOutputToEntry(this._network));
|
|
342
|
+
}
|
|
343
|
+
case sdk_core_1.TransactionType.Export:
|
|
344
|
+
if (this.isTransactionForCChain) {
|
|
345
|
+
// C-chain Export: exported outputs go to P-chain
|
|
346
|
+
const exportTx = tx;
|
|
347
|
+
return exportTx.exportedOutputs.map(utils_1.default.mapOutputToEntry(this._network));
|
|
348
|
+
}
|
|
349
|
+
else {
|
|
350
|
+
// P-chain Export: exported outputs go to C-chain
|
|
351
|
+
const pvmExportTx = tx;
|
|
352
|
+
return pvmExportTx.outs.map(utils_1.default.mapOutputToEntry(this._network));
|
|
353
|
+
}
|
|
354
|
+
case sdk_core_1.TransactionType.AddPermissionlessValidator:
|
|
355
|
+
return [
|
|
356
|
+
{
|
|
357
|
+
address: tx.subnetValidator.validator.nodeId.toString(),
|
|
358
|
+
value: tx.subnetValidator.validator.weight.toJSON(),
|
|
359
|
+
},
|
|
360
|
+
];
|
|
361
|
+
default:
|
|
362
|
+
return [];
|
|
363
|
+
}
|
|
364
|
+
}
|
|
365
|
+
get changeOutputs() {
|
|
366
|
+
const tx = this._flareTransaction.getTx();
|
|
367
|
+
// C-chain transactions and Import transactions don't have change outputs
|
|
368
|
+
if (this.isTransactionForCChain || this.type === sdk_core_1.TransactionType.Import) {
|
|
369
|
+
return [];
|
|
370
|
+
}
|
|
371
|
+
switch (this.type) {
|
|
372
|
+
case sdk_core_1.TransactionType.Export:
|
|
373
|
+
// P-chain Export: change outputs are in baseTx.outputs
|
|
374
|
+
return tx.baseTx.outputs.map(utils_1.default.mapOutputToEntry(this._network));
|
|
375
|
+
case sdk_core_1.TransactionType.AddPermissionlessValidator:
|
|
376
|
+
return tx.baseTx.outputs.map(utils_1.default.mapOutputToEntry(this._network));
|
|
377
|
+
default:
|
|
378
|
+
return [];
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
get inputs() {
|
|
382
|
+
const tx = this._flareTransaction.getTx();
|
|
383
|
+
switch (this.type) {
|
|
384
|
+
case sdk_core_1.TransactionType.Import:
|
|
385
|
+
if (this.isTransactionForCChain) {
|
|
386
|
+
// C-chain Import: inputs come from P-chain (importedInputs)
|
|
387
|
+
const importTx = tx;
|
|
388
|
+
return importTx.importedInputs.map((input) => ({
|
|
389
|
+
id: utils_1.default.cb58Encode(buffer_1.Buffer.from(input.utxoID.txID.toBytes())) + ':' + input.utxoID.outputIdx.value(),
|
|
390
|
+
address: this.fromAddresses.sort().join(iface_1.ADDRESS_SEPARATOR),
|
|
391
|
+
value: input.amount().toString(),
|
|
392
|
+
}));
|
|
393
|
+
}
|
|
394
|
+
else {
|
|
395
|
+
// P-chain Import: inputs are ins (imported from C-chain)
|
|
396
|
+
const pvmImportTx = tx;
|
|
397
|
+
return pvmImportTx.ins.map((input) => ({
|
|
398
|
+
id: utils_1.default.cb58Encode(buffer_1.Buffer.from(input.utxoID.txID.toBytes())) + ':' + input.utxoID.outputIdx.value(),
|
|
399
|
+
address: this.fromAddresses.sort().join(iface_1.ADDRESS_SEPARATOR),
|
|
400
|
+
value: input.amount().toString(),
|
|
401
|
+
}));
|
|
402
|
+
}
|
|
403
|
+
case sdk_core_1.TransactionType.Export:
|
|
404
|
+
if (this.isTransactionForCChain) {
|
|
405
|
+
// C-chain Export: inputs are from C-chain (EVM inputs)
|
|
406
|
+
const exportTx = tx;
|
|
407
|
+
return exportTx.ins.map((evmInput) => ({
|
|
408
|
+
address: '0x' + buffer_1.Buffer.from(evmInput.address.toBytes()).toString('hex'),
|
|
409
|
+
value: evmInput.amount.value().toString(),
|
|
410
|
+
nonce: Number(evmInput.nonce.value()),
|
|
411
|
+
}));
|
|
412
|
+
}
|
|
413
|
+
else {
|
|
414
|
+
// P-chain Export: inputs are from baseTx.inputs
|
|
415
|
+
const pvmExportTx = tx;
|
|
416
|
+
return pvmExportTx.baseTx.inputs.map((input) => ({
|
|
417
|
+
id: utils_1.default.cb58Encode(buffer_1.Buffer.from(input.utxoID.txID.toBytes())) + ':' + input.utxoID.outputIdx.value(),
|
|
418
|
+
address: this.fromAddresses.sort().join(iface_1.ADDRESS_SEPARATOR),
|
|
419
|
+
value: input.amount().toString(),
|
|
420
|
+
}));
|
|
421
|
+
}
|
|
422
|
+
case sdk_core_1.TransactionType.AddPermissionlessValidator:
|
|
423
|
+
default:
|
|
424
|
+
const baseTx = tx;
|
|
425
|
+
if (baseTx.baseTx?.inputs) {
|
|
426
|
+
return baseTx.baseTx.inputs.map((input) => ({
|
|
427
|
+
id: utils_1.default.cb58Encode(buffer_1.Buffer.from(input.utxoID.txID.toBytes())) + ':' + input.utxoID.outputIdx.value(),
|
|
428
|
+
address: this.fromAddresses.sort().join(iface_1.ADDRESS_SEPARATOR),
|
|
429
|
+
value: input.amount().toString(),
|
|
430
|
+
}));
|
|
431
|
+
}
|
|
432
|
+
return [];
|
|
433
|
+
}
|
|
434
|
+
}
|
|
435
|
+
explainTransaction() {
|
|
436
|
+
const txJson = this.toJson();
|
|
437
|
+
const displayOrder = ['id', 'inputs', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'type'];
|
|
438
|
+
const outputAmount = txJson.outputs.reduce((p, n) => p + BigInt(n.value), BigInt(0)).toString();
|
|
439
|
+
const changeAmount = txJson.changeOutputs.reduce((p, n) => p + BigInt(n.value), BigInt(0)).toString();
|
|
440
|
+
let rewardAddresses;
|
|
441
|
+
if ([sdk_core_1.TransactionType.AddPermissionlessValidator].includes(txJson.type)) {
|
|
442
|
+
rewardAddresses = this.rewardAddresses;
|
|
443
|
+
displayOrder.splice(6, 0, 'rewardAddresses');
|
|
444
|
+
}
|
|
445
|
+
return {
|
|
446
|
+
displayOrder,
|
|
447
|
+
id: txJson.id,
|
|
448
|
+
inputs: txJson.inputs,
|
|
449
|
+
outputs: txJson.outputs.map((o) => ({ address: o.address, amount: o.value })),
|
|
450
|
+
outputAmount,
|
|
451
|
+
changeOutputs: txJson.changeOutputs.map((o) => ({ address: o.address, amount: o.value })),
|
|
452
|
+
changeAmount,
|
|
453
|
+
rewardAddresses,
|
|
454
|
+
fee: this.fee,
|
|
455
|
+
type: txJson.type,
|
|
456
|
+
};
|
|
457
|
+
}
|
|
458
|
+
}
|
|
459
|
+
exports.Transaction = Transaction;
|
|
460
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":"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","sourcesContent":["import { FlareNetwork, BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport {\n  BaseKey,\n  BaseTransaction,\n  Entry,\n  InvalidTransactionError,\n  SigningError,\n  TransactionType,\n} from '@bitgo-beta/sdk-core';\nimport {\n  utils as FlareUtils,\n  Credential,\n  pvmSerial,\n  evmSerial,\n  UnsignedTx,\n  secp256k1,\n  EVMUnsignedTx,\n  Address,\n  Context,\n} from '@flarenetwork/flarejs';\nimport { Buffer } from 'buffer';\nimport { createHash } from 'crypto';\nimport {\n  TransactionExplanation,\n  Tx,\n  TxData,\n  ADDRESS_SEPARATOR,\n  FlareTransactionType,\n  FlrpTransactionFee,\n  DecodedUtxoObj,\n} from './iface';\nimport { FlrpFeeState } from '@bitgo/public-types';\nimport { KeyPair } from './keyPair';\nimport utils from './utils';\n\n/**\n * Checks if a signature is empty (first 90 hex chars are zeros)\n * @param signature\n * @returns {boolean}\n */\nfunction isEmptySignature(signature: string): boolean {\n  return !!signature && utils.removeHexPrefix(signature).startsWith(''.padStart(90, '0'));\n}\n\n/**\n * Interface for signature slot checking\n */\ninterface CheckSignature {\n  (signature: string, addressHex: string): boolean;\n}\n\n/**\n * Checks if an empty signature has an embedded address (non-zero bytes after position 90)\n * @param signature Hex string of the signature\n */\nfunction hasEmbeddedAddress(signature: string): boolean {\n  if (!isEmptySignature(signature)) return false;\n  const cleanSig = utils.removeHexPrefix(signature);\n  if (cleanSig.length < 130) return false;\n  const embeddedPart = cleanSig.substring(90, 130);\n  return embeddedPart !== '0'.repeat(40);\n}\n\n/**\n * Generates a function to check if a signature slot matches a given address.\n * If signatures have embedded addresses, it matches by address.\n * Otherwise, it just finds empty slots.\n * @param signatures Array of signature hex strings\n */\nfunction generateSelectorSignature(signatures: string[]): CheckSignature {\n  // Check if any empty signature has an embedded address\n  const hasEmbeddedAddresses = signatures.some((sig) => isEmptySignature(sig) && hasEmbeddedAddress(sig));\n\n  if (hasEmbeddedAddresses) {\n    // Look for address embedded in the empty signature (after position 90)\n    return function (sig: string, address: string): boolean {\n      try {\n        if (!isEmptySignature(sig)) {\n          return false;\n        }\n        const cleanSig = utils.removeHexPrefix(sig);\n        const embeddedAddr = cleanSig.substring(90, 130).toLowerCase();\n        return embeddedAddr === address.toLowerCase();\n      } catch (e) {\n        return false;\n      }\n    };\n  } else {\n    // Look for any empty slot (no embedded addresses)\n    return function (sig: string, address: string): boolean {\n      return isEmptySignature(sig);\n    };\n  }\n}\n\nexport class Transaction extends BaseTransaction {\n  protected _flareTransaction: Tx;\n  public _type: TransactionType;\n  public _network: FlareNetwork;\n  public _networkID: number;\n  public _assetId: string;\n  public _blockchainID: string;\n  public _nodeID: string;\n  public _startTime: bigint;\n  public _endTime: bigint;\n  public _stakeAmount: bigint;\n  public _threshold = 2;\n  public _locktime = BigInt(0);\n  public _fromAddresses: Uint8Array[] = [];\n  public _to: Uint8Array[] = [];\n  public _rewardAddresses: Uint8Array[] = [];\n  public _utxos: DecodedUtxoObj[] = [];\n  public _context: Context.Context;\n  public _fee: FlrpTransactionFee = { fee: '0' };\n  public _feeState: FlrpFeeState;\n  public _amount: bigint;\n  public _rawSignedBytes: Buffer | undefined;\n\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    this._network = coinConfig.network as FlareNetwork;\n    // Decode cb58-encoded asset ID to hex for use in transaction serialization\n    this._assetId = utils.cb58Decode(this._network.assetId).toString('hex');\n    this._blockchainID = this._network.blockchainID;\n    this._networkID = this._network.networkID;\n  }\n\n  get signature(): string[] {\n    if (!this.hasCredentials) {\n      return [];\n    }\n    return this.credentials[0].getSignatures().filter((s) => !isEmptySignature(s));\n  }\n\n  get credentials(): Credential[] {\n    return (this._flareTransaction as UnsignedTx)?.credentials;\n  }\n\n  get hasCredentials(): boolean {\n    return this.credentials !== undefined && this.credentials.length > 0;\n  }\n\n  /** @inheritdoc */\n  canSign({ key }: BaseKey): boolean {\n    return true;\n  }\n\n  async sign(keyPair: KeyPair): Promise<void> {\n    const prv = keyPair.getPrivateKey() as Uint8Array;\n    if (!prv) {\n      throw new SigningError('Missing private key');\n    }\n    if (!this._flareTransaction) {\n      throw new InvalidTransactionError('empty transaction to sign');\n    }\n    if (!this.hasCredentials) {\n      throw new InvalidTransactionError('empty credentials to sign');\n    }\n\n    const unsignedTx = this._flareTransaction as EVMUnsignedTx;\n    const unsignedBytes = unsignedTx.toBytes();\n    const publicKey = secp256k1.getPublicKey(prv);\n\n    // Derive both EVM and P-chain addresses from the public key\n    const evmAddressHex = new Address(secp256k1.publicKeyToEthAddress(publicKey)).toHex();\n\n    // P-chain address derivation: ripemd160(sha256(publicKey))\n    const sha256Hash = createHash('sha256').update(Buffer.from(publicKey)).digest();\n    const pChainAddressBuffer = createHash('ripemd160').update(sha256Hash).digest();\n    const pChainAddressHex = pChainAddressBuffer.toString('hex');\n\n    const addressMap = unsignedTx.getAddresses();\n\n    // Check for both EVM and P-chain address formats\n    const hasMatchingAddress = addressMap.some((addr) => {\n      const addrHex = Buffer.from(addr).toString('hex').toLowerCase();\n      return (\n        addrHex === utils.removeHexPrefix(evmAddressHex).toLowerCase() || addrHex === pChainAddressHex.toLowerCase()\n      );\n    });\n\n    const signature = await secp256k1.sign(unsignedBytes, prv);\n    let signatureSet = false;\n\n    if (hasMatchingAddress) {\n      // Use address-based slot matching (like AVAX-P)\n      let checkSign: CheckSignature | undefined = undefined;\n\n      for (const credential of unsignedTx.credentials) {\n        const signatures = credential.getSignatures();\n        if (checkSign === undefined) {\n          checkSign = generateSelectorSignature(signatures);\n        }\n\n        // Find the slot that matches this address\n        for (let i = 0; i < signatures.length; i++) {\n          const sig = signatures[i];\n          // Try matching with P-chain address first, then EVM address\n          if (checkSign(sig, pChainAddressHex) || checkSign(sig, utils.removeHexPrefix(evmAddressHex).toLowerCase())) {\n            credential.setSignature(i, signature);\n            signatureSet = true;\n            // Clear raw signed bytes since we've modified the transaction\n            this._rawSignedBytes = undefined;\n            break; // Break inner loop, but continue to sign other credentials\n          }\n        }\n        // Don't break outer loop - continue signing ALL credentials that have a matching slot\n      }\n    }\n\n    // Fallback: If address-based matching didn't work (e.g., ImportInC loaded from unsigned tx\n    // where P-chain addresses aren't in addressMaps), sign ALL empty slots across ALL credentials.\n    // This handles multisig where each UTXO needs a credential signed by the same key.\n    if (!signatureSet) {\n      for (const credential of unsignedTx.credentials) {\n        const signatures = credential.getSignatures();\n        for (let i = 0; i < signatures.length; i++) {\n          if (isEmptySignature(signatures[i])) {\n            credential.setSignature(i, signature);\n            signatureSet = true;\n            this._rawSignedBytes = undefined;\n            break; // Break inner loop, but continue to sign other credentials\n          }\n        }\n        // Don't break outer loop - continue signing ALL credentials with empty slots\n      }\n    }\n\n    if (!signatureSet) {\n      throw new SigningError('No matching signature slot found for this private key');\n    }\n  }\n\n  toBroadcastFormat(): string {\n    if (!this._flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\n    // If we have the original raw signed bytes, use them directly to preserve exact format\n    if (this._rawSignedBytes) {\n      return FlareUtils.bufferToHex(this._rawSignedBytes);\n    }\n    const unsignedTx = this._flareTransaction as UnsignedTx;\n    const signedTxBytes = unsignedTx.getSignedTx().toBytes();\n\n    // Both P-chain and C-chain transactions include checksum (matching avaxp behavior)\n    // avaxp P-chain: transaction.ts uses addChecksum() explicitly\n    // avaxp C-chain: deprecatedTransaction.ts uses Tx.toStringHex() which internally adds checksum\n    const rawTx = FlareUtils.bufferToHex(utils.addChecksum(signedTxBytes));\n    return rawTx;\n  }\n\n  toJson(): TxData {\n    if (!this._flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\n    return {\n      id: this.id,\n      inputs: this.inputs,\n      fromAddresses: this.fromAddresses,\n      threshold: this._threshold,\n      locktime: this._locktime.toString(),\n      type: this.type,\n      signatures: this.signature,\n      outputs: this.outputs,\n      changeOutputs: this.changeOutputs,\n      sourceChain: this.sourceChain,\n      destinationChain: this.destinationChain,\n    };\n  }\n\n  /**\n   * Get the source chain id or undefined if it's not a cross chain transfer.\n   */\n  get sourceChain(): string | undefined {\n    const tx = (this._flareTransaction as UnsignedTx).getTx();\n\n    switch (this.type) {\n      case TransactionType.Import:\n        if (this.isTransactionForCChain) {\n          // C-chain Import: source is the chain we're importing FROM (P-chain)\n          const importTx = tx as evmSerial.ImportTx;\n          return this.blockchainIDtoAlias(Buffer.from(importTx.sourceChain.toBytes()));\n        } else {\n          // P-chain Import: source is the chain we're importing FROM (C-chain)\n          const pvmImportTx = tx as pvmSerial.ImportTx;\n          return this.blockchainIDtoAlias(Buffer.from(pvmImportTx.sourceChain.toBytes()));\n        }\n\n      case TransactionType.Export:\n        if (this.isTransactionForCChain) {\n          // C-chain Export: source is C-chain (the blockchain ID)\n          const exportTx = tx as evmSerial.ExportTx;\n          return this.blockchainIDtoAlias(Buffer.from(exportTx.blockchainId.toBytes()));\n        } else {\n          // P-chain Export: source is P-chain (the blockchain ID from baseTx)\n          const pvmExportTx = tx as pvmSerial.ExportTx;\n          return this.blockchainIDtoAlias(Buffer.from(pvmExportTx.baseTx.BlockchainId.toBytes()));\n        }\n\n      default:\n        return undefined;\n    }\n  }\n\n  /**\n   * Get the destination chain id or undefined if it's not a cross chain transfer.\n   */\n  get destinationChain(): string | undefined {\n    const tx = (this._flareTransaction as UnsignedTx).getTx();\n\n    switch (this.type) {\n      case TransactionType.Import:\n        if (this.isTransactionForCChain) {\n          // C-chain Import: destination is C-chain (the blockchain ID)\n          const importTx = tx as evmSerial.ImportTx;\n          return this.blockchainIDtoAlias(Buffer.from(importTx.blockchainId.toBytes()));\n        } else {\n          // P-chain Import: destination is P-chain (the blockchain ID from baseTx)\n          const pvmImportTx = tx as pvmSerial.ImportTx;\n          return this.blockchainIDtoAlias(Buffer.from(pvmImportTx.baseTx.BlockchainId.toBytes()));\n        }\n\n      case TransactionType.Export:\n        if (this.isTransactionForCChain) {\n          // C-chain Export: destination is P-chain (the destination chain)\n          const exportTx = tx as evmSerial.ExportTx;\n          return this.blockchainIDtoAlias(Buffer.from(exportTx.destinationChain.toBytes()));\n        } else {\n          // P-chain Export: destination is C-chain (the destination chain)\n          const pvmExportTx = tx as pvmSerial.ExportTx;\n          return this.blockchainIDtoAlias(Buffer.from(pvmExportTx.destination.toBytes()));\n        }\n\n      default:\n        return undefined;\n    }\n  }\n\n  /**\n   * Convert a blockchainId buffer to string and return P or C alias if it matches any of those chains.\n   * @param {Buffer} blockchainIDBuffer\n   * @return {string} blockchainID or alias if exists.\n   * @private\n   */\n  blockchainIDtoAlias(blockchainIDBuffer: Buffer): string {\n    const blockchainId = utils.cb58Encode(blockchainIDBuffer);\n    if (blockchainId === this._network.cChainBlockchainID) {\n      return 'C';\n    }\n    if (blockchainId === this._network.blockchainID) {\n      return 'P';\n    }\n    return blockchainId;\n  }\n\n  setTransaction(tx: Tx): void {\n    this._flareTransaction = tx as UnsignedTx;\n  }\n\n  /**\n   * Get the underlying Flare transaction\n   * @returns The Flare transaction object\n   */\n  getFlareTransaction(): Tx {\n    return this._flareTransaction;\n  }\n\n  setTransactionType(transactionType: TransactionType): void {\n    if (![TransactionType.AddPermissionlessValidator].includes(transactionType)) {\n      throw new Error(`Transaction type ${transactionType} is not supported`);\n    }\n    this._type = transactionType;\n  }\n\n  get signablePayload(): Buffer {\n    return utils.sha256((this._flareTransaction as UnsignedTx).toBytes());\n  }\n\n  get id(): string {\n    const unsignedTx = this._flareTransaction as UnsignedTx;\n    const txBytes = unsignedTx.getSignedTx().toBytes();\n    const bufferArray = utils.sha256(txBytes);\n    return utils.cb58Encode(Buffer.from(bufferArray));\n  }\n\n  get fromAddresses(): string[] {\n    return this._fromAddresses.map((a) => FlareUtils.format(this._network.alias, this._network.hrp, a));\n  }\n\n  get rewardAddresses(): string[] {\n    return this._rewardAddresses.map((a) => FlareUtils.format(this._network.alias, this._network.hrp, a));\n  }\n\n  get fee(): FlrpTransactionFee {\n    return this._fee;\n  }\n\n  /**\n   * Check if this transaction is for C-chain (EVM transactions)\n   */\n  get isTransactionForCChain(): boolean {\n    const tx = (this._flareTransaction as UnsignedTx).getTx();\n    const txType = (tx as { _type?: string })._type;\n    return txType === FlareTransactionType.EvmExportTx || txType === FlareTransactionType.EvmImportTx;\n  }\n\n  get outputs(): Entry[] {\n    const tx = (this._flareTransaction as UnsignedTx).getTx();\n\n    switch (this.type) {\n      case TransactionType.Import:\n        if (this.isTransactionForCChain) {\n          // C-chain Import: output is to a C-chain address\n          const importTx = tx as evmSerial.ImportTx;\n          return importTx.Outs.map((out) => ({\n            address: '0x' + Buffer.from(out.address.toBytes()).toString('hex'),\n            value: out.amount.value().toString(),\n          }));\n        } else {\n          // P-chain Import: outputs are the baseTx.outputs (destination on P-chain)\n          const pvmImportTx = tx as pvmSerial.ImportTx;\n          return pvmImportTx.baseTx.outputs.map(utils.mapOutputToEntry(this._network));\n        }\n\n      case TransactionType.Export:\n        if (this.isTransactionForCChain) {\n          // C-chain Export: exported outputs go to P-chain\n          const exportTx = tx as evmSerial.ExportTx;\n          return exportTx.exportedOutputs.map(utils.mapOutputToEntry(this._network));\n        } else {\n          // P-chain Export: exported outputs go to C-chain\n          const pvmExportTx = tx as pvmSerial.ExportTx;\n          return pvmExportTx.outs.map(utils.mapOutputToEntry(this._network));\n        }\n\n      case TransactionType.AddPermissionlessValidator:\n        return [\n          {\n            address: (tx as pvmSerial.AddPermissionlessValidatorTx).subnetValidator.validator.nodeId.toString(),\n            value: (tx as pvmSerial.AddPermissionlessValidatorTx).subnetValidator.validator.weight.toJSON(),\n          },\n        ];\n\n      default:\n        return [];\n    }\n  }\n\n  get changeOutputs(): Entry[] {\n    const tx = (this._flareTransaction as UnsignedTx).getTx();\n\n    // C-chain transactions and Import transactions don't have change outputs\n    if (this.isTransactionForCChain || this.type === TransactionType.Import) {\n      return [];\n    }\n\n    switch (this.type) {\n      case TransactionType.Export:\n        // P-chain Export: change outputs are in baseTx.outputs\n        return (tx as pvmSerial.ExportTx).baseTx.outputs.map(utils.mapOutputToEntry(this._network));\n\n      case TransactionType.AddPermissionlessValidator:\n        return (tx as pvmSerial.AddPermissionlessValidatorTx).baseTx.outputs.map(utils.mapOutputToEntry(this._network));\n\n      default:\n        return [];\n    }\n  }\n\n  get inputs(): Entry[] {\n    const tx = (this._flareTransaction as UnsignedTx).getTx();\n\n    switch (this.type) {\n      case TransactionType.Import:\n        if (this.isTransactionForCChain) {\n          // C-chain Import: inputs come from P-chain (importedInputs)\n          const importTx = tx as evmSerial.ImportTx;\n          return importTx.importedInputs.map((input) => ({\n            id: utils.cb58Encode(Buffer.from(input.utxoID.txID.toBytes())) + ':' + input.utxoID.outputIdx.value(),\n            address: this.fromAddresses.sort().join(ADDRESS_SEPARATOR),\n            value: input.amount().toString(),\n          }));\n        } else {\n          // P-chain Import: inputs are ins (imported from C-chain)\n          const pvmImportTx = tx as pvmSerial.ImportTx;\n          return pvmImportTx.ins.map((input) => ({\n            id: utils.cb58Encode(Buffer.from(input.utxoID.txID.toBytes())) + ':' + input.utxoID.outputIdx.value(),\n            address: this.fromAddresses.sort().join(ADDRESS_SEPARATOR),\n            value: input.amount().toString(),\n          }));\n        }\n\n      case TransactionType.Export:\n        if (this.isTransactionForCChain) {\n          // C-chain Export: inputs are from C-chain (EVM inputs)\n          const exportTx = tx as evmSerial.ExportTx;\n          return exportTx.ins.map((evmInput) => ({\n            address: '0x' + Buffer.from(evmInput.address.toBytes()).toString('hex'),\n            value: evmInput.amount.value().toString(),\n            nonce: Number(evmInput.nonce.value()),\n          }));\n        } else {\n          // P-chain Export: inputs are from baseTx.inputs\n          const pvmExportTx = tx as pvmSerial.ExportTx;\n          return pvmExportTx.baseTx.inputs.map((input) => ({\n            id: utils.cb58Encode(Buffer.from(input.utxoID.txID.toBytes())) + ':' + input.utxoID.outputIdx.value(),\n            address: this.fromAddresses.sort().join(ADDRESS_SEPARATOR),\n            value: input.amount().toString(),\n          }));\n        }\n\n      case TransactionType.AddPermissionlessValidator:\n      default:\n        const baseTx = tx as pvmSerial.AddPermissionlessValidatorTx;\n        if (baseTx.baseTx?.inputs) {\n          return baseTx.baseTx.inputs.map((input) => ({\n            id: utils.cb58Encode(Buffer.from(input.utxoID.txID.toBytes())) + ':' + input.utxoID.outputIdx.value(),\n            address: this.fromAddresses.sort().join(ADDRESS_SEPARATOR),\n            value: input.amount().toString(),\n          }));\n        }\n        return [];\n    }\n  }\n\n  explainTransaction(): TransactionExplanation {\n    const txJson = this.toJson();\n    const displayOrder = ['id', 'inputs', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'type'];\n\n    const outputAmount = txJson.outputs.reduce((p, n) => p + BigInt(n.value), BigInt(0)).toString();\n    const changeAmount = txJson.changeOutputs.reduce((p, n) => p + BigInt(n.value), BigInt(0)).toString();\n\n    let rewardAddresses;\n    if ([TransactionType.AddPermissionlessValidator].includes(txJson.type)) {\n      rewardAddresses = this.rewardAddresses;\n      displayOrder.splice(6, 0, 'rewardAddresses');\n    }\n\n    return {\n      displayOrder,\n      id: txJson.id,\n      inputs: txJson.inputs,\n      outputs: txJson.outputs.map((o) => ({ address: o.address, amount: o.value })),\n      outputAmount,\n      changeOutputs: txJson.changeOutputs.map((o) => ({ address: o.address, amount: o.value })),\n      changeAmount,\n      rewardAddresses,\n      fee: this.fee,\n      type: txJson.type,\n    };\n  }\n}\n"]}
|