@bitgo-beta/sdk-coin-flrp 0.0.0-semantic-release-managed

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.
Files changed (76) hide show
  1. package/.eslintignore +5 -0
  2. package/.eslintrc.json +7 -0
  3. package/.mocharc.yml +8 -0
  4. package/CHANGELOG.md +0 -0
  5. package/LICENSE +191 -0
  6. package/dist/src/flrp.d.ts +91 -0
  7. package/dist/src/flrp.d.ts.map +1 -0
  8. package/dist/src/flrp.js +343 -0
  9. package/dist/src/iface.d.ts +25 -0
  10. package/dist/src/iface.d.ts.map +1 -0
  11. package/dist/src/iface.js +3 -0
  12. package/dist/src/index.d.ts +6 -0
  13. package/dist/src/index.d.ts.map +1 -0
  14. package/dist/src/index.js +45 -0
  15. package/dist/src/lib/atomicInCTransactionBuilder.d.ts +26 -0
  16. package/dist/src/lib/atomicInCTransactionBuilder.d.ts.map +1 -0
  17. package/dist/src/lib/atomicInCTransactionBuilder.js +64 -0
  18. package/dist/src/lib/atomicTransactionBuilder.d.ts +93 -0
  19. package/dist/src/lib/atomicTransactionBuilder.d.ts.map +1 -0
  20. package/dist/src/lib/atomicTransactionBuilder.js +252 -0
  21. package/dist/src/lib/constants.d.ts +11 -0
  22. package/dist/src/lib/constants.d.ts.map +1 -0
  23. package/dist/src/lib/constants.js +17 -0
  24. package/dist/src/lib/errors.d.ts +8 -0
  25. package/dist/src/lib/errors.d.ts.map +1 -0
  26. package/dist/src/lib/errors.js +19 -0
  27. package/dist/src/lib/exportInCTxBuilder.d.ts +77 -0
  28. package/dist/src/lib/exportInCTxBuilder.d.ts.map +1 -0
  29. package/dist/src/lib/exportInCTxBuilder.js +170 -0
  30. package/dist/src/lib/exportInPTxBuilder.d.ts +30 -0
  31. package/dist/src/lib/exportInPTxBuilder.d.ts.map +1 -0
  32. package/dist/src/lib/exportInPTxBuilder.js +56 -0
  33. package/dist/src/lib/iface.d.ts +119 -0
  34. package/dist/src/lib/iface.d.ts.map +1 -0
  35. package/dist/src/lib/iface.js +22 -0
  36. package/dist/src/lib/index.d.ts +7 -0
  37. package/dist/src/lib/index.d.ts.map +1 -0
  38. package/dist/src/lib/index.js +30 -0
  39. package/dist/src/lib/keyPair.d.ts +58 -0
  40. package/dist/src/lib/keyPair.d.ts.map +1 -0
  41. package/dist/src/lib/keyPair.js +142 -0
  42. package/dist/src/lib/transaction.d.ts +111 -0
  43. package/dist/src/lib/transaction.d.ts.map +1 -0
  44. package/dist/src/lib/transaction.js +321 -0
  45. package/dist/src/lib/transactionBuilderFactory.d.ts +37 -0
  46. package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -0
  47. package/dist/src/lib/transactionBuilderFactory.js +91 -0
  48. package/dist/src/lib/utils.d.ts +215 -0
  49. package/dist/src/lib/utils.d.ts.map +1 -0
  50. package/dist/src/lib/utils.js +487 -0
  51. package/dist/src/register.d.ts +3 -0
  52. package/dist/src/register.d.ts.map +1 -0
  53. package/dist/src/register.js +11 -0
  54. package/dist/src/tflrp.d.ts +8 -0
  55. package/dist/src/tflrp.d.ts.map +1 -0
  56. package/dist/src/tflrp.js +14 -0
  57. package/dist/test/unit/flrp.d.ts +2 -0
  58. package/dist/test/unit/flrp.d.ts.map +1 -0
  59. package/dist/test/unit/flrp.js +118 -0
  60. package/dist/test/unit/lib/atomicTransactionBuilder.d.ts +2 -0
  61. package/dist/test/unit/lib/atomicTransactionBuilder.d.ts.map +1 -0
  62. package/dist/test/unit/lib/atomicTransactionBuilder.js +222 -0
  63. package/dist/test/unit/lib/exportTxBuilder.d.ts +2 -0
  64. package/dist/test/unit/lib/exportTxBuilder.d.ts.map +1 -0
  65. package/dist/test/unit/lib/exportTxBuilder.js +45 -0
  66. package/dist/test/unit/lib/transaction.d.ts +2 -0
  67. package/dist/test/unit/lib/transaction.d.ts.map +1 -0
  68. package/dist/test/unit/lib/transaction.js +460 -0
  69. package/dist/test/unit/lib/utils.d.ts +2 -0
  70. package/dist/test/unit/lib/utils.d.ts.map +1 -0
  71. package/dist/test/unit/lib/utils.js +286 -0
  72. package/dist/test/unit/smoke.d.ts +2 -0
  73. package/dist/test/unit/smoke.d.ts.map +1 -0
  74. package/dist/test/unit/smoke.js +23 -0
  75. package/dist/tsconfig.tsbuildinfo +1 -0
  76. package/package.json +57 -0
