@bitgo-beta/sdk-coin-flrp 1.0.0-alpha.7 → 1.0.0-alpha.9
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/atomicTransactionBuilder.d.ts +10 -7
- package/dist/src/lib/atomicTransactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/atomicTransactionBuilder.js +155 -89
- package/dist/src/lib/constants.d.ts +160 -1
- package/dist/src/lib/constants.d.ts.map +1 -1
- package/dist/src/lib/constants.js +213 -3
- package/dist/src/lib/delegatorTxBuilder.d.ts +58 -0
- package/dist/src/lib/delegatorTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/delegatorTxBuilder.js +224 -0
- package/dist/src/lib/exportInCTxBuilder.d.ts +1 -1
- package/dist/src/lib/exportInCTxBuilder.d.ts.map +1 -1
- package/dist/src/lib/exportInCTxBuilder.js +46 -17
- package/dist/src/lib/exportInPTxBuilder.d.ts +1 -1
- package/dist/src/lib/exportInPTxBuilder.d.ts.map +1 -1
- package/dist/src/lib/exportInPTxBuilder.js +70 -6
- package/dist/src/lib/importInCTxBuilder.d.ts +67 -0
- package/dist/src/lib/importInCTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/importInCTxBuilder.js +403 -0
- package/dist/src/lib/importInPTxBuilder.d.ts +73 -0
- package/dist/src/lib/importInPTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/importInPTxBuilder.js +464 -0
- package/dist/src/lib/index.d.ts +5 -0
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +11 -2
- package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts +81 -0
- package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/permissionlessValidatorTxBuilder.js +248 -0
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +14 -13
- package/dist/src/lib/transactionBuilder.d.ts +85 -0
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder.js +167 -0
- package/dist/src/lib/types.d.ts +78 -0
- package/dist/src/lib/types.d.ts.map +1 -0
- package/dist/src/lib/types.js +5 -0
- package/dist/src/lib/utils.d.ts +2 -0
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +23 -12
- package/dist/src/lib/validatorTxBuilder.d.ts +40 -0
- package/dist/src/lib/validatorTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/validatorTxBuilder.js +180 -0
- package/dist/test/unit/delegatorTxBuilder.test.d.ts +2 -0
- package/dist/test/unit/delegatorTxBuilder.test.d.ts.map +1 -0
- package/dist/test/unit/delegatorTxBuilder.test.js +233 -0
- 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 +584 -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 +377 -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 +257 -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 +500 -0
- package/dist/test/unit/permissionlessValidatorTxBuilder.test.d.ts +2 -0
- package/dist/test/unit/permissionlessValidatorTxBuilder.test.d.ts.map +1 -0
- package/dist/test/unit/permissionlessValidatorTxBuilder.test.js +271 -0
- package/dist/test/unit/transactionBuilder.test.d.ts +2 -0
- package/dist/test/unit/transactionBuilder.test.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder.test.js +114 -0
- package/dist/test/unit/validatorTxBuilder.test.d.ts +2 -0
- package/dist/test/unit/validatorTxBuilder.test.d.ts.map +1 -0
- package/dist/test/unit/validatorTxBuilder.test.js +293 -0
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +7 -7
- package/dist/test/unit/lib/exportTxBuilder.d.ts +0 -2
- package/dist/test/unit/lib/exportTxBuilder.d.ts.map +0 -1
- package/dist/test/unit/lib/exportTxBuilder.js +0 -45
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import { TransactionType } from '@bitgo-beta/sdk-core';
|
|
2
|
+
import { BaseCoin as CoinConfig } from '@bitgo-beta/statics';
|
|
3
|
+
import { AtomicTransactionBuilder } from './atomicTransactionBuilder';
|
|
4
|
+
import { Tx } from './iface';
|
|
5
|
+
export declare class PermissionlessValidatorTxBuilder extends AtomicTransactionBuilder {
|
|
6
|
+
protected _nodeID: string | undefined;
|
|
7
|
+
protected _blsPublicKey: string | undefined;
|
|
8
|
+
protected _blsSignature: string | undefined;
|
|
9
|
+
protected _startTime: bigint | undefined;
|
|
10
|
+
protected _endTime: bigint | undefined;
|
|
11
|
+
protected _stakeAmount: bigint | undefined;
|
|
12
|
+
protected _delegationFeeRate: number | undefined;
|
|
13
|
+
/**
|
|
14
|
+
* @param coinConfig
|
|
15
|
+
*/
|
|
16
|
+
constructor(coinConfig: Readonly<CoinConfig>);
|
|
17
|
+
/**
|
|
18
|
+
* get transaction type
|
|
19
|
+
* @protected
|
|
20
|
+
*/
|
|
21
|
+
protected get transactionType(): TransactionType;
|
|
22
|
+
/**
|
|
23
|
+
* Set the node ID for permissionless validation
|
|
24
|
+
* @param nodeID - The node ID
|
|
25
|
+
*/
|
|
26
|
+
nodeID(nodeID: string): this;
|
|
27
|
+
/**
|
|
28
|
+
* Set the BLS public key for permissionless validation
|
|
29
|
+
* @param blsPublicKey - The BLS public key
|
|
30
|
+
*/
|
|
31
|
+
blsPublicKey(blsPublicKey: string): this;
|
|
32
|
+
/**
|
|
33
|
+
* Set the BLS signature for permissionless validation
|
|
34
|
+
* @param blsSignature - The BLS signature
|
|
35
|
+
*/
|
|
36
|
+
blsSignature(blsSignature: string): this;
|
|
37
|
+
/**
|
|
38
|
+
* Set the start time for validation
|
|
39
|
+
* @param startTime - Unix timestamp for when validation starts
|
|
40
|
+
*/
|
|
41
|
+
startTime(startTime: string | number | bigint): this;
|
|
42
|
+
/**
|
|
43
|
+
* Set the end time for validation
|
|
44
|
+
* @param endTime - Unix timestamp for when validation ends
|
|
45
|
+
*/
|
|
46
|
+
endTime(endTime: string | number | bigint): this;
|
|
47
|
+
/**
|
|
48
|
+
* Set the stake amount for validation
|
|
49
|
+
* @param amount - Amount to stake (in nFLR)
|
|
50
|
+
*/
|
|
51
|
+
stakeAmount(amount: string | number | bigint): this;
|
|
52
|
+
/**
|
|
53
|
+
* Set the delegation fee rate
|
|
54
|
+
* @param value - Delegation fee rate in basis points
|
|
55
|
+
*/
|
|
56
|
+
delegationFeeRate(value: number): this;
|
|
57
|
+
/**
|
|
58
|
+
* Set reward addresses where validation rewards should be sent
|
|
59
|
+
* @param addresses - Array of reward addresses
|
|
60
|
+
*/
|
|
61
|
+
rewardAddresses(addresses: string[]): this;
|
|
62
|
+
/**
|
|
63
|
+
* Validate that the delegation fee is at least the minDelegationFee
|
|
64
|
+
* @param delegationFeeRate number
|
|
65
|
+
*/
|
|
66
|
+
validateDelegationFeeRate(delegationFeeRate: number): void;
|
|
67
|
+
/** @inheritdoc */
|
|
68
|
+
initBuilder(tx: Tx): this;
|
|
69
|
+
/**
|
|
70
|
+
* Verify if the transaction is a permissionless validator transaction
|
|
71
|
+
* @param tx
|
|
72
|
+
*/
|
|
73
|
+
static verifyTxType(tx: unknown): boolean;
|
|
74
|
+
verifyTxType(tx: unknown): boolean;
|
|
75
|
+
/**
|
|
76
|
+
* Build the permissionless validator transaction
|
|
77
|
+
* @protected
|
|
78
|
+
*/
|
|
79
|
+
protected buildFlareTransaction(): Promise<void>;
|
|
80
|
+
}
|
|
81
|
+
//# sourceMappingURL=permissionlessValidatorTxBuilder.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"permissionlessValidatorTxBuilder.d.ts","sourceRoot":"","sources":["../../../src/lib/permissionlessValidatorTxBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,EAAyB,eAAe,EAAE,MAAM,sBAAsB,CAAC;AAC9E,OAAO,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC7D,OAAO,EAAE,wBAAwB,EAAE,MAAM,4BAA4B,CAAC;AACtE,OAAO,EAAE,EAAE,EAAE,MAAM,SAAS,CAAC;AAa7B,qBAAa,gCAAiC,SAAQ,wBAAwB;IAC5E,SAAS,CAAC,OAAO,EAAE,MAAM,GAAG,SAAS,CAAC;IACtC,SAAS,CAAC,aAAa,EAAE,MAAM,GAAG,SAAS,CAAC;IAC5C,SAAS,CAAC,aAAa,EAAE,MAAM,GAAG,SAAS,CAAC;IAC5C,SAAS,CAAC,UAAU,EAAE,MAAM,GAAG,SAAS,CAAC;IACzC,SAAS,CAAC,QAAQ,EAAE,MAAM,GAAG,SAAS,CAAC;IACvC,SAAS,CAAC,YAAY,EAAE,MAAM,GAAG,SAAS,CAAC;IAC3C,SAAS,CAAC,kBAAkB,EAAE,MAAM,GAAG,SAAS,CAAC;IAEjD;;OAEG;gBACS,UAAU,EAAE,QAAQ,CAAC,UAAU,CAAC;IAW5C;;;OAGG;IACH,SAAS,KAAK,eAAe,IAAI,eAAe,CAE/C;IAED;;;OAGG;IACH,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI;IAQ5B;;;OAGG;IACH,YAAY,CAAC,YAAY,EAAE,MAAM,GAAG,IAAI;IAiBxC;;;OAGG;IACH,YAAY,CAAC,YAAY,EAAE,MAAM,GAAG,IAAI;IAcxC;;;OAGG;IACH,SAAS,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,MAAM,GAAG,IAAI;IASpD;;;OAGG;IACH,OAAO,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,GAAG,MAAM,GAAG,IAAI;IAShD;;;OAGG;IACH,WAAW,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,GAAG,MAAM,GAAG,IAAI;IASnD;;;OAGG;IACH,iBAAiB,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAMtC;;;OAGG;IACH,eAAe,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,IAAI;IAS1C;;;OAGG;IACH,yBAAyB,CAAC,iBAAiB,EAAE,MAAM,GAAG,IAAI;IAW1D,kBAAkB;IAClB,WAAW,CAAC,EAAE,EAAE,EAAE,GAAG,IAAI;IAkCzB;;;OAGG;IACH,MAAM,CAAC,YAAY,CAAC,EAAE,EAAE,OAAO,GAAG,OAAO;IAOzC,YAAY,CAAC,EAAE,EAAE,OAAO,GAAG,OAAO;IAIlC;;;OAGG;cACa,qBAAqB,IAAI,OAAO,CAAC,IAAI,CAAC;CAgEvD"}
|
|
@@ -0,0 +1,248 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.PermissionlessValidatorTxBuilder = void 0;
|
|
4
|
+
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
5
|
+
const atomicTransactionBuilder_1 = require("./atomicTransactionBuilder");
|
|
6
|
+
const constants_1 = require("./constants");
|
|
7
|
+
const utils_1 = require("./utils");
|
|
8
|
+
class PermissionlessValidatorTxBuilder extends atomicTransactionBuilder_1.AtomicTransactionBuilder {
|
|
9
|
+
/**
|
|
10
|
+
* @param coinConfig
|
|
11
|
+
*/
|
|
12
|
+
constructor(coinConfig) {
|
|
13
|
+
super(coinConfig);
|
|
14
|
+
this._nodeID = undefined;
|
|
15
|
+
this._blsPublicKey = undefined;
|
|
16
|
+
this._blsSignature = undefined;
|
|
17
|
+
this._startTime = undefined;
|
|
18
|
+
this._endTime = undefined;
|
|
19
|
+
this._stakeAmount = undefined;
|
|
20
|
+
this._delegationFeeRate = undefined;
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* get transaction type
|
|
24
|
+
* @protected
|
|
25
|
+
*/
|
|
26
|
+
get transactionType() {
|
|
27
|
+
return sdk_core_1.TransactionType.AddPermissionlessValidator;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Set the node ID for permissionless validation
|
|
31
|
+
* @param nodeID - The node ID
|
|
32
|
+
*/
|
|
33
|
+
nodeID(nodeID) {
|
|
34
|
+
if (!nodeID || nodeID.length === 0) {
|
|
35
|
+
throw new sdk_core_1.BuildTransactionError('Node ID cannot be empty');
|
|
36
|
+
}
|
|
37
|
+
this._nodeID = nodeID;
|
|
38
|
+
return this;
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Set the BLS public key for permissionless validation
|
|
42
|
+
* @param blsPublicKey - The BLS public key
|
|
43
|
+
*/
|
|
44
|
+
blsPublicKey(blsPublicKey) {
|
|
45
|
+
if (!blsPublicKey || blsPublicKey.length === 0) {
|
|
46
|
+
throw new sdk_core_1.BuildTransactionError('BLS public key cannot be empty');
|
|
47
|
+
}
|
|
48
|
+
// BLS public key should be 48 bytes (96 hex characters) with 0x prefix or 192 hex characters with 0x prefix for uncompressed
|
|
49
|
+
if (!(0, utils_1.createHexRegex)(constants_1.BLS_PUBLIC_KEY_COMPRESSED_LENGTH, true).test(blsPublicKey) &&
|
|
50
|
+
!(0, utils_1.createHexRegex)(constants_1.BLS_PUBLIC_KEY_UNCOMPRESSED_LENGTH, true).test(blsPublicKey)) {
|
|
51
|
+
throw new sdk_core_1.BuildTransactionError('Invalid BLS public key format');
|
|
52
|
+
}
|
|
53
|
+
this._blsPublicKey = blsPublicKey;
|
|
54
|
+
return this;
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Set the BLS signature for permissionless validation
|
|
58
|
+
* @param blsSignature - The BLS signature
|
|
59
|
+
*/
|
|
60
|
+
blsSignature(blsSignature) {
|
|
61
|
+
if (!blsSignature || blsSignature.length === 0) {
|
|
62
|
+
throw new sdk_core_1.BuildTransactionError('BLS signature cannot be empty');
|
|
63
|
+
}
|
|
64
|
+
// BLS signature should be 96 bytes (192 hex characters) with 0x prefix
|
|
65
|
+
if (!(0, utils_1.createHexRegex)(constants_1.BLS_SIGNATURE_LENGTH, true).test(blsSignature)) {
|
|
66
|
+
throw new sdk_core_1.BuildTransactionError('Invalid BLS signature format');
|
|
67
|
+
}
|
|
68
|
+
this._blsSignature = blsSignature;
|
|
69
|
+
return this;
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Set the start time for validation
|
|
73
|
+
* @param startTime - Unix timestamp for when validation starts
|
|
74
|
+
*/
|
|
75
|
+
startTime(startTime) {
|
|
76
|
+
const time = BigInt(startTime);
|
|
77
|
+
if (time < 0) {
|
|
78
|
+
throw new sdk_core_1.BuildTransactionError('Start time must be non-negative');
|
|
79
|
+
}
|
|
80
|
+
this._startTime = time;
|
|
81
|
+
return this;
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Set the end time for validation
|
|
85
|
+
* @param endTime - Unix timestamp for when validation ends
|
|
86
|
+
*/
|
|
87
|
+
endTime(endTime) {
|
|
88
|
+
const time = BigInt(endTime);
|
|
89
|
+
if (time <= 0) {
|
|
90
|
+
throw new sdk_core_1.BuildTransactionError('End time must be positive');
|
|
91
|
+
}
|
|
92
|
+
this._endTime = time;
|
|
93
|
+
return this;
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Set the stake amount for validation
|
|
97
|
+
* @param amount - Amount to stake (in nFLR)
|
|
98
|
+
*/
|
|
99
|
+
stakeAmount(amount) {
|
|
100
|
+
const stake = BigInt(amount);
|
|
101
|
+
if (stake <= 0) {
|
|
102
|
+
throw new sdk_core_1.BuildTransactionError('Stake amount must be positive');
|
|
103
|
+
}
|
|
104
|
+
this._stakeAmount = stake;
|
|
105
|
+
return this;
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Set the delegation fee rate
|
|
109
|
+
* @param value - Delegation fee rate in basis points
|
|
110
|
+
*/
|
|
111
|
+
delegationFeeRate(value) {
|
|
112
|
+
this.validateDelegationFeeRate(value);
|
|
113
|
+
this._delegationFeeRate = value;
|
|
114
|
+
return this;
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* Set reward addresses where validation rewards should be sent
|
|
118
|
+
* @param addresses - Array of reward addresses
|
|
119
|
+
*/
|
|
120
|
+
rewardAddresses(addresses) {
|
|
121
|
+
if (!addresses || addresses.length === 0) {
|
|
122
|
+
throw new sdk_core_1.BuildTransactionError('At least one reward address is required');
|
|
123
|
+
}
|
|
124
|
+
// Store reward addresses in the transaction (we'll need to extend the type)
|
|
125
|
+
this.transaction._rewardAddresses = addresses;
|
|
126
|
+
return this;
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* Validate that the delegation fee is at least the minDelegationFee
|
|
130
|
+
* @param delegationFeeRate number
|
|
131
|
+
*/
|
|
132
|
+
validateDelegationFeeRate(delegationFeeRate) {
|
|
133
|
+
// For Flare, use a minimum delegation fee of 2% (20000 basis points)
|
|
134
|
+
const minDelegationFee = constants_1.MIN_DELEGATION_FEE_BASIS_POINTS;
|
|
135
|
+
if (delegationFeeRate < minDelegationFee) {
|
|
136
|
+
const minDelegationFeePercent = (minDelegationFee / constants_1.BASIS_POINTS_DIVISOR) * constants_1.PERCENTAGE_MULTIPLIER;
|
|
137
|
+
throw new sdk_core_1.BuildTransactionError(`Delegation fee cannot be less than ${minDelegationFee} basis points (${minDelegationFeePercent}%)`);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
/** @inheritdoc */
|
|
141
|
+
initBuilder(tx) {
|
|
142
|
+
// Extract permissionless validator-specific fields from transaction
|
|
143
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
144
|
+
const txData = tx;
|
|
145
|
+
if (txData.nodeID) {
|
|
146
|
+
this._nodeID = txData.nodeID;
|
|
147
|
+
}
|
|
148
|
+
if (txData.blsPublicKey) {
|
|
149
|
+
this._blsPublicKey = txData.blsPublicKey;
|
|
150
|
+
}
|
|
151
|
+
if (txData.blsSignature) {
|
|
152
|
+
this._blsSignature = txData.blsSignature;
|
|
153
|
+
}
|
|
154
|
+
if (txData.startTime) {
|
|
155
|
+
this._startTime = BigInt(txData.startTime);
|
|
156
|
+
}
|
|
157
|
+
if (txData.endTime) {
|
|
158
|
+
this._endTime = BigInt(txData.endTime);
|
|
159
|
+
}
|
|
160
|
+
if (txData.stakeAmount) {
|
|
161
|
+
this._stakeAmount = BigInt(txData.stakeAmount);
|
|
162
|
+
}
|
|
163
|
+
if (txData.delegationFeeRate !== undefined) {
|
|
164
|
+
this._delegationFeeRate = txData.delegationFeeRate;
|
|
165
|
+
}
|
|
166
|
+
if (txData.rewardAddresses) {
|
|
167
|
+
this.transaction._rewardAddresses =
|
|
168
|
+
txData.rewardAddresses;
|
|
169
|
+
}
|
|
170
|
+
return this;
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* Verify if the transaction is a permissionless validator transaction
|
|
174
|
+
* @param tx
|
|
175
|
+
*/
|
|
176
|
+
static verifyTxType(tx) {
|
|
177
|
+
// Check if transaction has permissionless validator-specific properties
|
|
178
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
179
|
+
const txData = tx;
|
|
180
|
+
return txData && txData.blsPublicKey && txData.blsSignature;
|
|
181
|
+
}
|
|
182
|
+
verifyTxType(tx) {
|
|
183
|
+
return PermissionlessValidatorTxBuilder.verifyTxType(tx);
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* Build the permissionless validator transaction
|
|
187
|
+
* @protected
|
|
188
|
+
*/
|
|
189
|
+
async buildFlareTransaction() {
|
|
190
|
+
// Basic validation
|
|
191
|
+
if (!this._nodeID) {
|
|
192
|
+
throw new sdk_core_1.BuildTransactionError('Node ID is required for permissionless validator transaction');
|
|
193
|
+
}
|
|
194
|
+
if (!this._blsPublicKey) {
|
|
195
|
+
throw new sdk_core_1.BuildTransactionError('BLS public key is required for permissionless validator transaction');
|
|
196
|
+
}
|
|
197
|
+
if (!this._blsSignature) {
|
|
198
|
+
throw new sdk_core_1.BuildTransactionError('BLS signature is required for permissionless validator transaction');
|
|
199
|
+
}
|
|
200
|
+
if (!this._startTime) {
|
|
201
|
+
throw new sdk_core_1.BuildTransactionError('Start time is required for permissionless validator transaction');
|
|
202
|
+
}
|
|
203
|
+
if (!this._endTime) {
|
|
204
|
+
throw new sdk_core_1.BuildTransactionError('End time is required for permissionless validator transaction');
|
|
205
|
+
}
|
|
206
|
+
if (!this._stakeAmount) {
|
|
207
|
+
throw new sdk_core_1.BuildTransactionError('Stake amount is required for permissionless validator transaction');
|
|
208
|
+
}
|
|
209
|
+
if (this._delegationFeeRate === undefined) {
|
|
210
|
+
throw new sdk_core_1.BuildTransactionError('Delegation fee rate is required for permissionless validator transaction');
|
|
211
|
+
}
|
|
212
|
+
const rewardAddresses = this.transaction
|
|
213
|
+
._rewardAddresses;
|
|
214
|
+
if (!rewardAddresses || rewardAddresses.length === 0) {
|
|
215
|
+
throw new sdk_core_1.BuildTransactionError('Reward addresses are required for permissionless validator transaction');
|
|
216
|
+
}
|
|
217
|
+
// Validate time range
|
|
218
|
+
if (this._endTime <= this._startTime) {
|
|
219
|
+
throw new sdk_core_1.BuildTransactionError('End time must be after start time');
|
|
220
|
+
}
|
|
221
|
+
try {
|
|
222
|
+
// TODO: Implement actual FlareJS PVM API call when available
|
|
223
|
+
// For now, create a placeholder transaction structure
|
|
224
|
+
const validatorTx = {
|
|
225
|
+
type: constants_1.ADD_PERMISSIONLESS_VALIDATOR_TYPE,
|
|
226
|
+
nodeID: this._nodeID,
|
|
227
|
+
blsPublicKey: this._blsPublicKey,
|
|
228
|
+
blsSignature: this._blsSignature,
|
|
229
|
+
startTime: this._startTime,
|
|
230
|
+
endTime: this._endTime,
|
|
231
|
+
stakeAmount: this._stakeAmount,
|
|
232
|
+
delegationFeeRate: this._delegationFeeRate,
|
|
233
|
+
rewardAddress: rewardAddresses[0],
|
|
234
|
+
fromAddresses: this.transaction._fromAddresses,
|
|
235
|
+
networkId: this.transaction._networkID,
|
|
236
|
+
sourceBlockchainId: this.transaction._blockchainID,
|
|
237
|
+
threshold: this.transaction._threshold || 1,
|
|
238
|
+
locktime: this.transaction._locktime || 0n,
|
|
239
|
+
};
|
|
240
|
+
this.transaction.setTransaction(validatorTx);
|
|
241
|
+
}
|
|
242
|
+
catch (error) {
|
|
243
|
+
throw new sdk_core_1.BuildTransactionError(`Failed to build permissionless validator transaction: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
exports.PermissionlessValidatorTxBuilder = PermissionlessValidatorTxBuilder;
|
|
248
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissionlessValidatorTxBuilder.js","sourceRoot":"","sources":["../../../src/lib/permissionlessValidatorTxBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { BuildTransactionError, TransactionType } from '@bitgo-beta/sdk-core';\nimport { BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { AtomicTransactionBuilder } from './atomicTransactionBuilder';\nimport { Tx } from './iface';\nimport { PermissionlessValidatorExtendedTransaction } from './types';\nimport {\n  ADD_PERMISSIONLESS_VALIDATOR_TYPE,\n  BASIS_POINTS_DIVISOR,\n  BLS_PUBLIC_KEY_COMPRESSED_LENGTH,\n  BLS_PUBLIC_KEY_UNCOMPRESSED_LENGTH,\n  BLS_SIGNATURE_LENGTH,\n  MIN_DELEGATION_FEE_BASIS_POINTS,\n  PERCENTAGE_MULTIPLIER,\n} from './constants';\nimport { createHexRegex } from './utils';\n\nexport class PermissionlessValidatorTxBuilder extends AtomicTransactionBuilder {\n  protected _nodeID: string | undefined;\n  protected _blsPublicKey: string | undefined;\n  protected _blsSignature: string | undefined;\n  protected _startTime: bigint | undefined;\n  protected _endTime: bigint | undefined;\n  protected _stakeAmount: bigint | undefined;\n  protected _delegationFeeRate: number | undefined;\n\n  /**\n   * @param coinConfig\n   */\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    this._nodeID = undefined;\n    this._blsPublicKey = undefined;\n    this._blsSignature = undefined;\n    this._startTime = undefined;\n    this._endTime = undefined;\n    this._stakeAmount = undefined;\n    this._delegationFeeRate = undefined;\n  }\n\n  /**\n   * get transaction type\n   * @protected\n   */\n  protected get transactionType(): TransactionType {\n    return TransactionType.AddPermissionlessValidator;\n  }\n\n  /**\n   * Set the node ID for permissionless validation\n   * @param nodeID - The node ID\n   */\n  nodeID(nodeID: string): this {\n    if (!nodeID || nodeID.length === 0) {\n      throw new BuildTransactionError('Node ID cannot be empty');\n    }\n    this._nodeID = nodeID;\n    return this;\n  }\n\n  /**\n   * Set the BLS public key for permissionless validation\n   * @param blsPublicKey - The BLS public key\n   */\n  blsPublicKey(blsPublicKey: string): this {\n    if (!blsPublicKey || blsPublicKey.length === 0) {\n      throw new BuildTransactionError('BLS public key cannot be empty');\n    }\n\n    // BLS public key should be 48 bytes (96 hex characters) with 0x prefix or 192 hex characters with 0x prefix for uncompressed\n    if (\n      !createHexRegex(BLS_PUBLIC_KEY_COMPRESSED_LENGTH, true).test(blsPublicKey) &&\n      !createHexRegex(BLS_PUBLIC_KEY_UNCOMPRESSED_LENGTH, true).test(blsPublicKey)\n    ) {\n      throw new BuildTransactionError('Invalid BLS public key format');\n    }\n\n    this._blsPublicKey = blsPublicKey;\n    return this;\n  }\n\n  /**\n   * Set the BLS signature for permissionless validation\n   * @param blsSignature - The BLS signature\n   */\n  blsSignature(blsSignature: string): this {\n    if (!blsSignature || blsSignature.length === 0) {\n      throw new BuildTransactionError('BLS signature cannot be empty');\n    }\n\n    // BLS signature should be 96 bytes (192 hex characters) with 0x prefix\n    if (!createHexRegex(BLS_SIGNATURE_LENGTH, true).test(blsSignature)) {\n      throw new BuildTransactionError('Invalid BLS signature format');\n    }\n\n    this._blsSignature = blsSignature;\n    return this;\n  }\n\n  /**\n   * Set the start time for validation\n   * @param startTime - Unix timestamp for when validation starts\n   */\n  startTime(startTime: string | number | bigint): this {\n    const time = BigInt(startTime);\n    if (time < 0) {\n      throw new BuildTransactionError('Start time must be non-negative');\n    }\n    this._startTime = time;\n    return this;\n  }\n\n  /**\n   * Set the end time for validation\n   * @param endTime - Unix timestamp for when validation ends\n   */\n  endTime(endTime: string | number | bigint): this {\n    const time = BigInt(endTime);\n    if (time <= 0) {\n      throw new BuildTransactionError('End time must be positive');\n    }\n    this._endTime = time;\n    return this;\n  }\n\n  /**\n   * Set the stake amount for validation\n   * @param amount - Amount to stake (in nFLR)\n   */\n  stakeAmount(amount: string | number | bigint): this {\n    const stake = BigInt(amount);\n    if (stake <= 0) {\n      throw new BuildTransactionError('Stake amount must be positive');\n    }\n    this._stakeAmount = stake;\n    return this;\n  }\n\n  /**\n   * Set the delegation fee rate\n   * @param value - Delegation fee rate in basis points\n   */\n  delegationFeeRate(value: number): this {\n    this.validateDelegationFeeRate(value);\n    this._delegationFeeRate = value;\n    return this;\n  }\n\n  /**\n   * Set reward addresses where validation rewards should be sent\n   * @param addresses - Array of reward addresses\n   */\n  rewardAddresses(addresses: string[]): this {\n    if (!addresses || addresses.length === 0) {\n      throw new BuildTransactionError('At least one reward address is required');\n    }\n    // Store reward addresses in the transaction (we'll need to extend the type)\n    (this.transaction as unknown as PermissionlessValidatorExtendedTransaction)._rewardAddresses = addresses;\n    return this;\n  }\n\n  /**\n   * Validate that the delegation fee is at least the minDelegationFee\n   * @param delegationFeeRate number\n   */\n  validateDelegationFeeRate(delegationFeeRate: number): void {\n    // For Flare, use a minimum delegation fee of 2% (20000 basis points)\n    const minDelegationFee = MIN_DELEGATION_FEE_BASIS_POINTS;\n    if (delegationFeeRate < minDelegationFee) {\n      const minDelegationFeePercent = (minDelegationFee / BASIS_POINTS_DIVISOR) * PERCENTAGE_MULTIPLIER;\n      throw new BuildTransactionError(\n        `Delegation fee cannot be less than ${minDelegationFee} basis points (${minDelegationFeePercent}%)`\n      );\n    }\n  }\n\n  /** @inheritdoc */\n  initBuilder(tx: Tx): this {\n    // Extract permissionless validator-specific fields from transaction\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const txData = tx as any;\n\n    if (txData.nodeID) {\n      this._nodeID = txData.nodeID;\n    }\n    if (txData.blsPublicKey) {\n      this._blsPublicKey = txData.blsPublicKey;\n    }\n    if (txData.blsSignature) {\n      this._blsSignature = txData.blsSignature;\n    }\n    if (txData.startTime) {\n      this._startTime = BigInt(txData.startTime);\n    }\n    if (txData.endTime) {\n      this._endTime = BigInt(txData.endTime);\n    }\n    if (txData.stakeAmount) {\n      this._stakeAmount = BigInt(txData.stakeAmount);\n    }\n    if (txData.delegationFeeRate !== undefined) {\n      this._delegationFeeRate = txData.delegationFeeRate;\n    }\n    if (txData.rewardAddresses) {\n      (this.transaction as unknown as PermissionlessValidatorExtendedTransaction)._rewardAddresses =\n        txData.rewardAddresses;\n    }\n\n    return this;\n  }\n\n  /**\n   * Verify if the transaction is a permissionless validator transaction\n   * @param tx\n   */\n  static verifyTxType(tx: unknown): boolean {\n    // Check if transaction has permissionless validator-specific properties\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const txData = tx as any;\n    return txData && txData.blsPublicKey && txData.blsSignature;\n  }\n\n  verifyTxType(tx: unknown): boolean {\n    return PermissionlessValidatorTxBuilder.verifyTxType(tx);\n  }\n\n  /**\n   * Build the permissionless validator transaction\n   * @protected\n   */\n  protected async buildFlareTransaction(): Promise<void> {\n    // Basic validation\n    if (!this._nodeID) {\n      throw new BuildTransactionError('Node ID is required for permissionless validator transaction');\n    }\n    if (!this._blsPublicKey) {\n      throw new BuildTransactionError('BLS public key is required for permissionless validator transaction');\n    }\n    if (!this._blsSignature) {\n      throw new BuildTransactionError('BLS signature is required for permissionless validator transaction');\n    }\n    if (!this._startTime) {\n      throw new BuildTransactionError('Start time is required for permissionless validator transaction');\n    }\n    if (!this._endTime) {\n      throw new BuildTransactionError('End time is required for permissionless validator transaction');\n    }\n    if (!this._stakeAmount) {\n      throw new BuildTransactionError('Stake amount is required for permissionless validator transaction');\n    }\n    if (this._delegationFeeRate === undefined) {\n      throw new BuildTransactionError('Delegation fee rate is required for permissionless validator transaction');\n    }\n\n    const rewardAddresses = (this.transaction as unknown as PermissionlessValidatorExtendedTransaction)\n      ._rewardAddresses;\n    if (!rewardAddresses || rewardAddresses.length === 0) {\n      throw new BuildTransactionError('Reward addresses are required for permissionless validator transaction');\n    }\n\n    // Validate time range\n    if (this._endTime <= this._startTime) {\n      throw new BuildTransactionError('End time must be after start time');\n    }\n\n    try {\n      // TODO: Implement actual FlareJS PVM API call when available\n      // For now, create a placeholder transaction structure\n      const validatorTx = {\n        type: ADD_PERMISSIONLESS_VALIDATOR_TYPE,\n        nodeID: this._nodeID,\n        blsPublicKey: this._blsPublicKey,\n        blsSignature: this._blsSignature,\n        startTime: this._startTime,\n        endTime: this._endTime,\n        stakeAmount: this._stakeAmount,\n        delegationFeeRate: this._delegationFeeRate,\n        rewardAddress: rewardAddresses[0],\n        fromAddresses: this.transaction._fromAddresses,\n        networkId: this.transaction._networkID,\n        sourceBlockchainId: this.transaction._blockchainID,\n        threshold: this.transaction._threshold || 1,\n        locktime: this.transaction._locktime || 0n,\n      };\n\n      this.transaction.setTransaction(validatorTx);\n    } catch (error) {\n      throw new BuildTransactionError(\n        `Failed to build permissionless validator transaction: ${\n          error instanceof Error ? error.message : 'Unknown error'\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,UAAU,EAAE,UAAU,EAAE,MAAM,uBAAuB,CAAC;AAC/D,OAAO,EACL,OAAO,EACP,eAAe,EACf,KAAK,EAGL,cAAc,EACd,eAAe,EAChB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAAE,YAAY,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC3E,OAAO,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAChC,OAAO,EAEL,cAAc,EAEd,sBAAsB,EACtB,EAAE,EACF,MAAM,EACN,SAAS,EACV,MAAM,SAAS,CAAC;AACjB,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;
|
|
1
|
+
{"version":3,"file":"transaction.d.ts","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":"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"}
|
|
@@ -8,6 +8,7 @@ const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
|
8
8
|
const buffer_1 = require("buffer");
|
|
9
9
|
const iface_1 = require("./iface");
|
|
10
10
|
const utils_1 = __importDefault(require("./utils"));
|
|
11
|
+
const constants_1 = require("./constants");
|
|
11
12
|
/**
|
|
12
13
|
* Flare P-chain transaction implementation using FlareJS
|
|
13
14
|
* Based on AVAX transaction patterns adapted for Flare network
|
|
@@ -23,7 +24,7 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
23
24
|
this._fee = {};
|
|
24
25
|
this._memo = new Uint8Array(); // FlareJS memo field
|
|
25
26
|
this._network = coinConfig.network;
|
|
26
|
-
this._assetId =
|
|
27
|
+
this._assetId = constants_1.FLR_ASSET_ID; // Default FLR asset
|
|
27
28
|
this._blockchainID = this._network.blockchainID || '';
|
|
28
29
|
this._networkID = this._network.networkID || 0;
|
|
29
30
|
}
|
|
@@ -113,7 +114,7 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
113
114
|
return this._memo.length > 0;
|
|
114
115
|
}
|
|
115
116
|
toHexString(byteArray) {
|
|
116
|
-
return buffer_1.Buffer.from(byteArray).toString(
|
|
117
|
+
return buffer_1.Buffer.from(byteArray).toString(constants_1.HEX_ENCODING);
|
|
117
118
|
}
|
|
118
119
|
/** @inheritdoc */
|
|
119
120
|
toBroadcastFormat() {
|
|
@@ -122,7 +123,7 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
122
123
|
}
|
|
123
124
|
// TODO: Implement FlareJS transaction serialization
|
|
124
125
|
// For now, return placeholder
|
|
125
|
-
return
|
|
126
|
+
return constants_1.FLARE_TX_HEX_PLACEHOLDER;
|
|
126
127
|
}
|
|
127
128
|
toJson() {
|
|
128
129
|
if (!this.flareTransaction) {
|
|
@@ -174,7 +175,7 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
174
175
|
}
|
|
175
176
|
// TODO: Implement FlareJS signable payload extraction
|
|
176
177
|
// For now, return placeholder
|
|
177
|
-
return buffer_1.Buffer.from(
|
|
178
|
+
return buffer_1.Buffer.from(constants_1.FLARE_SIGNABLE_PAYLOAD);
|
|
178
179
|
}
|
|
179
180
|
get id() {
|
|
180
181
|
if (!this.flareTransaction) {
|
|
@@ -182,7 +183,7 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
182
183
|
}
|
|
183
184
|
// TODO: Implement FlareJS transaction ID generation
|
|
184
185
|
// For now, return placeholder
|
|
185
|
-
return
|
|
186
|
+
return constants_1.FLARE_TRANSACTION_ID_PLACEHOLDER;
|
|
186
187
|
}
|
|
187
188
|
get fromAddresses() {
|
|
188
189
|
return this._fromAddresses.map((address) => {
|
|
@@ -212,7 +213,7 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
212
213
|
// TODO: Extract validator outputs from FlareJS transaction
|
|
213
214
|
return [
|
|
214
215
|
{
|
|
215
|
-
address: this._nodeID ||
|
|
216
|
+
address: this._nodeID || constants_1.PLACEHOLDER_NODE_ID,
|
|
216
217
|
value: this._stakeAmount?.toString() || '0',
|
|
217
218
|
},
|
|
218
219
|
];
|
|
@@ -221,7 +222,7 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
221
222
|
// TODO: Extract delegator outputs from FlareJS transaction
|
|
222
223
|
return [
|
|
223
224
|
{
|
|
224
|
-
address: this._nodeID ||
|
|
225
|
+
address: this._nodeID || constants_1.PLACEHOLDER_NODE_ID,
|
|
225
226
|
value: this._stakeAmount?.toString() || '0',
|
|
226
227
|
},
|
|
227
228
|
];
|
|
@@ -255,7 +256,7 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
255
256
|
// TODO: Implement FlareJS signature creation
|
|
256
257
|
// This should use FlareJS signing utilities
|
|
257
258
|
const signval = utils_1.default.createSignature(this._network, this.signablePayload, prv);
|
|
258
|
-
return signval.toString(
|
|
259
|
+
return signval.toString(constants_1.HEX_ENCODING);
|
|
259
260
|
}
|
|
260
261
|
/**
|
|
261
262
|
* Check if transaction is for C-chain (cross-chain)
|
|
@@ -266,10 +267,10 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
266
267
|
/** @inheritdoc */
|
|
267
268
|
explainTransaction() {
|
|
268
269
|
const txJson = this.toJson();
|
|
269
|
-
const displayOrder = [
|
|
270
|
+
const displayOrder = [...constants_1.DISPLAY_ORDER_BASE];
|
|
270
271
|
// Add memo to display order if present
|
|
271
272
|
if (this.hasMemo()) {
|
|
272
|
-
displayOrder.push(
|
|
273
|
+
displayOrder.push(constants_1.MEMO_FIELD);
|
|
273
274
|
}
|
|
274
275
|
// Calculate total output amount
|
|
275
276
|
const outputAmount = txJson.outputs
|
|
@@ -292,11 +293,11 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
292
293
|
];
|
|
293
294
|
if (stakingTypes.includes(txJson.type)) {
|
|
294
295
|
rewardAddresses = this.rewardAddresses;
|
|
295
|
-
displayOrder.splice(6, 0,
|
|
296
|
+
displayOrder.splice(6, 0, constants_1.REWARD_ADDRESSES_FIELD);
|
|
296
297
|
}
|
|
297
298
|
// Add cross-chain information for export/import
|
|
298
299
|
if (this.isTransactionForCChain) {
|
|
299
|
-
displayOrder.push(
|
|
300
|
+
displayOrder.push(constants_1.SOURCE_CHAIN_FIELD, constants_1.DESTINATION_CHAIN_FIELD);
|
|
300
301
|
}
|
|
301
302
|
const explanation = {
|
|
302
303
|
displayOrder,
|
|
@@ -318,4 +319,4 @@ class Transaction extends sdk_core_1.BaseTransaction {
|
|
|
318
319
|
}
|
|
319
320
|
}
|
|
320
321
|
exports.Transaction = Transaction;
|
|
321
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":"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","sourcesContent":["import { UnsignedTx, Credential } from '@flarenetwork/flarejs';\nimport {\n  BaseKey,\n  BaseTransaction,\n  Entry,\n  InvalidTransactionError,\n  SigningError,\n  TransactionFee,\n  TransactionType,\n} from '@bitgo-beta/sdk-core';\nimport { FlareNetwork, BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { Buffer } from 'buffer';\nimport {\n  ADDRESS_SEPARATOR,\n  DecodedUtxoObj,\n  INPUT_SEPARATOR,\n  TransactionExplanation,\n  Tx,\n  TxData,\n  FlrpEntry,\n} from './iface';\nimport { KeyPair } from './keyPair';\nimport utils from './utils';\n\n/**\n * Flare P-chain transaction implementation using FlareJS\n * Based on AVAX transaction patterns adapted for Flare network\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: string[] = [];\n  public _rewardAddresses: string[] = [];\n  public _utxos: DecodedUtxoObj[] = [];\n  public _to: string[];\n  public _fee: Partial<TransactionFee> = {};\n  public _blsPublicKey: string;\n  public _blsSignature: string;\n  public _memo: Uint8Array = new Uint8Array(); // FlareJS memo field\n\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    this._network = coinConfig.network as FlareNetwork;\n    this._assetId = 'FLR'; // Default FLR asset\n    this._blockchainID = this._network.blockchainID || '';\n    this._networkID = this._network.networkID || 0;\n  }\n\n  /**\n   * Get the base transaction from FlareJS UnsignedTx\n   * TODO: Implement proper FlareJS transaction extraction\n   */\n  get flareTransaction(): UnsignedTx {\n    return this._flareTransaction as UnsignedTx;\n  }\n\n  get signature(): string[] {\n    if (this.credentials.length === 0) {\n      return [];\n    }\n    // TODO: Extract signatures from FlareJS credentials\n    // For now, return placeholder\n    return [];\n  }\n\n  get credentials(): Credential[] {\n    // TODO: Extract credentials from FlareJS transaction\n    // For now, return empty array\n    return [];\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    // TODO: Implement proper signing validation for FlareJS\n    return true;\n  }\n\n  /**\n   * Sign a Flare transaction using FlareJS\n   * @param {KeyPair} keyPair\n   */\n  async sign(keyPair: KeyPair): Promise<void> {\n    const prv = keyPair.getPrivateKey() as Buffer;\n\n    if (!prv) {\n      throw new SigningError('Missing private key');\n    }\n\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('empty transaction to sign');\n    }\n\n    if (!this.hasCredentials) {\n      throw new InvalidTransactionError('empty credentials to sign');\n    }\n\n    // TODO: Implement FlareJS signing process\n    // This will involve:\n    // 1. Creating FlareJS signature using private key\n    // 2. Attaching signature to appropriate credential\n    // 3. Updating transaction with signed credentials\n\n    throw new Error('FlareJS signing not yet implemented - placeholder');\n  }\n\n  /**\n   * Set memo from string\n   * @param {string} memo - Memo text\n   */\n  setMemo(memo: string): void {\n    this._memo = utils.stringToBytes(memo);\n  }\n\n  /**\n   * Set memo from various formats\n   * @param {string | Record<string, unknown> | Uint8Array} memo - Memo data\n   */\n  setMemoData(memo: string | Record<string, unknown> | Uint8Array): void {\n    this._memo = utils.createMemoBytes(memo);\n  }\n\n  /**\n   * Get memo as bytes (FlareJS format)\n   * @returns {Uint8Array} Memo bytes\n   */\n  getMemoBytes(): Uint8Array {\n    return this._memo;\n  }\n\n  /**\n   * Get memo as string\n   * @returns {string} Memo string\n   */\n  getMemoString(): string {\n    return utils.parseMemoBytes(this._memo);\n  }\n\n  /**\n   * Check if transaction has memo\n   * @returns {boolean} Whether memo exists and is not empty\n   */\n  hasMemo(): boolean {\n    return this._memo.length > 0;\n  }\n\n  toHexString(byteArray: Uint8Array): string {\n    return Buffer.from(byteArray).toString('hex');\n  }\n\n  /** @inheritdoc */\n  toBroadcastFormat(): string {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\n\n    // TODO: Implement FlareJS transaction serialization\n    // For now, return placeholder\n    return 'flare-tx-hex-placeholder';\n  }\n\n  toJson(): TxData {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\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._network.blockchainID,\n      destinationChain: this._network.cChainBlockchainID,\n      memo: this.getMemoString(), // Include memo in JSON representation\n    };\n  }\n\n  setTransaction(tx: Tx): void {\n    this._flareTransaction = tx;\n  }\n\n  /**\n   * Set the transaction type\n   * @param {TransactionType} transactionType The transaction type to be set\n   */\n  setTransactionType(transactionType: TransactionType): void {\n    const supportedTypes = [\n      TransactionType.Export,\n      TransactionType.Import,\n      TransactionType.AddValidator,\n      TransactionType.AddDelegator,\n      TransactionType.AddPermissionlessValidator,\n      TransactionType.AddPermissionlessDelegator,\n    ];\n\n    if (!supportedTypes.includes(transactionType)) {\n      throw new Error(`Transaction type ${transactionType} is not supported`);\n    }\n    this._type = transactionType;\n  }\n\n  /**\n   * Returns the portion of the transaction that needs to be signed in Buffer format.\n   * Only needed for coins that support adding signatures directly (e.g. TSS).\n   */\n  get signablePayload(): Buffer {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction for signing');\n    }\n\n    // TODO: Implement FlareJS signable payload extraction\n    // For now, return placeholder\n    return Buffer.from('flare-signable-payload');\n  }\n\n  get id(): string {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction for ID generation');\n    }\n\n    // TODO: Implement FlareJS transaction ID generation\n    // For now, return placeholder\n    return 'flare-transaction-id-placeholder';\n  }\n\n  get fromAddresses(): string[] {\n    return this._fromAddresses.map((address) => {\n      // TODO: Format addresses using FlareJS utilities\n      return address;\n    });\n  }\n\n  get rewardAddresses(): string[] {\n    return this._rewardAddresses.map((address) => {\n      // TODO: Format addresses using FlareJS utilities\n      return address;\n    });\n  }\n\n  /**\n   * Get the list of outputs. Amounts are expressed in absolute value.\n   */\n  get outputs(): Entry[] {\n    switch (this.type) {\n      case TransactionType.Export:\n        // TODO: Extract export outputs from FlareJS transaction\n        return [];\n      case TransactionType.Import:\n        // TODO: Extract import outputs from FlareJS transaction\n        return [];\n      case TransactionType.AddValidator:\n      case TransactionType.AddPermissionlessValidator:\n        // TODO: Extract validator outputs from FlareJS transaction\n        return [\n          {\n            address: this._nodeID || 'placeholder-node-id',\n            value: this._stakeAmount?.toString() || '0',\n          },\n        ];\n      case TransactionType.AddDelegator:\n      case TransactionType.AddPermissionlessDelegator:\n        // TODO: Extract delegator outputs from FlareJS transaction\n        return [\n          {\n            address: this._nodeID || 'placeholder-node-id',\n            value: this._stakeAmount?.toString() || '0',\n          },\n        ];\n      default:\n        return [];\n    }\n  }\n\n  get fee(): TransactionFee {\n    return { fee: '0', ...this._fee };\n  }\n\n  get changeOutputs(): Entry[] {\n    // TODO: Extract change outputs from FlareJS transaction\n    // For now, return empty array\n    return [];\n  }\n\n  get inputs(): FlrpEntry[] {\n    // TODO: Extract inputs from FlareJS transaction\n    // For now, return placeholder based on UTXOs\n    return this._utxos.map((utxo) => ({\n      id: utxo.txid + INPUT_SEPARATOR + utxo.outputidx,\n      address: this.fromAddresses.sort().join(ADDRESS_SEPARATOR),\n      value: utxo.amount,\n    }));\n  }\n\n  /**\n   * Flare wrapper to create signature and return it for credentials\n   * @param prv\n   * @return hexstring\n   */\n  createSignature(prv: Buffer): string {\n    // TODO: Implement FlareJS signature creation\n    // This should use FlareJS signing utilities\n    const signval = utils.createSignature(this._network, this.signablePayload, prv);\n    return signval.toString('hex');\n  }\n\n  /**\n   * Check if transaction is for C-chain (cross-chain)\n   */\n  get isTransactionForCChain(): boolean {\n    return this.type === TransactionType.Export || this.type === TransactionType.Import;\n  }\n\n  /** @inheritdoc */\n  explainTransaction(): TransactionExplanation {\n    const txJson = this.toJson();\n    const displayOrder = ['id', 'inputs', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'type'];\n\n    // Add memo to display order if present\n    if (this.hasMemo()) {\n      displayOrder.push('memo');\n    }\n\n    // Calculate total output amount\n    const outputAmount = txJson.outputs\n      .reduce((sum, output) => {\n        return sum + BigInt(output.value || '0');\n      }, BigInt(0))\n      .toString();\n\n    // Calculate total change amount\n    const changeAmount = txJson.changeOutputs\n      .reduce((sum, output) => {\n        return sum + BigInt(output.value || '0');\n      }, BigInt(0))\n      .toString();\n\n    let rewardAddresses;\n    const stakingTypes = [\n      TransactionType.AddValidator,\n      TransactionType.AddDelegator,\n      TransactionType.AddPermissionlessValidator,\n      TransactionType.AddPermissionlessDelegator,\n    ];\n\n    if (stakingTypes.includes(txJson.type)) {\n      rewardAddresses = this.rewardAddresses;\n      displayOrder.splice(6, 0, 'rewardAddresses');\n    }\n\n    // Add cross-chain information for export/import\n    if (this.isTransactionForCChain) {\n      displayOrder.push('sourceChain', 'destinationChain');\n    }\n\n    const explanation: TransactionExplanation & { memo?: string } = {\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    // Add memo to explanation if present\n    if (this.hasMemo()) {\n      explanation.memo = this.getMemoString();\n    }\n\n    return explanation;\n  }\n}\n"]}
|
|
322
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":";;;;;;AACA,mDAQ8B;AAE9B,mCAAgC;AAChC,mCAQiB;AAEjB,oDAA4B;AAC5B,2CAYqB;AAErB;;;GAGG;AACH,MAAa,WAAY,SAAQ,0BAAe;IAsB9C,YAAY,UAAgC;QAC1C,KAAK,CAAC,UAAU,CAAC,CAAC;QAZb,eAAU,GAAG,CAAC,CAAC;QACf,cAAS,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;QACtB,mBAAc,GAAa,EAAE,CAAC;QAC9B,qBAAgB,GAAa,EAAE,CAAC;QAChC,WAAM,GAAqB,EAAE,CAAC;QAE9B,SAAI,GAA4B,EAAE,CAAC;QAGnC,UAAK,GAAe,IAAI,UAAU,EAAE,CAAC,CAAC,qBAAqB;QAIhE,IAAI,CAAC,QAAQ,GAAG,UAAU,CAAC,OAAuB,CAAC;QACnD,IAAI,CAAC,QAAQ,GAAG,wBAAY,CAAC,CAAC,oBAAoB;QAClD,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,IAAI,EAAE,CAAC;QACtD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,SAAS,IAAI,CAAC,CAAC;IACjD,CAAC;IAED;;;OAGG;IACH,IAAI,gBAAgB;QAClB,OAAO,IAAI,CAAC,iBAA+B,CAAC;IAC9C,CAAC;IAED,IAAI,SAAS;QACX,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAClC,OAAO,EAAE,CAAC;QACZ,CAAC;QACD,oDAAoD;QACpD,8BAA8B;QAC9B,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,IAAI,WAAW;QACb,qDAAqD;QACrD,8BAA8B;QAC9B,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,IAAI,cAAc;QAChB,OAAO,IAAI,CAAC,WAAW,KAAK,SAAS,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC;IACvE,CAAC;IAED,kBAAkB;IAClB,OAAO,CAAC,EAAE,GAAG,EAAW;QACtB,wDAAwD;QACxD,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,IAAI,CAAC,OAAgB;QACzB,MAAM,GAAG,GAAG,OAAO,CAAC,aAAa,EAAY,CAAC;QAE9C,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,MAAM,IAAI,uBAAY,CAAC,qBAAqB,CAAC,CAAC;QAChD,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC3B,MAAM,IAAI,kCAAuB,CAAC,2BAA2B,CAAC,CAAC;QACjE,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC;YACzB,MAAM,IAAI,kCAAuB,CAAC,2BAA2B,CAAC,CAAC;QACjE,CAAC;QAED,0CAA0C;QAC1C,qBAAqB;QACrB,kDAAkD;QAClD,mDAAmD;QACnD,kDAAkD;QAElD,MAAM,IAAI,KAAK,CAAC,mDAAmD,CAAC,CAAC;IACvE,CAAC;IAED;;;OAGG;IACH,OAAO,CAAC,IAAY;QAClB,IAAI,CAAC,KAAK,GAAG,eAAK,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;IACzC,CAAC;IAED;;;OAGG;IACH,WAAW,CAAC,IAAmD;QAC7D,IAAI,CAAC,KAAK,GAAG,eAAK,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;IAC3C,CAAC;IAED;;;OAGG;IACH,YAAY;QACV,OAAO,IAAI,CAAC,KAAK,CAAC;IACpB,CAAC;IAED;;;OAGG;IACH,aAAa;QACX,OAAO,eAAK,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC1C,CAAC;IAED;;;OAGG;IACH,OAAO;QACL,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC;IAC/B,CAAC;IAED,WAAW,CAAC,SAAqB;QAC/B,OAAO,eAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,QAAQ,CAAC,wBAAY,CAAC,CAAC;IACvD,CAAC;IAED,kBAAkB;IAClB,iBAAiB;QACf,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC3B,MAAM,IAAI,kCAAuB,CAAC,wBAAwB,CAAC,CAAC;QAC9D,CAAC;QAED,oDAAoD;QACpD,8BAA8B;QAC9B,OAAO,oCAAwB,CAAC;IAClC,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC3B,MAAM,IAAI,kCAAuB,CAAC,wBAAwB,CAAC,CAAC;QAC9D,CAAC;QAED,OAAO;YACL,EAAE,EAAE,IAAI,CAAC,EAAE;YACX,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,SAAS,EAAE,IAAI,CAAC,UAAU;YAC1B,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE;YACnC,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,UAAU,EAAE,IAAI,CAAC,SAAS;YAC1B,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,WAAW,EAAE,IAAI,CAAC,QAAQ,CAAC,YAAY;YACvC,gBAAgB,EAAE,IAAI,CAAC,QAAQ,CAAC,kBAAkB;YAClD,IAAI,EAAE,IAAI,CAAC,aAAa,EAAE,EAAE,sCAAsC;SACnE,CAAC;IACJ,CAAC;IAED,cAAc,CAAC,EAAM;QACnB,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;IAC9B,CAAC;IAED;;;OAGG;IACH,kBAAkB,CAAC,eAAgC;QACjD,MAAM,cAAc,GAAG;YACrB,0BAAe,CAAC,MAAM;YACtB,0BAAe,CAAC,MAAM;YACtB,0BAAe,CAAC,YAAY;YAC5B,0BAAe,CAAC,YAAY;YAC5B,0BAAe,CAAC,0BAA0B;YAC1C,0BAAe,CAAC,0BAA0B;SAC3C,CAAC;QAEF,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,eAAe,CAAC,EAAE,CAAC;YAC9C,MAAM,IAAI,KAAK,CAAC,oBAAoB,eAAe,mBAAmB,CAAC,CAAC;QAC1E,CAAC;QACD,IAAI,CAAC,KAAK,GAAG,eAAe,CAAC;IAC/B,CAAC;IAED;;;OAGG;IACH,IAAI,eAAe;QACjB,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC3B,MAAM,IAAI,kCAAuB,CAAC,+BAA+B,CAAC,CAAC;QACrE,CAAC;QAED,sDAAsD;QACtD,8BAA8B;QAC9B,OAAO,eAAM,CAAC,IAAI,CAAC,kCAAsB,CAAC,CAAC;IAC7C,CAAC;IAED,IAAI,EAAE;QACJ,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC3B,MAAM,IAAI,kCAAuB,CAAC,qCAAqC,CAAC,CAAC;QAC3E,CAAC;QAED,oDAAoD;QACpD,8BAA8B;QAC9B,OAAO,4CAAgC,CAAC;IAC1C,CAAC;IAED,IAAI,aAAa;QACf,OAAO,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;YACzC,iDAAiD;YACjD,OAAO,OAAO,CAAC;QACjB,CAAC,CAAC,CAAC;IACL,CAAC;IAED,IAAI,eAAe;QACjB,OAAO,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;YAC3C,iDAAiD;YACjD,OAAO,OAAO,CAAC;QACjB,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,IAAI,OAAO;QACT,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;YAClB,KAAK,0BAAe,CAAC,MAAM;gBACzB,wDAAwD;gBACxD,OAAO,EAAE,CAAC;YACZ,KAAK,0BAAe,CAAC,MAAM;gBACzB,wDAAwD;gBACxD,OAAO,EAAE,CAAC;YACZ,KAAK,0BAAe,CAAC,YAAY,CAAC;YAClC,KAAK,0BAAe,CAAC,0BAA0B;gBAC7C,2DAA2D;gBAC3D,OAAO;oBACL;wBACE,OAAO,EAAE,IAAI,CAAC,OAAO,IAAI,+BAAmB;wBAC5C,KAAK,EAAE,IAAI,CAAC,YAAY,EAAE,QAAQ,EAAE,IAAI,GAAG;qBAC5C;iBACF,CAAC;YACJ,KAAK,0BAAe,CAAC,YAAY,CAAC;YAClC,KAAK,0BAAe,CAAC,0BAA0B;gBAC7C,2DAA2D;gBAC3D,OAAO;oBACL;wBACE,OAAO,EAAE,IAAI,CAAC,OAAO,IAAI,+BAAmB;wBAC5C,KAAK,EAAE,IAAI,CAAC,YAAY,EAAE,QAAQ,EAAE,IAAI,GAAG;qBAC5C;iBACF,CAAC;YACJ;gBACE,OAAO,EAAE,CAAC;QACd,CAAC;IACH,CAAC;IAED,IAAI,GAAG;QACL,OAAO,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;IACpC,CAAC;IAED,IAAI,aAAa;QACf,wDAAwD;QACxD,8BAA8B;QAC9B,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,IAAI,MAAM;QACR,gDAAgD;QAChD,6CAA6C;QAC7C,OAAO,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;YAChC,EAAE,EAAE,IAAI,CAAC,IAAI,GAAG,uBAAe,GAAG,IAAI,CAAC,SAAS;YAChD,OAAO,EAAE,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC,IAAI,CAAC,yBAAiB,CAAC;YAC1D,KAAK,EAAE,IAAI,CAAC,MAAM;SACnB,CAAC,CAAC,CAAC;IACN,CAAC;IAED;;;;OAIG;IACH,eAAe,CAAC,GAAW;QACzB,6CAA6C;QAC7C,4CAA4C;QAC5C,MAAM,OAAO,GAAG,eAAK,CAAC,eAAe,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,eAAe,EAAE,GAAG,CAAC,CAAC;QAChF,OAAO,OAAO,CAAC,QAAQ,CAAC,wBAAY,CAAC,CAAC;IACxC,CAAC;IAED;;OAEG;IACH,IAAI,sBAAsB;QACxB,OAAO,IAAI,CAAC,IAAI,KAAK,0BAAe,CAAC,MAAM,IAAI,IAAI,CAAC,IAAI,KAAK,0BAAe,CAAC,MAAM,CAAC;IACtF,CAAC;IAED,kBAAkB;IAClB,kBAAkB;QAChB,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;QAC7B,MAAM,YAAY,GAAG,CAAC,GAAG,8BAAkB,CAAC,CAAC;QAE7C,uCAAuC;QACvC,IAAI,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC;YACnB,YAAY,CAAC,IAAI,CAAC,sBAAU,CAAC,CAAC;QAChC,CAAC;QAED,gCAAgC;QAChC,MAAM,YAAY,GAAG,MAAM,CAAC,OAAO;aAChC,MAAM,CAAC,CAAC,GAAG,EAAE,MAAM,EAAE,EAAE;YACtB,OAAO,GAAG,GAAG,MAAM,CAAC,MAAM,CAAC,KAAK,IAAI,GAAG,CAAC,CAAC;QAC3C,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC;aACZ,QAAQ,EAAE,CAAC;QAEd,gCAAgC;QAChC,MAAM,YAAY,GAAG,MAAM,CAAC,aAAa;aACtC,MAAM,CAAC,CAAC,GAAG,EAAE,MAAM,EAAE,EAAE;YACtB,OAAO,GAAG,GAAG,MAAM,CAAC,MAAM,CAAC,KAAK,IAAI,GAAG,CAAC,CAAC;QAC3C,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC;aACZ,QAAQ,EAAE,CAAC;QAEd,IAAI,eAAe,CAAC;QACpB,MAAM,YAAY,GAAG;YACnB,0BAAe,CAAC,YAAY;YAC5B,0BAAe,CAAC,YAAY;YAC5B,0BAAe,CAAC,0BAA0B;YAC1C,0BAAe,CAAC,0BAA0B;SAC3C,CAAC;QAEF,IAAI,YAAY,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC;YACvC,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC;YACvC,YAAY,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,EAAE,kCAAsB,CAAC,CAAC;QACpD,CAAC;QAED,gDAAgD;QAChD,IAAI,IAAI,CAAC,sBAAsB,EAAE,CAAC;YAChC,YAAY,CAAC,IAAI,CAAC,8BAAkB,EAAE,mCAAuB,CAAC,CAAC;QACjE,CAAC;QAED,MAAM,WAAW,GAA+C;YAC9D,YAAY;YACZ,EAAE,EAAE,MAAM,CAAC,EAAE;YACb,MAAM,EAAE,MAAM,CAAC,MAAM;YACrB,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;YAC7E,YAAY;YACZ,aAAa,EAAE,MAAM,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;YACzF,YAAY;YACZ,eAAe;YACf,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,IAAI,EAAE,MAAM,CAAC,IAAI;SAClB,CAAC;QAEF,qCAAqC;QACrC,IAAI,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC;YACnB,WAAW,CAAC,IAAI,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QAC1C,CAAC;QAED,OAAO,WAAW,CAAC;IACrB,CAAC;CACF;AA5WD,kCA4WC","sourcesContent":["import { UnsignedTx, Credential } from '@flarenetwork/flarejs';\nimport {\n  BaseKey,\n  BaseTransaction,\n  Entry,\n  InvalidTransactionError,\n  SigningError,\n  TransactionFee,\n  TransactionType,\n} from '@bitgo-beta/sdk-core';\nimport { FlareNetwork, BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { Buffer } from 'buffer';\nimport {\n  ADDRESS_SEPARATOR,\n  DecodedUtxoObj,\n  INPUT_SEPARATOR,\n  TransactionExplanation,\n  Tx,\n  TxData,\n  FlrpEntry,\n} from './iface';\nimport { KeyPair } from './keyPair';\nimport utils from './utils';\nimport {\n  FLR_ASSET_ID,\n  FLARE_TX_HEX_PLACEHOLDER,\n  FLARE_SIGNABLE_PAYLOAD,\n  FLARE_TRANSACTION_ID_PLACEHOLDER,\n  PLACEHOLDER_NODE_ID,\n  HEX_ENCODING,\n  MEMO_FIELD,\n  DISPLAY_ORDER_BASE,\n  REWARD_ADDRESSES_FIELD,\n  SOURCE_CHAIN_FIELD,\n  DESTINATION_CHAIN_FIELD,\n} from './constants';\n\n/**\n * Flare P-chain transaction implementation using FlareJS\n * Based on AVAX transaction patterns adapted for Flare network\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: string[] = [];\n  public _rewardAddresses: string[] = [];\n  public _utxos: DecodedUtxoObj[] = [];\n  public _to: string[];\n  public _fee: Partial<TransactionFee> = {};\n  public _blsPublicKey: string;\n  public _blsSignature: string;\n  public _memo: Uint8Array = new Uint8Array(); // FlareJS memo field\n\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    this._network = coinConfig.network as FlareNetwork;\n    this._assetId = FLR_ASSET_ID; // Default FLR asset\n    this._blockchainID = this._network.blockchainID || '';\n    this._networkID = this._network.networkID || 0;\n  }\n\n  /**\n   * Get the base transaction from FlareJS UnsignedTx\n   * TODO: Implement proper FlareJS transaction extraction\n   */\n  get flareTransaction(): UnsignedTx {\n    return this._flareTransaction as UnsignedTx;\n  }\n\n  get signature(): string[] {\n    if (this.credentials.length === 0) {\n      return [];\n    }\n    // TODO: Extract signatures from FlareJS credentials\n    // For now, return placeholder\n    return [];\n  }\n\n  get credentials(): Credential[] {\n    // TODO: Extract credentials from FlareJS transaction\n    // For now, return empty array\n    return [];\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    // TODO: Implement proper signing validation for FlareJS\n    return true;\n  }\n\n  /**\n   * Sign a Flare transaction using FlareJS\n   * @param {KeyPair} keyPair\n   */\n  async sign(keyPair: KeyPair): Promise<void> {\n    const prv = keyPair.getPrivateKey() as Buffer;\n\n    if (!prv) {\n      throw new SigningError('Missing private key');\n    }\n\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('empty transaction to sign');\n    }\n\n    if (!this.hasCredentials) {\n      throw new InvalidTransactionError('empty credentials to sign');\n    }\n\n    // TODO: Implement FlareJS signing process\n    // This will involve:\n    // 1. Creating FlareJS signature using private key\n    // 2. Attaching signature to appropriate credential\n    // 3. Updating transaction with signed credentials\n\n    throw new Error('FlareJS signing not yet implemented - placeholder');\n  }\n\n  /**\n   * Set memo from string\n   * @param {string} memo - Memo text\n   */\n  setMemo(memo: string): void {\n    this._memo = utils.stringToBytes(memo);\n  }\n\n  /**\n   * Set memo from various formats\n   * @param {string | Record<string, unknown> | Uint8Array} memo - Memo data\n   */\n  setMemoData(memo: string | Record<string, unknown> | Uint8Array): void {\n    this._memo = utils.createMemoBytes(memo);\n  }\n\n  /**\n   * Get memo as bytes (FlareJS format)\n   * @returns {Uint8Array} Memo bytes\n   */\n  getMemoBytes(): Uint8Array {\n    return this._memo;\n  }\n\n  /**\n   * Get memo as string\n   * @returns {string} Memo string\n   */\n  getMemoString(): string {\n    return utils.parseMemoBytes(this._memo);\n  }\n\n  /**\n   * Check if transaction has memo\n   * @returns {boolean} Whether memo exists and is not empty\n   */\n  hasMemo(): boolean {\n    return this._memo.length > 0;\n  }\n\n  toHexString(byteArray: Uint8Array): string {\n    return Buffer.from(byteArray).toString(HEX_ENCODING);\n  }\n\n  /** @inheritdoc */\n  toBroadcastFormat(): string {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\n\n    // TODO: Implement FlareJS transaction serialization\n    // For now, return placeholder\n    return FLARE_TX_HEX_PLACEHOLDER;\n  }\n\n  toJson(): TxData {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\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._network.blockchainID,\n      destinationChain: this._network.cChainBlockchainID,\n      memo: this.getMemoString(), // Include memo in JSON representation\n    };\n  }\n\n  setTransaction(tx: Tx): void {\n    this._flareTransaction = tx;\n  }\n\n  /**\n   * Set the transaction type\n   * @param {TransactionType} transactionType The transaction type to be set\n   */\n  setTransactionType(transactionType: TransactionType): void {\n    const supportedTypes = [\n      TransactionType.Export,\n      TransactionType.Import,\n      TransactionType.AddValidator,\n      TransactionType.AddDelegator,\n      TransactionType.AddPermissionlessValidator,\n      TransactionType.AddPermissionlessDelegator,\n    ];\n\n    if (!supportedTypes.includes(transactionType)) {\n      throw new Error(`Transaction type ${transactionType} is not supported`);\n    }\n    this._type = transactionType;\n  }\n\n  /**\n   * Returns the portion of the transaction that needs to be signed in Buffer format.\n   * Only needed for coins that support adding signatures directly (e.g. TSS).\n   */\n  get signablePayload(): Buffer {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction for signing');\n    }\n\n    // TODO: Implement FlareJS signable payload extraction\n    // For now, return placeholder\n    return Buffer.from(FLARE_SIGNABLE_PAYLOAD);\n  }\n\n  get id(): string {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction for ID generation');\n    }\n\n    // TODO: Implement FlareJS transaction ID generation\n    // For now, return placeholder\n    return FLARE_TRANSACTION_ID_PLACEHOLDER;\n  }\n\n  get fromAddresses(): string[] {\n    return this._fromAddresses.map((address) => {\n      // TODO: Format addresses using FlareJS utilities\n      return address;\n    });\n  }\n\n  get rewardAddresses(): string[] {\n    return this._rewardAddresses.map((address) => {\n      // TODO: Format addresses using FlareJS utilities\n      return address;\n    });\n  }\n\n  /**\n   * Get the list of outputs. Amounts are expressed in absolute value.\n   */\n  get outputs(): Entry[] {\n    switch (this.type) {\n      case TransactionType.Export:\n        // TODO: Extract export outputs from FlareJS transaction\n        return [];\n      case TransactionType.Import:\n        // TODO: Extract import outputs from FlareJS transaction\n        return [];\n      case TransactionType.AddValidator:\n      case TransactionType.AddPermissionlessValidator:\n        // TODO: Extract validator outputs from FlareJS transaction\n        return [\n          {\n            address: this._nodeID || PLACEHOLDER_NODE_ID,\n            value: this._stakeAmount?.toString() || '0',\n          },\n        ];\n      case TransactionType.AddDelegator:\n      case TransactionType.AddPermissionlessDelegator:\n        // TODO: Extract delegator outputs from FlareJS transaction\n        return [\n          {\n            address: this._nodeID || PLACEHOLDER_NODE_ID,\n            value: this._stakeAmount?.toString() || '0',\n          },\n        ];\n      default:\n        return [];\n    }\n  }\n\n  get fee(): TransactionFee {\n    return { fee: '0', ...this._fee };\n  }\n\n  get changeOutputs(): Entry[] {\n    // TODO: Extract change outputs from FlareJS transaction\n    // For now, return empty array\n    return [];\n  }\n\n  get inputs(): FlrpEntry[] {\n    // TODO: Extract inputs from FlareJS transaction\n    // For now, return placeholder based on UTXOs\n    return this._utxos.map((utxo) => ({\n      id: utxo.txid + INPUT_SEPARATOR + utxo.outputidx,\n      address: this.fromAddresses.sort().join(ADDRESS_SEPARATOR),\n      value: utxo.amount,\n    }));\n  }\n\n  /**\n   * Flare wrapper to create signature and return it for credentials\n   * @param prv\n   * @return hexstring\n   */\n  createSignature(prv: Buffer): string {\n    // TODO: Implement FlareJS signature creation\n    // This should use FlareJS signing utilities\n    const signval = utils.createSignature(this._network, this.signablePayload, prv);\n    return signval.toString(HEX_ENCODING);\n  }\n\n  /**\n   * Check if transaction is for C-chain (cross-chain)\n   */\n  get isTransactionForCChain(): boolean {\n    return this.type === TransactionType.Export || this.type === TransactionType.Import;\n  }\n\n  /** @inheritdoc */\n  explainTransaction(): TransactionExplanation {\n    const txJson = this.toJson();\n    const displayOrder = [...DISPLAY_ORDER_BASE];\n\n    // Add memo to display order if present\n    if (this.hasMemo()) {\n      displayOrder.push(MEMO_FIELD);\n    }\n\n    // Calculate total output amount\n    const outputAmount = txJson.outputs\n      .reduce((sum, output) => {\n        return sum + BigInt(output.value || '0');\n      }, BigInt(0))\n      .toString();\n\n    // Calculate total change amount\n    const changeAmount = txJson.changeOutputs\n      .reduce((sum, output) => {\n        return sum + BigInt(output.value || '0');\n      }, BigInt(0))\n      .toString();\n\n    let rewardAddresses;\n    const stakingTypes = [\n      TransactionType.AddValidator,\n      TransactionType.AddDelegator,\n      TransactionType.AddPermissionlessValidator,\n      TransactionType.AddPermissionlessDelegator,\n    ];\n\n    if (stakingTypes.includes(txJson.type)) {\n      rewardAddresses = this.rewardAddresses;\n      displayOrder.splice(6, 0, REWARD_ADDRESSES_FIELD);\n    }\n\n    // Add cross-chain information for export/import\n    if (this.isTransactionForCChain) {\n      displayOrder.push(SOURCE_CHAIN_FIELD, DESTINATION_CHAIN_FIELD);\n    }\n\n    const explanation: TransactionExplanation & { memo?: string } = {\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    // Add memo to explanation if present\n    if (this.hasMemo()) {\n      explanation.memo = this.getMemoString();\n    }\n\n    return explanation;\n  }\n}\n"]}
|