@@ -0,0 +1,142 @@
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.KeyPair = exports.addressFormat = void 0;
7
+ const sdk_core_1 = require("@bitgo-beta/sdk-core");
8
+ const secp256k1_1 = require("@bitgo-beta/secp256k1");
9
+ const crypto_1 = require("crypto");
10
+ const utils_1 = __importDefault(require("./utils"));
11
+ const DEFAULT_SEED_SIZE_BYTES = 16;
12
+ var addressFormat;
13
+ (function (addressFormat) {
14
+ addressFormat["testnet"] = "fuji";
15
+ addressFormat["mainnet"] = "flr";
16
+ })(addressFormat || (exports.addressFormat = addressFormat = {}));
17
+ class KeyPair extends sdk_core_1.Secp256k1ExtendedKeyPair {
18
+ /**
19
+ * Public constructor. By default, creates a key pair with a random master seed.
20
+ *
21
+ * @param { KeyPairOptions } source Either a master seed, a private key, or a public key
22
+ */
23
+ constructor(source) {
24
+ super(source);
25
+ if (!source) {
26
+ const seed = (0, crypto_1.randomBytes)(DEFAULT_SEED_SIZE_BYTES);
27
+ this.hdNode = secp256k1_1.bip32.fromSeed(seed);
28
+ }
29
+ else if ((0, sdk_core_1.isSeed)(source)) {
30
+ this.hdNode = secp256k1_1.bip32.fromSeed(source.seed);
31
+ }
32
+ else if ((0, sdk_core_1.isPrivateKey)(source)) {
33
+ this.recordKeysFromPrivateKey(source.prv);
34
+ }
35
+ else if ((0, sdk_core_1.isPublicKey)(source)) {
36
+ this.recordKeysFromPublicKey(source.pub);
37
+ }
38
+ else {
39
+ throw new Error('Invalid key pair options');
40
+ }
41
+ if (this.hdNode) {
42
+ this.keyPair = sdk_core_1.Secp256k1ExtendedKeyPair.toKeyPair(this.hdNode);
43
+ }
44
+ }
45
+ /**
46
+ * Build a keypair from a protocol private key or extended private key.
47
+ *
48
+ * @param {string} prv A raw private key
49
+ */
50
+ recordKeysFromPrivateKey(prv) {
51
+ if (!utils_1.default.isValidPrivateKey(prv)) {
52
+ throw new Error('Unsupported private key');
53
+ }
54
+ if ((0, sdk_core_1.isValidXprv)(prv)) {
55
+ this.hdNode = secp256k1_1.bip32.fromBase58(prv);
56
+ }
57
+ else {
58
+ this.keyPair = secp256k1_1.ECPair.fromPrivateKey(Buffer.from(prv.slice(0, 64), 'hex'));
59
+ }
60
+ }
61
+ /**
62
+ * Build an ECPair from a protocol public key or extended public key.
63
+ *
64
+ * @param {string} pub A raw public key
65
+ */
66
+ recordKeysFromPublicKey(pub) {
67
+ try {
68
+ if ((0, sdk_core_1.isValidXpub)(pub)) {
69
+ this.hdNode = secp256k1_1.bip32.fromBase58(pub);
70
+ }
71
+ else {
72
+ this.keyPair = secp256k1_1.ECPair.fromPublicKey(Buffer.from(pub, 'hex'));
73
+ }
74
+ return;
75
+ }
76
+ catch (e) {
77
+ try {
78
+ this.keyPair = secp256k1_1.ECPair.fromPublicKey(Buffer.from(utils_1.default.cb58Decode(pub)));
79
+ return;
80
+ }
81
+ catch (e) {
82
+ throw new Error('Unsupported public key');
83
+ }
84
+ }
85
+ }
86
+ /**
87
+ * Default keys format is a pair of Uint8Array keys
88
+ *
89
+ * @returns { DefaultKeys } The keys in the defined format
90
+ */
91
+ getKeys() {
92
+ return {
93
+ pub: this.getPublicKey({ compressed: true }).toString('hex'),
94
+ prv: this.getPrivateKey()?.toString('hex'),
95
+ };
96
+ }
97
+ /**
98
+ * Get a Flare P-Chain public mainnet address
99
+ *
100
+ * @param {string} format - flare hrp selector: Mainnet(flr) or Testnet(fuji)
101
+ * @returns {string} The mainnet address derived from the public key
102
+ */
103
+ getAddress(format = 'mainnet') {
104
+ return this.getFlrPAddress(addressFormat[format]);
105
+ }
106
+ /**
107
+ * Get a public address of public key.
108
+ *
109
+ * @param {string} hrp - select Mainnet(flr) or Testnet(fuji) for the address
110
+ * @returns {string} The address derived from the public key and hrp
111
+ */
112
+ getFlrPAddress(hrp) {
113
+ const addressBuffer = Buffer.from(this.getAddressBuffer());
114
+ return utils_1.default.addressToString(hrp, 'P', addressBuffer);
115
+ }
116
+ /**
117
+ * Get a Flare P-Chain public mainnet address buffer
118
+ *
119
+ * @returns {Buffer} The address buffer derived from the public key
120
+ */
121
+ getAddressBuffer() {
122
+ try {
123
+ // Use the safe buffer method for address derivation
124
+ return this.getAddressSafeBuffer();
125
+ }
126
+ catch (error) {
127
+ return this.getAddressSafeBuffer();
128
+ }
129
+ }
130
+ /**
131
+ * Use the safe Buffer instead of the regular buffer to derive the address buffer. Used in the OVC.
132
+ *
133
+ * @returns {Buffer}
134
+ */
135
+ getAddressSafeBuffer() {
136
+ const publicKeyHex = this.keyPair.publicKey.toString('hex');
137
+ const sha256 = (0, crypto_1.createHash)('sha256').update(publicKeyHex, 'hex').digest();
138
+ return (0, crypto_1.createHash)('ripemd160').update(sha256).digest();
139
+ }
140
+ }
141
+ exports.KeyPair = KeyPair;
142
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"keyPair.js","sourceRoot":"","sources":["../../../src/lib/keyPair.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  DefaultKeys,\n  isPrivateKey,\n  isPublicKey,\n  isSeed,\n  isValidXprv,\n  isValidXpub,\n  KeyPairOptions,\n  Secp256k1ExtendedKeyPair,\n} from '@bitgo-beta/sdk-core';\nimport { bip32, ECPair } from '@bitgo-beta/secp256k1';\nimport { randomBytes, createHash } from 'crypto';\nimport utils from './utils';\n\nconst DEFAULT_SEED_SIZE_BYTES = 16;\nexport enum addressFormat {\n  testnet = 'fuji',\n  mainnet = 'flr',\n}\n\nexport class KeyPair extends Secp256k1ExtendedKeyPair {\n  /**\n   * Public constructor. By default, creates a key pair with a random master seed.\n   *\n   * @param { KeyPairOptions } source Either a master seed, a private key, or a public key\n   */\n  constructor(source?: KeyPairOptions) {\n    super(source);\n    if (!source) {\n      const seed = randomBytes(DEFAULT_SEED_SIZE_BYTES);\n      this.hdNode = bip32.fromSeed(seed);\n    } else if (isSeed(source)) {\n      this.hdNode = bip32.fromSeed(source.seed);\n    } else if (isPrivateKey(source)) {\n      this.recordKeysFromPrivateKey(source.prv);\n    } else if (isPublicKey(source)) {\n      this.recordKeysFromPublicKey(source.pub);\n    } else {\n      throw new Error('Invalid key pair options');\n    }\n\n    if (this.hdNode) {\n      this.keyPair = Secp256k1ExtendedKeyPair.toKeyPair(this.hdNode);\n    }\n  }\n\n  /**\n   * Build a keypair from a protocol private key or extended private key.\n   *\n   * @param {string} prv A raw private key\n   */\n  recordKeysFromPrivateKey(prv: string): void {\n    if (!utils.isValidPrivateKey(prv)) {\n      throw new Error('Unsupported private key');\n    }\n    if (isValidXprv(prv)) {\n      this.hdNode = bip32.fromBase58(prv);\n    } else {\n      this.keyPair = ECPair.fromPrivateKey(Buffer.from(prv.slice(0, 64), 'hex'));\n    }\n  }\n\n  /**\n   * Build an ECPair from a protocol public key or extended public key.\n   *\n   * @param {string} pub A raw public key\n   */\n  recordKeysFromPublicKey(pub: string): void {\n    try {\n      if (isValidXpub(pub)) {\n        this.hdNode = bip32.fromBase58(pub);\n      } else {\n        this.keyPair = ECPair.fromPublicKey(Buffer.from(pub, 'hex'));\n      }\n      return;\n    } catch (e) {\n      try {\n        this.keyPair = ECPair.fromPublicKey(Buffer.from(utils.cb58Decode(pub)));\n        return;\n      } catch (e) {\n        throw new Error('Unsupported public key');\n      }\n    }\n  }\n\n  /**\n   * Default keys format is a pair of Uint8Array keys\n   *\n   * @returns { DefaultKeys } The keys in the defined format\n   */\n  getKeys(): DefaultKeys {\n    return {\n      pub: this.getPublicKey({ compressed: true }).toString('hex'),\n      prv: this.getPrivateKey()?.toString('hex'),\n    };\n  }\n\n  /**\n   * Get a Flare P-Chain public mainnet address\n   *\n   * @param {string} format - flare hrp selector: Mainnet(flr) or Testnet(fuji)\n   * @returns {string} The mainnet address derived from the public key\n   */\n  getAddress(format = 'mainnet'): string {\n    return this.getFlrPAddress(addressFormat[format]);\n  }\n  /**\n   * Get a public address of public key.\n   *\n   * @param {string} hrp - select Mainnet(flr) or Testnet(fuji) for the address\n   * @returns {string} The address derived from the public key and hrp\n   */\n  getFlrPAddress(hrp: string): string {\n    const addressBuffer = Buffer.from(this.getAddressBuffer());\n    return utils.addressToString(hrp, 'P', addressBuffer);\n  }\n\n  /**\n   * Get a Flare P-Chain public mainnet address buffer\n   *\n   * @returns {Buffer} The address buffer derived from the public key\n   */\n  private getAddressBuffer(): Buffer {\n    try {\n      // Use the safe buffer method for address derivation\n      return this.getAddressSafeBuffer();\n    } catch (error) {\n      return this.getAddressSafeBuffer();\n    }\n  }\n\n  /**\n   * Use the safe Buffer instead of the regular buffer to derive the address buffer. Used in the OVC.\n   *\n   * @returns {Buffer}\n   */\n  private getAddressSafeBuffer(): Buffer {\n    const publicKeyHex = this.keyPair.publicKey.toString('hex');\n    const sha256 = createHash('sha256').update(publicKeyHex, 'hex').digest();\n    return createHash('ripemd160').update(sha256).digest();\n  }\n}\n"]}
@@ -0,0 +1,111 @@
1
+ import { UnsignedTx, Credential } from '@flarenetwork/flarejs';
2
+ import { BaseKey, BaseTransaction, Entry, TransactionFee, TransactionType } from '@bitgo-beta/sdk-core';
3
+ import { FlareNetwork, BaseCoin as CoinConfig } from '@bitgo-beta/statics';
4
+ import { Buffer } from 'buffer';
5
+ import { DecodedUtxoObj, TransactionExplanation, Tx, TxData, FlrpEntry } from './iface';
6
+ import { KeyPair } from './keyPair';
7
+ /**
8
+ * Flare P-chain transaction implementation using FlareJS
9
+ * Based on AVAX transaction patterns adapted for Flare network
10
+ */
11
+ export declare class Transaction extends BaseTransaction {
12
+ protected _flareTransaction: Tx;
13
+ _type: TransactionType;
14
+ _network: FlareNetwork;
15
+ _networkID: number;
16
+ _assetId: string;
17
+ _blockchainID: string;
18
+ _nodeID: string;
19
+ _startTime: bigint;
20
+ _endTime: bigint;
21
+ _stakeAmount: bigint;
22
+ _threshold: number;
23
+ _locktime: bigint;
24
+ _fromAddresses: string[];
25
+ _rewardAddresses: string[];
26
+ _utxos: DecodedUtxoObj[];
27
+ _to: string[];
28
+ _fee: Partial<TransactionFee>;
29
+ _blsPublicKey: string;
30
+ _blsSignature: string;
31
+ _memo: Uint8Array;
32
+ constructor(coinConfig: Readonly<CoinConfig>);
33
+ /**
34
+ * Get the base transaction from FlareJS UnsignedTx
35
+ * TODO: Implement proper FlareJS transaction extraction
36
+ */
37
+ get flareTransaction(): UnsignedTx;
38
+ get signature(): string[];
39
+ get credentials(): Credential[];
40
+ get hasCredentials(): boolean;
41
+ /** @inheritdoc */
42
+ canSign({ key }: BaseKey): boolean;
43
+ /**
44
+ * Sign a Flare transaction using FlareJS
45
+ * @param {KeyPair} keyPair
46
+ */
47
+ sign(keyPair: KeyPair): Promise<void>;
48
+ /**
49
+ * Set memo from string
50
+ * @param {string} memo - Memo text
51
+ */
52
+ setMemo(memo: string): void;
53
+ /**
54
+ * Set memo from various formats
55
+ * @param {string | Record<string, unknown> | Uint8Array} memo - Memo data
56
+ */
57
+ setMemoData(memo: string | Record<string, unknown> | Uint8Array): void;
58
+ /**
59
+ * Get memo as bytes (FlareJS format)
60
+ * @returns {Uint8Array} Memo bytes
61
+ */
62
+ getMemoBytes(): Uint8Array;
63
+ /**
64
+ * Get memo as string
65
+ * @returns {string} Memo string
66
+ */
67
+ getMemoString(): string;
68
+ /**
69
+ * Check if transaction has memo
70
+ * @returns {boolean} Whether memo exists and is not empty
71
+ */
72
+ hasMemo(): boolean;
73
+ toHexString(byteArray: Uint8Array): string;
74
+ /** @inheritdoc */
75
+ toBroadcastFormat(): string;
76
+ toJson(): TxData;
77
+ setTransaction(tx: Tx): void;
78
+ /**
79
+ * Set the transaction type
80
+ * @param {TransactionType} transactionType The transaction type to be set
81
+ */
82
+ setTransactionType(transactionType: TransactionType): void;
83
+ /**
84
+ * Returns the portion of the transaction that needs to be signed in Buffer format.
85
+ * Only needed for coins that support adding signatures directly (e.g. TSS).
86
+ */
87
+ get signablePayload(): Buffer;
88
+ get id(): string;
89
+ get fromAddresses(): string[];
90
+ get rewardAddresses(): string[];
91
+ /**
92
+ * Get the list of outputs. Amounts are expressed in absolute value.
93
+ */
94
+ get outputs(): Entry[];
95
+ get fee(): TransactionFee;
96
+ get changeOutputs(): Entry[];
97
+ get inputs(): FlrpEntry[];
98
+ /**
99
+ * Flare wrapper to create signature and return it for credentials
100
+ * @param prv
101
+ * @return hexstring
102
+ */
103
+ createSignature(prv: Buffer): string;
104
+ /**
105
+ * Check if transaction is for C-chain (cross-chain)
106
+ */
107
+ get isTransactionForCChain(): boolean;
108
+ /** @inheritdoc */
109
+ explainTransaction(): TransactionExplanation;
110
+ }
111
+ //# sourceMappingURL=transaction.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"transaction.d.ts","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,321 @@
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 buffer_1 = require("buffer");
9
+ const iface_1 = require("./iface");
10
+ const utils_1 = __importDefault(require("./utils"));
11
+ /**
12
+ * Flare P-chain transaction implementation using FlareJS
13
+ * Based on AVAX transaction patterns adapted for Flare network
14
+ */
15
+ class Transaction extends sdk_core_1.BaseTransaction {
16
+ constructor(coinConfig) {
17
+ super(coinConfig);
18
+ this._threshold = 2;
19
+ this._locktime = BigInt(0);
20
+ this._fromAddresses = [];
21
+ this._rewardAddresses = [];
22
+ this._utxos = [];
23
+ this._fee = {};
24
+ this._memo = new Uint8Array(); // FlareJS memo field
25
+ this._network = coinConfig.network;
26
+ this._assetId = 'FLR'; // Default FLR asset
27
+ this._blockchainID = this._network.blockchainID || '';
28
+ this._networkID = this._network.networkID || 0;
29
+ }
30
+ /**
31
+ * Get the base transaction from FlareJS UnsignedTx
32
+ * TODO: Implement proper FlareJS transaction extraction
33
+ */
34
+ get flareTransaction() {
35
+ return this._flareTransaction;
36
+ }
37
+ get signature() {
38
+ if (this.credentials.length === 0) {
39
+ return [];
40
+ }
41
+ // TODO: Extract signatures from FlareJS credentials
42
+ // For now, return placeholder
43
+ return [];
44
+ }
45
+ get credentials() {
46
+ // TODO: Extract credentials from FlareJS transaction
47
+ // For now, return empty array
48
+ return [];
49
+ }
50
+ get hasCredentials() {
51
+ return this.credentials !== undefined && this.credentials.length > 0;
52
+ }
53
+ /** @inheritdoc */
54
+ canSign({ key }) {
55
+ // TODO: Implement proper signing validation for FlareJS
56
+ return true;
57
+ }
58
+ /**
59
+ * Sign a Flare transaction using FlareJS
60
+ * @param {KeyPair} keyPair
61
+ */
62
+ async sign(keyPair) {
63
+ const prv = keyPair.getPrivateKey();
64
+ if (!prv) {
65
+ throw new sdk_core_1.SigningError('Missing private key');
66
+ }
67
+ if (!this.flareTransaction) {
68
+ throw new sdk_core_1.InvalidTransactionError('empty transaction to sign');
69
+ }
70
+ if (!this.hasCredentials) {
71
+ throw new sdk_core_1.InvalidTransactionError('empty credentials to sign');
72
+ }
73
+ // TODO: Implement FlareJS signing process
74
+ // This will involve:
75
+ // 1. Creating FlareJS signature using private key
76
+ // 2. Attaching signature to appropriate credential
77
+ // 3. Updating transaction with signed credentials
78
+ throw new Error('FlareJS signing not yet implemented - placeholder');
79
+ }
80
+ /**
81
+ * Set memo from string
82
+ * @param {string} memo - Memo text
83
+ */
84
+ setMemo(memo) {
85
+ this._memo = utils_1.default.stringToBytes(memo);
86
+ }
87
+ /**
88
+ * Set memo from various formats
89
+ * @param {string | Record<string, unknown> | Uint8Array} memo - Memo data
90
+ */
91
+ setMemoData(memo) {
92
+ this._memo = utils_1.default.createMemoBytes(memo);
93
+ }
94
+ /**
95
+ * Get memo as bytes (FlareJS format)
96
+ * @returns {Uint8Array} Memo bytes
97
+ */
98
+ getMemoBytes() {
99
+ return this._memo;
100
+ }
101
+ /**
102
+ * Get memo as string
103
+ * @returns {string} Memo string
104
+ */
105
+ getMemoString() {
106
+ return utils_1.default.parseMemoBytes(this._memo);
107
+ }
108
+ /**
109
+ * Check if transaction has memo
110
+ * @returns {boolean} Whether memo exists and is not empty
111
+ */
112
+ hasMemo() {
113
+ return this._memo.length > 0;
114
+ }
115
+ toHexString(byteArray) {
116
+ return buffer_1.Buffer.from(byteArray).toString('hex');
117
+ }
118
+ /** @inheritdoc */
119
+ toBroadcastFormat() {
120
+ if (!this.flareTransaction) {
121
+ throw new sdk_core_1.InvalidTransactionError('Empty transaction data');
122
+ }
123
+ // TODO: Implement FlareJS transaction serialization
124
+ // For now, return placeholder
125
+ return 'flare-tx-hex-placeholder';
126
+ }
127
+ toJson() {
128
+ if (!this.flareTransaction) {
129
+ throw new sdk_core_1.InvalidTransactionError('Empty transaction data');
130
+ }
131
+ return {
132
+ id: this.id,
133
+ inputs: this.inputs,
134
+ fromAddresses: this.fromAddresses,
135
+ threshold: this._threshold,
136
+ locktime: this._locktime.toString(),
137
+ type: this.type,
138
+ signatures: this.signature,
139
+ outputs: this.outputs,
140
+ changeOutputs: this.changeOutputs,
141
+ sourceChain: this._network.blockchainID,
142
+ destinationChain: this._network.cChainBlockchainID,
143
+ memo: this.getMemoString(), // Include memo in JSON representation
144
+ };
145
+ }
146
+ setTransaction(tx) {
147
+ this._flareTransaction = tx;
148
+ }
149
+ /**
150
+ * Set the transaction type
151
+ * @param {TransactionType} transactionType The transaction type to be set
152
+ */
153
+ setTransactionType(transactionType) {
154
+ const supportedTypes = [
155
+ sdk_core_1.TransactionType.Export,
156
+ sdk_core_1.TransactionType.Import,
157
+ sdk_core_1.TransactionType.AddValidator,
158
+ sdk_core_1.TransactionType.AddDelegator,
159
+ sdk_core_1.TransactionType.AddPermissionlessValidator,
160
+ sdk_core_1.TransactionType.AddPermissionlessDelegator,
161
+ ];
162
+ if (!supportedTypes.includes(transactionType)) {
163
+ throw new Error(`Transaction type ${transactionType} is not supported`);
164
+ }
165
+ this._type = transactionType;
166
+ }
167
+ /**
168
+ * Returns the portion of the transaction that needs to be signed in Buffer format.
169
+ * Only needed for coins that support adding signatures directly (e.g. TSS).
170
+ */
171
+ get signablePayload() {
172
+ if (!this.flareTransaction) {
173
+ throw new sdk_core_1.InvalidTransactionError('Empty transaction for signing');
174
+ }
175
+ // TODO: Implement FlareJS signable payload extraction
176
+ // For now, return placeholder
177
+ return buffer_1.Buffer.from('flare-signable-payload');
178
+ }
179
+ get id() {
180
+ if (!this.flareTransaction) {
181
+ throw new sdk_core_1.InvalidTransactionError('Empty transaction for ID generation');
182
+ }
183
+ // TODO: Implement FlareJS transaction ID generation
184
+ // For now, return placeholder
185
+ return 'flare-transaction-id-placeholder';
186
+ }
187
+ get fromAddresses() {
188
+ return this._fromAddresses.map((address) => {
189
+ // TODO: Format addresses using FlareJS utilities
190
+ return address;
191
+ });
192
+ }
193
+ get rewardAddresses() {
194
+ return this._rewardAddresses.map((address) => {
195
+ // TODO: Format addresses using FlareJS utilities
196
+ return address;
197
+ });
198
+ }
199
+ /**
200
+ * Get the list of outputs. Amounts are expressed in absolute value.
201
+ */
202
+ get outputs() {
203
+ switch (this.type) {
204
+ case sdk_core_1.TransactionType.Export:
205
+ // TODO: Extract export outputs from FlareJS transaction
206
+ return [];
207
+ case sdk_core_1.TransactionType.Import:
208
+ // TODO: Extract import outputs from FlareJS transaction
209
+ return [];
210
+ case sdk_core_1.TransactionType.AddValidator:
211
+ case sdk_core_1.TransactionType.AddPermissionlessValidator:
212
+ // TODO: Extract validator outputs from FlareJS transaction
213
+ return [
214
+ {
215
+ address: this._nodeID || 'placeholder-node-id',
216
+ value: this._stakeAmount?.toString() || '0',
217
+ },
218
+ ];
219
+ case sdk_core_1.TransactionType.AddDelegator:
220
+ case sdk_core_1.TransactionType.AddPermissionlessDelegator:
221
+ // TODO: Extract delegator outputs from FlareJS transaction
222
+ return [
223
+ {
224
+ address: this._nodeID || 'placeholder-node-id',
225
+ value: this._stakeAmount?.toString() || '0',
226
+ },
227
+ ];
228
+ default:
229
+ return [];
230
+ }
231
+ }
232
+ get fee() {
233
+ return { fee: '0', ...this._fee };
234
+ }
235
+ get changeOutputs() {
236
+ // TODO: Extract change outputs from FlareJS transaction
237
+ // For now, return empty array
238
+ return [];
239
+ }
240
+ get inputs() {
241
+ // TODO: Extract inputs from FlareJS transaction
242
+ // For now, return placeholder based on UTXOs
243
+ return this._utxos.map((utxo) => ({
244
+ id: utxo.txid + iface_1.INPUT_SEPARATOR + utxo.outputidx,
245
+ address: this.fromAddresses.sort().join(iface_1.ADDRESS_SEPARATOR),
246
+ value: utxo.amount,
247
+ }));
248
+ }
249
+ /**
250
+ * Flare wrapper to create signature and return it for credentials
251
+ * @param prv
252
+ * @return hexstring
253
+ */
254
+ createSignature(prv) {
255
+ // TODO: Implement FlareJS signature creation
256
+ // This should use FlareJS signing utilities
257
+ const signval = utils_1.default.createSignature(this._network, this.signablePayload, prv);
258
+ return signval.toString('hex');
259
+ }
260
+ /**
261
+ * Check if transaction is for C-chain (cross-chain)
262
+ */
263
+ get isTransactionForCChain() {
264
+ return this.type === sdk_core_1.TransactionType.Export || this.type === sdk_core_1.TransactionType.Import;
265
+ }
266
+ /** @inheritdoc */
267
+ explainTransaction() {
268
+ const txJson = this.toJson();
269
+ const displayOrder = ['id', 'inputs', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'type'];
270
+ // Add memo to display order if present
271
+ if (this.hasMemo()) {
272
+ displayOrder.push('memo');
273
+ }
274
+ // Calculate total output amount
275
+ const outputAmount = txJson.outputs
276
+ .reduce((sum, output) => {
277
+ return sum + BigInt(output.value || '0');
278
+ }, BigInt(0))
279
+ .toString();
280
+ // Calculate total change amount
281
+ const changeAmount = txJson.changeOutputs
282
+ .reduce((sum, output) => {
283
+ return sum + BigInt(output.value || '0');
284
+ }, BigInt(0))
285
+ .toString();
286
+ let rewardAddresses;
287
+ const stakingTypes = [
288
+ sdk_core_1.TransactionType.AddValidator,
289
+ sdk_core_1.TransactionType.AddDelegator,
290
+ sdk_core_1.TransactionType.AddPermissionlessValidator,
291
+ sdk_core_1.TransactionType.AddPermissionlessDelegator,
292
+ ];
293
+ if (stakingTypes.includes(txJson.type)) {
294
+ rewardAddresses = this.rewardAddresses;
295
+ displayOrder.splice(6, 0, 'rewardAddresses');
296
+ }
297
+ // Add cross-chain information for export/import
298
+ if (this.isTransactionForCChain) {
299
+ displayOrder.push('sourceChain', 'destinationChain');
300
+ }
301
+ const explanation = {
302
+ displayOrder,
303
+ id: txJson.id,
304
+ inputs: txJson.inputs,
305
+ outputs: txJson.outputs.map((o) => ({ address: o.address, amount: o.value })),
306
+ outputAmount,
307
+ changeOutputs: txJson.changeOutputs.map((o) => ({ address: o.address, amount: o.value })),
308
+ changeAmount,
309
+ rewardAddresses,
310
+ fee: this.fee,
311
+ type: txJson.type,
312
+ };
313
+ // Add memo to explanation if present
314
+ if (this.hasMemo()) {
315
+ explanation.memo = this.getMemoString();
316
+ }
317
+ return explanation;
318
+ }
319
+ }
320
+ 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"]}