@bitgo-beta/sdk-coin-flrp 1.0.1-beta.278 → 1.0.1-beta.279

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -10,13 +10,59 @@ const buffer_1 = require("buffer");
10
10
  const crypto_1 = require("crypto");
11
11
  const utils_1 = __importDefault(require("./utils"));
12
12
  /**
13
- * Checks if a signature is empty
13
+ * Checks if a signature is empty (first 90 hex chars are zeros)
14
14
  * @param signature
15
15
  * @returns {boolean}
16
16
  */
17
17
  function isEmptySignature(signature) {
18
18
  return !!signature && utils_1.default.removeHexPrefix(signature).startsWith(''.padStart(90, '0'));
19
19
  }
20
+ /**
21
+ * Checks if an empty signature has an embedded address (non-zero bytes after position 90)
22
+ * @param signature Hex string of the signature
23
+ */
24
+ function hasEmbeddedAddress(signature) {
25
+ if (!isEmptySignature(signature))
26
+ return false;
27
+ const cleanSig = utils_1.default.removeHexPrefix(signature);
28
+ if (cleanSig.length < 130)
29
+ return false;
30
+ const embeddedPart = cleanSig.substring(90, 130);
31
+ // Check if it's not all zeros
32
+ return embeddedPart !== '0'.repeat(40);
33
+ }
34
+ /**
35
+ * Generates a function to check if a signature slot matches a given address.
36
+ * If signatures have embedded addresses, it matches by address.
37
+ * Otherwise, it just finds empty slots.
38
+ * @param signatures Array of signature hex strings
39
+ */
40
+ function generateSelectorSignature(signatures) {
41
+ // Check if any empty signature has an embedded address
42
+ const hasEmbeddedAddresses = signatures.some((sig) => isEmptySignature(sig) && hasEmbeddedAddress(sig));
43
+ if (hasEmbeddedAddresses) {
44
+ // Look for address embedded in the empty signature (after position 90)
45
+ return function (sig, address) {
46
+ try {
47
+ if (!isEmptySignature(sig)) {
48
+ return false;
49
+ }
50
+ const cleanSig = utils_1.default.removeHexPrefix(sig);
51
+ const embeddedAddr = cleanSig.substring(90, 130).toLowerCase();
52
+ return embeddedAddr === address.toLowerCase();
53
+ }
54
+ catch (e) {
55
+ return false;
56
+ }
57
+ };
58
+ }
59
+ else {
60
+ // Look for any empty slot (no embedded addresses)
61
+ return function (sig, address) {
62
+ return isEmptySignature(sig);
63
+ };
64
+ }
65
+ }
20
66
  class Transaction extends sdk_core_1.BaseTransaction {
21
67
  constructor(coinConfig) {
22
68
  super(coinConfig);
@@ -78,19 +124,30 @@ class Transaction extends sdk_core_1.BaseTransaction {
78
124
  if (hasMatchingAddress) {
79
125
  const signature = await flarejs_1.secp256k1.sign(unsignedBytes, prv);
80
126
  let signatureSet = false;
81
- // Find first empty signature slot and set it
127
+ // Use address-based slot matching (like AVAX-P)
128
+ let checkSign = undefined;
82
129
  for (const credential of unsignedTx.credentials) {
83
- const emptySlotIndex = credential.getSignatures().findIndex((sig) => isEmptySignature(sig));
84
- if (emptySlotIndex !== -1) {
85
- credential.setSignature(emptySlotIndex, signature);
86
- signatureSet = true;
87
- // Clear raw signed bytes since we've modified the transaction
88
- this._rawSignedBytes = undefined;
89
- break;
130
+ const signatures = credential.getSignatures();
131
+ if (checkSign === undefined) {
132
+ checkSign = generateSelectorSignature(signatures);
90
133
  }
134
+ // Find the slot that matches this address
135
+ for (let i = 0; i < signatures.length; i++) {
136
+ const sig = signatures[i];
137
+ // Try matching with P-chain address first, then EVM address
138
+ if (checkSign(sig, pChainAddressHex) || checkSign(sig, utils_1.default.removeHexPrefix(evmAddressHex).toLowerCase())) {
139
+ credential.setSignature(i, signature);
140
+ signatureSet = true;
141
+ // Clear raw signed bytes since we've modified the transaction
142
+ this._rawSignedBytes = undefined;
143
+ break;
144
+ }
145
+ }
146
+ if (signatureSet)
147
+ break;
91
148
  }
92
149
  if (!signatureSet) {
93
- throw new sdk_core_1.SigningError('No empty signature slot found');
150
+ throw new sdk_core_1.SigningError('No matching signature slot found for this private key');
94
151
  }
95
152
  }
96
153
  }
@@ -193,4 +250,4 @@ class Transaction extends sdk_core_1.BaseTransaction {
193
250
  }
194
251
  }
195
252
  exports.Transaction = Transaction;
196
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":"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","sourcesContent":["import { FlareNetwork, BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport {\n  BaseKey,\n  BaseTransaction,\n  Entry,\n  InvalidTransactionError,\n  SigningError,\n  TransactionType,\n  TransactionFee,\n} from '@bitgo-beta/sdk-core';\nimport {\n  utils as FlareUtils,\n  Credential,\n  pvmSerial,\n  UnsignedTx,\n  secp256k1,\n  EVMUnsignedTx,\n  Address,\n} from '@flarenetwork/flarejs';\nimport { Buffer } from 'buffer';\nimport { createHash } from 'crypto';\nimport { DecodedUtxoObj, TransactionExplanation, Tx, TxData } from './iface';\nimport { KeyPair } from './keyPair';\nimport utils from './utils';\n\n/**\n * Checks if a signature is empty\n * @param signature\n * @returns {boolean}\n */\nfunction isEmptySignature(signature: string): boolean {\n  return !!signature && utils.removeHexPrefix(signature).startsWith(''.padStart(90, '0'));\n}\n\nexport class Transaction extends BaseTransaction {\n  protected _flareTransaction: Tx;\n  public _type: TransactionType;\n  public _network: FlareNetwork;\n  public _networkID: number;\n  public _assetId: string;\n  public _blockchainID: string;\n  public _nodeID: string;\n  public _startTime: bigint;\n  public _endTime: bigint;\n  public _stakeAmount: bigint;\n  public _threshold = 2;\n  public _locktime = BigInt(0);\n  public _fromAddresses: Uint8Array[] = [];\n  public _to: Uint8Array[] = [];\n  public _rewardAddresses: Uint8Array[] = [];\n  public _utxos: DecodedUtxoObj[] = []; // Define proper type based on Flare's UTXO structure\n  public _fee: Partial<TransactionFee> = {};\n  // Store original raw signed bytes to preserve exact format when re-serializing\n  public _rawSignedBytes: Buffer | undefined;\n\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    this._network = coinConfig.network as FlareNetwork;\n    // Decode cb58-encoded asset ID to hex for use in transaction serialization\n    this._assetId = utils.cb58Decode(this._network.assetId).toString('hex');\n    this._blockchainID = this._network.blockchainID;\n    this._networkID = this._network.networkID;\n  }\n\n  get signature(): string[] {\n    if (!this.hasCredentials) {\n      return [];\n    }\n    return this.credentials[0].getSignatures().filter((s) => !isEmptySignature(s));\n  }\n\n  get credentials(): Credential[] {\n    return (this._flareTransaction as UnsignedTx)?.credentials;\n  }\n\n  get hasCredentials(): boolean {\n    return this.credentials !== undefined && this.credentials.length > 0;\n  }\n\n  /** @inheritdoc */\n  canSign({ key }: BaseKey): boolean {\n    return true;\n  }\n\n  async sign(keyPair: KeyPair): Promise<void> {\n    const prv = keyPair.getPrivateKey() as Uint8Array;\n    if (!prv) {\n      throw new SigningError('Missing private key');\n    }\n    if (!this._flareTransaction) {\n      throw new InvalidTransactionError('empty transaction to sign');\n    }\n    if (!this.hasCredentials) {\n      throw new InvalidTransactionError('empty credentials to sign');\n    }\n\n    const unsignedTx = this._flareTransaction as EVMUnsignedTx;\n    const unsignedBytes = unsignedTx.toBytes();\n    const publicKey = secp256k1.getPublicKey(prv);\n\n    // Derive both EVM and P-chain addresses from the public key\n    const evmAddressHex = new Address(secp256k1.publicKeyToEthAddress(publicKey)).toHex();\n\n    // P-chain address derivation: ripemd160(sha256(publicKey))\n    const sha256Hash = createHash('sha256').update(Buffer.from(publicKey)).digest();\n    const pChainAddressBuffer = createHash('ripemd160').update(sha256Hash).digest();\n    const pChainAddressHex = pChainAddressBuffer.toString('hex');\n\n    const addressMap = unsignedTx.getAddresses();\n\n    // Check for both EVM and P-chain address formats\n    const hasMatchingAddress = addressMap.some((addr) => {\n      const addrHex = Buffer.from(addr).toString('hex').toLowerCase();\n      return (\n        addrHex === utils.removeHexPrefix(evmAddressHex).toLowerCase() || addrHex === pChainAddressHex.toLowerCase()\n      );\n    });\n\n    if (hasMatchingAddress) {\n      const signature = await secp256k1.sign(unsignedBytes, prv);\n\n      let signatureSet = false;\n      // Find first empty signature slot and set it\n      for (const credential of unsignedTx.credentials) {\n        const emptySlotIndex = credential.getSignatures().findIndex((sig) => isEmptySignature(sig));\n        if (emptySlotIndex !== -1) {\n          credential.setSignature(emptySlotIndex, signature);\n          signatureSet = true;\n          // Clear raw signed bytes since we've modified the transaction\n          this._rawSignedBytes = undefined;\n          break;\n        }\n      }\n\n      if (!signatureSet) {\n        throw new SigningError('No empty signature slot found');\n      }\n    }\n  }\n\n  toBroadcastFormat(): string {\n    if (!this._flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\n    // If we have the original raw signed bytes, use them directly to preserve exact format\n    if (this._rawSignedBytes) {\n      return FlareUtils.bufferToHex(this._rawSignedBytes);\n    }\n    const unsignedTx = this._flareTransaction as UnsignedTx;\n    // For signed transactions, return the full signed tx with credentials\n    // Check signature.length for robustness\n    if (this.signature.length > 0) {\n      return FlareUtils.bufferToHex(unsignedTx.getSignedTx().toBytes());\n    }\n    // For unsigned transactions, return just the transaction bytes\n    return FlareUtils.bufferToHex(unsignedTx.toBytes());\n  }\n\n  toJson(): TxData {\n    if (!this._flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\n    return {\n      id: this.id,\n      inputs: this.inputs,\n      fromAddresses: this.fromAddresses,\n      threshold: this._threshold,\n      locktime: this._locktime.toString(),\n      type: this.type,\n      signatures: this.signature,\n      outputs: this.outputs,\n      changeOutputs: this.changeOutputs,\n    };\n  }\n\n  setTransaction(tx: Tx): void {\n    this._flareTransaction = tx as UnsignedTx;\n  }\n\n  setTransactionType(transactionType: TransactionType): void {\n    if (![TransactionType.AddPermissionlessValidator].includes(transactionType)) {\n      throw new Error(`Transaction type ${transactionType} is not supported`);\n    }\n    this._type = transactionType;\n  }\n\n  get signablePayload(): Buffer {\n    return utils.sha256((this._flareTransaction as UnsignedTx).toBytes());\n  }\n\n  get id(): string {\n    const bufferArray = utils.sha256((this._flareTransaction as UnsignedTx).toBytes());\n    return utils.cb58Encode(Buffer.from(bufferArray));\n  }\n\n  get fromAddresses(): string[] {\n    return this._fromAddresses.map((a) => FlareUtils.format(this._network.alias, this._network.hrp, a));\n  }\n\n  get rewardAddresses(): string[] {\n    return this._rewardAddresses.map((a) => FlareUtils.format(this._network.alias, this._network.hrp, a));\n  }\n\n  get fee(): TransactionFee {\n    return { fee: '0', ...this._fee };\n  }\n\n  get outputs(): Entry[] {\n    switch (this.type) {\n      case TransactionType.AddPermissionlessValidator:\n        return [\n          {\n            address: (\n              (this._flareTransaction as UnsignedTx).getTx() as pvmSerial.AddPermissionlessValidatorTx\n            ).subnetValidator.validator.nodeId.toString(),\n            value: (\n              (this._flareTransaction as UnsignedTx).getTx() as pvmSerial.AddPermissionlessValidatorTx\n            ).subnetValidator.validator.weight.toJSON(),\n          },\n        ];\n      default:\n        return [];\n    }\n  }\n\n  get changeOutputs(): Entry[] {\n    return (\n      (this._flareTransaction as UnsignedTx).getTx() as pvmSerial.AddPermissionlessValidatorTx\n    ).baseTx.outputs.map(utils.mapOutputToEntry(this._network));\n  }\n\n  explainTransaction(): TransactionExplanation {\n    const txJson = this.toJson();\n    const displayOrder = ['id', 'inputs', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'type'];\n\n    const outputAmount = txJson.outputs.reduce((p, n) => p + BigInt(n.value), BigInt(0)).toString();\n    const changeAmount = txJson.changeOutputs.reduce((p, n) => p + BigInt(n.value), BigInt(0)).toString();\n\n    let rewardAddresses;\n    if ([TransactionType.AddPermissionlessValidator].includes(txJson.type)) {\n      rewardAddresses = this.rewardAddresses;\n      displayOrder.splice(6, 0, 'rewardAddresses');\n    }\n\n    return {\n      displayOrder,\n      id: txJson.id,\n      inputs: txJson.inputs,\n      outputs: txJson.outputs.map((o) => ({ address: o.address, amount: o.value })),\n      outputAmount,\n      changeOutputs: txJson.changeOutputs.map((o) => ({ address: o.address, amount: o.value })),\n      changeAmount,\n      rewardAddresses,\n      fee: this.fee,\n      type: txJson.type,\n    };\n  }\n}\n"]}
253
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":"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","sourcesContent":["import { FlareNetwork, BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport {\n  BaseKey,\n  BaseTransaction,\n  Entry,\n  InvalidTransactionError,\n  SigningError,\n  TransactionType,\n  TransactionFee,\n} from '@bitgo-beta/sdk-core';\nimport {\n  utils as FlareUtils,\n  Credential,\n  pvmSerial,\n  UnsignedTx,\n  secp256k1,\n  EVMUnsignedTx,\n  Address,\n} from '@flarenetwork/flarejs';\nimport { Buffer } from 'buffer';\nimport { createHash } from 'crypto';\nimport { DecodedUtxoObj, TransactionExplanation, Tx, TxData } from './iface';\nimport { KeyPair } from './keyPair';\nimport utils from './utils';\n\n/**\n * Checks if a signature is empty (first 90 hex chars are zeros)\n * @param signature\n * @returns {boolean}\n */\nfunction isEmptySignature(signature: string): boolean {\n  return !!signature && utils.removeHexPrefix(signature).startsWith(''.padStart(90, '0'));\n}\n\n/**\n * Interface for signature slot checking\n */\ninterface CheckSignature {\n  (signature: string, addressHex: string): boolean;\n}\n\n/**\n * Checks if an empty signature has an embedded address (non-zero bytes after position 90)\n * @param signature Hex string of the signature\n */\nfunction hasEmbeddedAddress(signature: string): boolean {\n  if (!isEmptySignature(signature)) return false;\n  const cleanSig = utils.removeHexPrefix(signature);\n  if (cleanSig.length < 130) return false;\n  const embeddedPart = cleanSig.substring(90, 130);\n  // Check if it's not all zeros\n  return embeddedPart !== '0'.repeat(40);\n}\n\n/**\n * Generates a function to check if a signature slot matches a given address.\n * If signatures have embedded addresses, it matches by address.\n * Otherwise, it just finds empty slots.\n * @param signatures Array of signature hex strings\n */\nfunction generateSelectorSignature(signatures: string[]): CheckSignature {\n  // Check if any empty signature has an embedded address\n  const hasEmbeddedAddresses = signatures.some((sig) => isEmptySignature(sig) && hasEmbeddedAddress(sig));\n\n  if (hasEmbeddedAddresses) {\n    // Look for address embedded in the empty signature (after position 90)\n    return function (sig: string, address: string): boolean {\n      try {\n        if (!isEmptySignature(sig)) {\n          return false;\n        }\n        const cleanSig = utils.removeHexPrefix(sig);\n        const embeddedAddr = cleanSig.substring(90, 130).toLowerCase();\n        return embeddedAddr === address.toLowerCase();\n      } catch (e) {\n        return false;\n      }\n    };\n  } else {\n    // Look for any empty slot (no embedded addresses)\n    return function (sig: string, address: string): boolean {\n      return isEmptySignature(sig);\n    };\n  }\n}\n\nexport class Transaction extends BaseTransaction {\n  protected _flareTransaction: Tx;\n  public _type: TransactionType;\n  public _network: FlareNetwork;\n  public _networkID: number;\n  public _assetId: string;\n  public _blockchainID: string;\n  public _nodeID: string;\n  public _startTime: bigint;\n  public _endTime: bigint;\n  public _stakeAmount: bigint;\n  public _threshold = 2;\n  public _locktime = BigInt(0);\n  public _fromAddresses: Uint8Array[] = [];\n  public _to: Uint8Array[] = [];\n  public _rewardAddresses: Uint8Array[] = [];\n  public _utxos: DecodedUtxoObj[] = []; // Define proper type based on Flare's UTXO structure\n  public _fee: Partial<TransactionFee> = {};\n  // Store original raw signed bytes to preserve exact format when re-serializing\n  public _rawSignedBytes: Buffer | undefined;\n\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    this._network = coinConfig.network as FlareNetwork;\n    // Decode cb58-encoded asset ID to hex for use in transaction serialization\n    this._assetId = utils.cb58Decode(this._network.assetId).toString('hex');\n    this._blockchainID = this._network.blockchainID;\n    this._networkID = this._network.networkID;\n  }\n\n  get signature(): string[] {\n    if (!this.hasCredentials) {\n      return [];\n    }\n    return this.credentials[0].getSignatures().filter((s) => !isEmptySignature(s));\n  }\n\n  get credentials(): Credential[] {\n    return (this._flareTransaction as UnsignedTx)?.credentials;\n  }\n\n  get hasCredentials(): boolean {\n    return this.credentials !== undefined && this.credentials.length > 0;\n  }\n\n  /** @inheritdoc */\n  canSign({ key }: BaseKey): boolean {\n    return true;\n  }\n\n  async sign(keyPair: KeyPair): Promise<void> {\n    const prv = keyPair.getPrivateKey() as Uint8Array;\n    if (!prv) {\n      throw new SigningError('Missing private key');\n    }\n    if (!this._flareTransaction) {\n      throw new InvalidTransactionError('empty transaction to sign');\n    }\n    if (!this.hasCredentials) {\n      throw new InvalidTransactionError('empty credentials to sign');\n    }\n\n    const unsignedTx = this._flareTransaction as EVMUnsignedTx;\n    const unsignedBytes = unsignedTx.toBytes();\n    const publicKey = secp256k1.getPublicKey(prv);\n\n    // Derive both EVM and P-chain addresses from the public key\n    const evmAddressHex = new Address(secp256k1.publicKeyToEthAddress(publicKey)).toHex();\n\n    // P-chain address derivation: ripemd160(sha256(publicKey))\n    const sha256Hash = createHash('sha256').update(Buffer.from(publicKey)).digest();\n    const pChainAddressBuffer = createHash('ripemd160').update(sha256Hash).digest();\n    const pChainAddressHex = pChainAddressBuffer.toString('hex');\n\n    const addressMap = unsignedTx.getAddresses();\n\n    // Check for both EVM and P-chain address formats\n    const hasMatchingAddress = addressMap.some((addr) => {\n      const addrHex = Buffer.from(addr).toString('hex').toLowerCase();\n      return (\n        addrHex === utils.removeHexPrefix(evmAddressHex).toLowerCase() || addrHex === pChainAddressHex.toLowerCase()\n      );\n    });\n\n    if (hasMatchingAddress) {\n      const signature = await secp256k1.sign(unsignedBytes, prv);\n\n      let signatureSet = false;\n      // Use address-based slot matching (like AVAX-P)\n      let checkSign: CheckSignature | undefined = undefined;\n\n      for (const credential of unsignedTx.credentials) {\n        const signatures = credential.getSignatures();\n        if (checkSign === undefined) {\n          checkSign = generateSelectorSignature(signatures);\n        }\n\n        // Find the slot that matches this address\n        for (let i = 0; i < signatures.length; i++) {\n          const sig = signatures[i];\n          // Try matching with P-chain address first, then EVM address\n          if (checkSign(sig, pChainAddressHex) || checkSign(sig, utils.removeHexPrefix(evmAddressHex).toLowerCase())) {\n            credential.setSignature(i, signature);\n            signatureSet = true;\n            // Clear raw signed bytes since we've modified the transaction\n            this._rawSignedBytes = undefined;\n            break;\n          }\n        }\n\n        if (signatureSet) break;\n      }\n\n      if (!signatureSet) {\n        throw new SigningError('No matching signature slot found for this private key');\n      }\n    }\n  }\n\n  toBroadcastFormat(): string {\n    if (!this._flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\n    // If we have the original raw signed bytes, use them directly to preserve exact format\n    if (this._rawSignedBytes) {\n      return FlareUtils.bufferToHex(this._rawSignedBytes);\n    }\n    const unsignedTx = this._flareTransaction as UnsignedTx;\n    // For signed transactions, return the full signed tx with credentials\n    // Check signature.length for robustness\n    if (this.signature.length > 0) {\n      return FlareUtils.bufferToHex(unsignedTx.getSignedTx().toBytes());\n    }\n    // For unsigned transactions, return just the transaction bytes\n    return FlareUtils.bufferToHex(unsignedTx.toBytes());\n  }\n\n  toJson(): TxData {\n    if (!this._flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\n    return {\n      id: this.id,\n      inputs: this.inputs,\n      fromAddresses: this.fromAddresses,\n      threshold: this._threshold,\n      locktime: this._locktime.toString(),\n      type: this.type,\n      signatures: this.signature,\n      outputs: this.outputs,\n      changeOutputs: this.changeOutputs,\n    };\n  }\n\n  setTransaction(tx: Tx): void {\n    this._flareTransaction = tx as UnsignedTx;\n  }\n\n  setTransactionType(transactionType: TransactionType): void {\n    if (![TransactionType.AddPermissionlessValidator].includes(transactionType)) {\n      throw new Error(`Transaction type ${transactionType} is not supported`);\n    }\n    this._type = transactionType;\n  }\n\n  get signablePayload(): Buffer {\n    return utils.sha256((this._flareTransaction as UnsignedTx).toBytes());\n  }\n\n  get id(): string {\n    const bufferArray = utils.sha256((this._flareTransaction as UnsignedTx).toBytes());\n    return utils.cb58Encode(Buffer.from(bufferArray));\n  }\n\n  get fromAddresses(): string[] {\n    return this._fromAddresses.map((a) => FlareUtils.format(this._network.alias, this._network.hrp, a));\n  }\n\n  get rewardAddresses(): string[] {\n    return this._rewardAddresses.map((a) => FlareUtils.format(this._network.alias, this._network.hrp, a));\n  }\n\n  get fee(): TransactionFee {\n    return { fee: '0', ...this._fee };\n  }\n\n  get outputs(): Entry[] {\n    switch (this.type) {\n      case TransactionType.AddPermissionlessValidator:\n        return [\n          {\n            address: (\n              (this._flareTransaction as UnsignedTx).getTx() as pvmSerial.AddPermissionlessValidatorTx\n            ).subnetValidator.validator.nodeId.toString(),\n            value: (\n              (this._flareTransaction as UnsignedTx).getTx() as pvmSerial.AddPermissionlessValidatorTx\n            ).subnetValidator.validator.weight.toJSON(),\n          },\n        ];\n      default:\n        return [];\n    }\n  }\n\n  get changeOutputs(): Entry[] {\n    return (\n      (this._flareTransaction as UnsignedTx).getTx() as pvmSerial.AddPermissionlessValidatorTx\n    ).baseTx.outputs.map(utils.mapOutputToEntry(this._network));\n  }\n\n  explainTransaction(): TransactionExplanation {\n    const txJson = this.toJson();\n    const displayOrder = ['id', 'inputs', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'type'];\n\n    const outputAmount = txJson.outputs.reduce((p, n) => p + BigInt(n.value), BigInt(0)).toString();\n    const changeAmount = txJson.changeOutputs.reduce((p, n) => p + BigInt(n.value), BigInt(0)).toString();\n\n    let rewardAddresses;\n    if ([TransactionType.AddPermissionlessValidator].includes(txJson.type)) {\n      rewardAddresses = this.rewardAddresses;\n      displayOrder.splice(6, 0, 'rewardAddresses');\n    }\n\n    return {\n      displayOrder,\n      id: txJson.id,\n      inputs: txJson.inputs,\n      outputs: txJson.outputs.map((o) => ({ address: o.address, amount: o.value })),\n      outputAmount,\n      changeOutputs: txJson.changeOutputs.map((o) => ({ address: o.address, amount: o.value })),\n      changeAmount,\n      rewardAddresses,\n      fee: this.fee,\n      type: txJson.type,\n    };\n  }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"transactionBuilderFactory.d.ts","sourceRoot":"","sources":["../../../src/lib/transactionBuilderFactory.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,6BAA6B,EAAgB,MAAM,sBAAsB,CAAC;AACnF,OAAO,EAAgB,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC3E,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAG1D,qBAAa,yBAA0B,SAAQ,6BAA6B;IAC1E,SAAS,CAAC,aAAa,UAAS;gBAEpB,WAAW,EAAE,QAAQ,CAAC,UAAU,CAAC;IAI7C,kBAAkB;IAClB,IAAI,CAAC,GAAG,EAAE,MAAM,GAAG,kBAAkB;IA4CrC,kBAAkB;IAClB,kBAAkB,IAAI,kBAAkB;IAIxC;;OAEG;IACH,mBAAmB,IAAI,kBAAkB;IAIzC;;OAEG;IACH,mBAAmB,IAAI,kBAAkB;IAIzC;;OAEG;IACH,mBAAmB,IAAI,kBAAkB;IAIzC;;OAEG;IACH,mBAAmB,IAAI,kBAAkB;IAIzC,kBAAkB;IAClB,8BAA8B,IAAI,kBAAkB;CAGrD"}
1
+ {"version":3,"file":"transactionBuilderFactory.d.ts","sourceRoot":"","sources":["../../../src/lib/transactionBuilderFactory.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,6BAA6B,EAAgB,MAAM,sBAAsB,CAAC;AACnF,OAAO,EAAgB,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC3E,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAG1D,qBAAa,yBAA0B,SAAQ,6BAA6B;IAC1E,SAAS,CAAC,aAAa,UAAS;gBAEpB,WAAW,EAAE,QAAQ,CAAC,UAAU,CAAC;IAI7C,kBAAkB;IAClB,IAAI,CAAC,GAAG,EAAE,MAAM,GAAG,kBAAkB;IAgDrC,kBAAkB;IAClB,kBAAkB,IAAI,kBAAkB;IAIxC;;OAEG;IACH,mBAAmB,IAAI,kBAAkB;IAIzC;;OAEG;IACH,mBAAmB,IAAI,kBAAkB;IAIzC;;OAEG;IACH,mBAAmB,IAAI,kBAAkB;IAIzC;;OAEG;IACH,mBAAmB,IAAI,kBAAkB;IAIzC,kBAAkB;IAClB,8BAA8B,IAAI,kBAAkB;CAGrD"}
@@ -55,6 +55,11 @@ class TransactionBuilderFactory extends sdk_core_1.BaseTransactionBuilderFactory
55
55
  importBuilder.initBuilder(tx, rawBuffer);
56
56
  return importBuilder;
57
57
  }
58
+ else if (ExportInPTxBuilder_1.ExportInPTxBuilder.verifyTxType(tx._type)) {
59
+ const exportBuilder = this.getExportInPBuilder();
60
+ exportBuilder.initBuilder(tx, rawBuffer);
61
+ return exportBuilder;
62
+ }
58
63
  }
59
64
  throw new sdk_core_1.NotSupported('Transaction type not supported');
60
65
  }
@@ -92,4 +97,4 @@ class TransactionBuilderFactory extends sdk_core_1.BaseTransactionBuilderFactory
92
97
  }
93
98
  }
94
99
  exports.TransactionBuilderFactory = TransactionBuilderFactory;
95
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhbnNhY3Rpb25CdWlsZGVyRmFjdG9yeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9saWIvdHJhbnNhY3Rpb25CdWlsZGVyRmFjdG9yeS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQSxtREFBa0Y7QUFDbEYsbURBQW1GO0FBR25GLDZEQUEwRDtBQUMxRCw2REFBMEQ7QUFDMUQsNkRBQTBEO0FBQzFELDZEQUEwRDtBQUMxRCxvREFBNEI7QUFFNUIsTUFBYSx5QkFBMEIsU0FBUSx3Q0FBNkI7SUFHMUUsWUFBWSxXQUFpQztRQUMzQyxLQUFLLENBQUMsV0FBVyxDQUFDLENBQUM7UUFIWCxrQkFBYSxHQUFHLEtBQUssQ0FBQztJQUloQyxDQUFDO0lBRUQsa0JBQWtCO0lBQ2xCLElBQUksQ0FBQyxHQUFXO1FBQ2QsZUFBSyxDQUFDLHNCQUFzQixDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQ2xDLE1BQU0sUUFBUSxHQUFHLGVBQUssQ0FBQyxlQUFlLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDNUMsTUFBTSxTQUFTLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDL0MsSUFBSSxRQUF1QixDQUFDO1FBRTVCLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUMsT0FBdUIsQ0FBQztRQUN6RCxJQUFJLEVBQU8sQ0FBQztRQUNaLElBQUksQ0FBQztZQUNILFFBQVEsR0FBRyxLQUFLLENBQUM7WUFDakIsTUFBTSxVQUFVLEdBQUcsZUFBVSxDQUFDLGVBQWUsQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUNyRCxFQUFFLEdBQUcsVUFBVSxDQUFDLGlCQUFpQixDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQzdDLE1BQU0sWUFBWSxHQUFHLEVBQUUsQ0FBQyxlQUFlLEVBQUUsQ0FBQztZQUUxQyxJQUFJLFlBQVksS0FBSyxPQUFPLENBQUMsa0JBQWtCLEVBQUUsQ0FBQztnQkFDaEQsT0FBTyxDQUFDLEdBQUcsQ0FBQyxzQ0FBc0MsQ0FBQyxDQUFDO1lBQ3RELENBQUM7UUFDSCxDQUFDO1FBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNYLFFBQVEsR0FBRyxLQUFLLENBQUM7WUFDakIsTUFBTSxVQUFVLEdBQUcsZUFBVSxDQUFDLGVBQWUsQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUNyRCxFQUFFLEdBQUcsVUFBVSxDQUFDLGlCQUFpQixDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQzdDLE1BQU0sWUFBWSxHQUFHLEVBQUUsQ0FBQyxlQUFlLEVBQUUsQ0FBQztZQUUxQyxJQUFJLFlBQVksS0FBSyxPQUFPLENBQUMsWUFBWSxFQUFFLENBQUM7Z0JBQzFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsc0NBQXNDLENBQUMsQ0FBQztZQUN0RCxDQUFDO1FBQ0gsQ0FBQztRQUVELElBQUksUUFBUSxLQUFLLEtBQUssRUFBRSxDQUFDO1lBQ3ZCLElBQUksdUNBQWtCLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO2dCQUM5QyxNQUFNLGFBQWEsR0FBRyxJQUFJLENBQUMsbUJBQW1CLEVBQUUsQ0FBQztnQkFDakQsYUFBYSxDQUFDLFdBQVcsQ0FBQyxFQUF3QixFQUFFLFNBQVMsQ0FBQyxDQUFDO2dCQUMvRCxPQUFPLGFBQWEsQ0FBQztZQUN2QixDQUFDO1FBQ0gsQ0FBQzthQUFNLElBQUksUUFBUSxLQUFLLEtBQUssRUFBRSxDQUFDO1lBQzlCLElBQUksdUNBQWtCLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO2dCQUM5QyxNQUFNLGFBQWEsR0FBRyxJQUFJLENBQUMsbUJBQW1CLEVBQUUsQ0FBQztnQkFDakQsYUFBYSxDQUFDLFdBQVcsQ0FBQyxFQUF3QixFQUFFLFNBQVMsQ0FBQyxDQUFDO2dCQUMvRCxPQUFPLGFBQWEsQ0FBQztZQUN2QixDQUFDO1FBQ0gsQ0FBQztRQUNELE1BQU0sSUFBSSx1QkFBWSxDQUFDLGdDQUFnQyxDQUFDLENBQUM7SUFDM0QsQ0FBQztJQUVELGtCQUFrQjtJQUNsQixrQkFBa0I7UUFDaEIsTUFBTSxJQUFJLHVCQUFZLENBQUMsc0NBQXNDLENBQUMsQ0FBQztJQUNqRSxDQUFDO0lBRUQ7O09BRUc7SUFDSCxtQkFBbUI7UUFDakIsT0FBTyxJQUFJLHVDQUFrQixDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsQ0FBQztJQUNsRCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxtQkFBbUI7UUFDakIsT0FBTyxJQUFJLHVDQUFrQixDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsQ0FBQztJQUNsRCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxtQkFBbUI7UUFDakIsT0FBTyxJQUFJLHVDQUFrQixDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsQ0FBQztJQUNsRCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxtQkFBbUI7UUFDakIsT0FBTyxJQUFJLHVDQUFrQixDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsQ0FBQztJQUNsRCxDQUFDO0lBRUQsa0JBQWtCO0lBQ2xCLDhCQUE4QjtRQUM1QixNQUFNLElBQUksdUJBQVksQ0FBQyxxQ0FBcUMsQ0FBQyxDQUFDO0lBQ2hFLENBQUM7Q0FDRjtBQXpGRCw4REF5RkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB1dGlscyBhcyBGbGFyZVV0aWxzLCBldm1TZXJpYWwsIHB2bVNlcmlhbCB9IGZyb20gJ0BmbGFyZW5ldHdvcmsvZmxhcmVqcyc7XG5pbXBvcnQgeyBCYXNlVHJhbnNhY3Rpb25CdWlsZGVyRmFjdG9yeSwgTm90U3VwcG9ydGVkIH0gZnJvbSAnQGJpdGdvLWJldGEvc2RrLWNvcmUnO1xuaW1wb3J0IHsgRmxhcmVOZXR3b3JrLCBCYXNlQ29pbiBhcyBDb2luQ29uZmlnIH0gZnJvbSAnQGJpdGdvLWJldGEvc3RhdGljcyc7XG5pbXBvcnQgeyBUcmFuc2FjdGlvbkJ1aWxkZXIgfSBmcm9tICcuL3RyYW5zYWN0aW9uQnVpbGRlcic7XG5pbXBvcnQgeyBFeHBvcnRJblBUeEJ1aWxkZXIgfSBmcm9tICcuL0V4cG9ydEluUFR4QnVpbGRlcic7XG5pbXBvcnQgeyBJbXBvcnRJblBUeEJ1aWxkZXIgfSBmcm9tICcuL0ltcG9ydEluUFR4QnVpbGRlcic7XG5pbXBvcnQgeyBFeHBvcnRJbkNUeEJ1aWxkZXIgfSBmcm9tICcuL0V4cG9ydEluQ1R4QnVpbGRlcic7XG5pbXBvcnQgeyBJbXBvcnRJbkNUeEJ1aWxkZXIgfSBmcm9tICcuL0ltcG9ydEluQ1R4QnVpbGRlcic7XG5pbXBvcnQgdXRpbHMgZnJvbSAnLi91dGlscyc7XG5cbmV4cG9ydCBjbGFzcyBUcmFuc2FjdGlvbkJ1aWxkZXJGYWN0b3J5IGV4dGVuZHMgQmFzZVRyYW5zYWN0aW9uQnVpbGRlckZhY3Rvcnkge1xuICBwcm90ZWN0ZWQgcmVjb3ZlclNpZ25lciA9IGZhbHNlO1xuXG4gIGNvbnN0cnVjdG9yKF9jb2luQ29uZmlnOiBSZWFkb25seTxDb2luQ29uZmlnPikge1xuICAgIHN1cGVyKF9jb2luQ29uZmlnKTtcbiAgfVxuXG4gIC8qKiBAaW5oZXJpdGRvYyAqL1xuICBmcm9tKHJhdzogc3RyaW5nKTogVHJhbnNhY3Rpb25CdWlsZGVyIHtcbiAgICB1dGlscy52YWxpZGF0ZVJhd1RyYW5zYWN0aW9uKHJhdyk7XG4gICAgY29uc3QgcmF3Tm9IZXggPSB1dGlscy5yZW1vdmVIZXhQcmVmaXgocmF3KTtcbiAgICBjb25zdCByYXdCdWZmZXIgPSBCdWZmZXIuZnJvbShyYXdOb0hleCwgJ2hleCcpO1xuICAgIGxldCB0eFNvdXJjZTogJ0VWTScgfCAnUFZNJztcblxuICAgIGNvbnN0IG5ldHdvcmsgPSB0aGlzLl9jb2luQ29uZmlnLm5ldHdvcmsgYXMgRmxhcmVOZXR3b3JrO1xuICAgIGxldCB0eDogYW55O1xuICAgIHRyeSB7XG4gICAgICB0eFNvdXJjZSA9ICdFVk0nO1xuICAgICAgY29uc3QgZXZtTWFuYWdlciA9IEZsYXJlVXRpbHMuZ2V0TWFuYWdlckZvclZNKCdFVk0nKTtcbiAgICAgIHR4ID0gZXZtTWFuYWdlci51bnBhY2tUcmFuc2FjdGlvbihyYXdCdWZmZXIpO1xuICAgICAgY29uc3QgYmxvY2tjaGFpbklkID0gdHguZ2V0QmxvY2tjaGFpbklkKCk7XG5cbiAgICAgIGlmIChibG9ja2NoYWluSWQgPT09IG5ldHdvcmsuY0NoYWluQmxvY2tjaGFpbklEKSB7XG4gICAgICAgIGNvbnNvbGUubG9nKCdQYXJzZWQgYXMgRVZNIHRyYW5zYWN0aW9uIG9uIEMtQ2hhaW4nKTtcbiAgICAgIH1cbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICB0eFNvdXJjZSA9ICdQVk0nO1xuICAgICAgY29uc3QgcHZtTWFuYWdlciA9IEZsYXJlVXRpbHMuZ2V0TWFuYWdlckZvclZNKCdQVk0nKTtcbiAgICAgIHR4ID0gcHZtTWFuYWdlci51bnBhY2tUcmFuc2FjdGlvbihyYXdCdWZmZXIpO1xuICAgICAgY29uc3QgYmxvY2tjaGFpbklkID0gdHguZ2V0QmxvY2tjaGFpbklkKCk7XG5cbiAgICAgIGlmIChibG9ja2NoYWluSWQgPT09IG5ldHdvcmsuYmxvY2tjaGFpbklEKSB7XG4gICAgICAgIGNvbnNvbGUubG9nKCdQYXJzZWQgYXMgUFZNIHRyYW5zYWN0aW9uIG9uIFAtQ2hhaW4nKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBpZiAodHhTb3VyY2UgPT09ICdFVk0nKSB7XG4gICAgICBpZiAoRXhwb3J0SW5DVHhCdWlsZGVyLnZlcmlmeVR4VHlwZSh0eC5fdHlwZSkpIHtcbiAgICAgICAgY29uc3QgZXhwb3J0QnVpbGRlciA9IHRoaXMuZ2V0RXhwb3J0SW5DQnVpbGRlcigpO1xuICAgICAgICBleHBvcnRCdWlsZGVyLmluaXRCdWlsZGVyKHR4IGFzIGV2bVNlcmlhbC5FeHBvcnRUeCwgcmF3QnVmZmVyKTtcbiAgICAgICAgcmV0dXJuIGV4cG9ydEJ1aWxkZXI7XG4gICAgICB9XG4gICAgfSBlbHNlIGlmICh0eFNvdXJjZSA9PT0gJ1BWTScpIHtcbiAgICAgIGlmIChJbXBvcnRJblBUeEJ1aWxkZXIudmVyaWZ5VHhUeXBlKHR4Ll90eXBlKSkge1xuICAgICAgICBjb25zdCBpbXBvcnRCdWlsZGVyID0gdGhpcy5nZXRJbXBvcnRJblBCdWlsZGVyKCk7XG4gICAgICAgIGltcG9ydEJ1aWxkZXIuaW5pdEJ1aWxkZXIodHggYXMgcHZtU2VyaWFsLkltcG9ydFR4LCByYXdCdWZmZXIpO1xuICAgICAgICByZXR1cm4gaW1wb3J0QnVpbGRlcjtcbiAgICAgIH1cbiAgICB9XG4gICAgdGhyb3cgbmV3IE5vdFN1cHBvcnRlZCgnVHJhbnNhY3Rpb24gdHlwZSBub3Qgc3VwcG9ydGVkJyk7XG4gIH1cblxuICAvKiogQGluaGVyaXRkb2MgKi9cbiAgZ2V0VHJhbnNmZXJCdWlsZGVyKCk6IFRyYW5zYWN0aW9uQnVpbGRlciB7XG4gICAgdGhyb3cgbmV3IE5vdFN1cHBvcnRlZCgnVHJhbnNmZXIgaXMgbm90IHN1cHBvcnRlZCBpbiBQIENoYWluJyk7XG4gIH1cblxuICAvKipcbiAgICogRXhwb3J0IENyb3NzIGNoYWluIHRyYW5zZmVyXG4gICAqL1xuICBnZXRFeHBvcnRJblBCdWlsZGVyKCk6IEV4cG9ydEluUFR4QnVpbGRlciB7XG4gICAgcmV0dXJuIG5ldyBFeHBvcnRJblBUeEJ1aWxkZXIodGhpcy5fY29pbkNvbmZpZyk7XG4gIH1cblxuICAvKipcbiAgICogSW1wb3J0IENyb3NzIGNoYWluIHRyYW5zZmVyXG4gICAqL1xuICBnZXRJbXBvcnRJblBCdWlsZGVyKCk6IEltcG9ydEluUFR4QnVpbGRlciB7XG4gICAgcmV0dXJuIG5ldyBJbXBvcnRJblBUeEJ1aWxkZXIodGhpcy5fY29pbkNvbmZpZyk7XG4gIH1cblxuICAvKipcbiAgICogSW1wb3J0IGluIEMgY2hhaW4gQ3Jvc3MgY2hhaW4gdHJhbnNmZXJcbiAgICovXG4gIGdldEltcG9ydEluQ0J1aWxkZXIoKTogSW1wb3J0SW5DVHhCdWlsZGVyIHtcbiAgICByZXR1cm4gbmV3IEltcG9ydEluQ1R4QnVpbGRlcih0aGlzLl9jb2luQ29uZmlnKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBFeHBvcnQgaW4gQyBjaGFpbiBDcm9zcyBjaGFpbiB0cmFuc2ZlclxuICAgKi9cbiAgZ2V0RXhwb3J0SW5DQnVpbGRlcigpOiBFeHBvcnRJbkNUeEJ1aWxkZXIge1xuICAgIHJldHVybiBuZXcgRXhwb3J0SW5DVHhCdWlsZGVyKHRoaXMuX2NvaW5Db25maWcpO1xuICB9XG5cbiAgLyoqIEBpbmhlcml0ZG9jICovXG4gIGdldFdhbGxldEluaXRpYWxpemF0aW9uQnVpbGRlcigpOiBUcmFuc2FjdGlvbkJ1aWxkZXIge1xuICAgIHRocm93IG5ldyBOb3RTdXBwb3J0ZWQoJ1dhbGxldCBpbml0aWFsaXphdGlvbiBpcyBub3QgbmVlZGVkJyk7XG4gIH1cbn1cbiJdfQ==
100
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transactionBuilderFactory.js","sourceRoot":"","sources":["../../../src/lib/transactionBuilderFactory.ts"],"names":[],"mappings":";;;;;;AAAA,mDAAkF;AAClF,mDAAmF;AAGnF,6DAA0D;AAC1D,6DAA0D;AAC1D,6DAA0D;AAC1D,6DAA0D;AAC1D,oDAA4B;AAE5B,MAAa,yBAA0B,SAAQ,wCAA6B;IAG1E,YAAY,WAAiC;QAC3C,KAAK,CAAC,WAAW,CAAC,CAAC;QAHX,kBAAa,GAAG,KAAK,CAAC;IAIhC,CAAC;IAED,kBAAkB;IAClB,IAAI,CAAC,GAAW;QACd,eAAK,CAAC,sBAAsB,CAAC,GAAG,CAAC,CAAC;QAClC,MAAM,QAAQ,GAAG,eAAK,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC;QAC5C,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;QAC/C,IAAI,QAAuB,CAAC;QAE5B,MAAM,OAAO,GAAG,IAAI,CAAC,WAAW,CAAC,OAAuB,CAAC;QACzD,IAAI,EAAO,CAAC;QACZ,IAAI,CAAC;YACH,QAAQ,GAAG,KAAK,CAAC;YACjB,MAAM,UAAU,GAAG,eAAU,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;YACrD,EAAE,GAAG,UAAU,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC;YAC7C,MAAM,YAAY,GAAG,EAAE,CAAC,eAAe,EAAE,CAAC;YAE1C,IAAI,YAAY,KAAK,OAAO,CAAC,kBAAkB,EAAE,CAAC;gBAChD,OAAO,CAAC,GAAG,CAAC,sCAAsC,CAAC,CAAC;YACtD,CAAC;QACH,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,QAAQ,GAAG,KAAK,CAAC;YACjB,MAAM,UAAU,GAAG,eAAU,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;YACrD,EAAE,GAAG,UAAU,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC;YAC7C,MAAM,YAAY,GAAG,EAAE,CAAC,eAAe,EAAE,CAAC;YAE1C,IAAI,YAAY,KAAK,OAAO,CAAC,YAAY,EAAE,CAAC;gBAC1C,OAAO,CAAC,GAAG,CAAC,sCAAsC,CAAC,CAAC;YACtD,CAAC;QACH,CAAC;QAED,IAAI,QAAQ,KAAK,KAAK,EAAE,CAAC;YACvB,IAAI,uCAAkB,CAAC,YAAY,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC9C,MAAM,aAAa,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;gBACjD,aAAa,CAAC,WAAW,CAAC,EAAwB,EAAE,SAAS,CAAC,CAAC;gBAC/D,OAAO,aAAa,CAAC;YACvB,CAAC;QACH,CAAC;aAAM,IAAI,QAAQ,KAAK,KAAK,EAAE,CAAC;YAC9B,IAAI,uCAAkB,CAAC,YAAY,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC9C,MAAM,aAAa,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;gBACjD,aAAa,CAAC,WAAW,CAAC,EAAwB,EAAE,SAAS,CAAC,CAAC;gBAC/D,OAAO,aAAa,CAAC;YACvB,CAAC;iBAAM,IAAI,uCAAkB,CAAC,YAAY,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC;gBACrD,MAAM,aAAa,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;gBACjD,aAAa,CAAC,WAAW,CAAC,EAAwB,EAAE,SAAS,CAAC,CAAC;gBAC/D,OAAO,aAAa,CAAC;YACvB,CAAC;QACH,CAAC;QACD,MAAM,IAAI,uBAAY,CAAC,gCAAgC,CAAC,CAAC;IAC3D,CAAC;IAED,kBAAkB;IAClB,kBAAkB;QAChB,MAAM,IAAI,uBAAY,CAAC,sCAAsC,CAAC,CAAC;IACjE,CAAC;IAED;;OAEG;IACH,mBAAmB;QACjB,OAAO,IAAI,uCAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IAClD,CAAC;IAED;;OAEG;IACH,mBAAmB;QACjB,OAAO,IAAI,uCAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IAClD,CAAC;IAED;;OAEG;IACH,mBAAmB;QACjB,OAAO,IAAI,uCAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IAClD,CAAC;IAED;;OAEG;IACH,mBAAmB;QACjB,OAAO,IAAI,uCAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IAClD,CAAC;IAED,kBAAkB;IAClB,8BAA8B;QAC5B,MAAM,IAAI,uBAAY,CAAC,qCAAqC,CAAC,CAAC;IAChE,CAAC;CACF;AA7FD,8DA6FC","sourcesContent":["import { utils as FlareUtils, evmSerial, pvmSerial } from '@flarenetwork/flarejs';\nimport { BaseTransactionBuilderFactory, NotSupported } from '@bitgo-beta/sdk-core';\nimport { FlareNetwork, BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { TransactionBuilder } from './transactionBuilder';\nimport { ExportInPTxBuilder } from './ExportInPTxBuilder';\nimport { ImportInPTxBuilder } from './ImportInPTxBuilder';\nimport { ExportInCTxBuilder } from './ExportInCTxBuilder';\nimport { ImportInCTxBuilder } from './ImportInCTxBuilder';\nimport utils from './utils';\n\nexport class TransactionBuilderFactory extends BaseTransactionBuilderFactory {\n  protected recoverSigner = false;\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n  }\n\n  /** @inheritdoc */\n  from(raw: string): TransactionBuilder {\n    utils.validateRawTransaction(raw);\n    const rawNoHex = utils.removeHexPrefix(raw);\n    const rawBuffer = Buffer.from(rawNoHex, 'hex');\n    let txSource: 'EVM' | 'PVM';\n\n    const network = this._coinConfig.network as FlareNetwork;\n    let tx: any;\n    try {\n      txSource = 'EVM';\n      const evmManager = FlareUtils.getManagerForVM('EVM');\n      tx = evmManager.unpackTransaction(rawBuffer);\n      const blockchainId = tx.getBlockchainId();\n\n      if (blockchainId === network.cChainBlockchainID) {\n        console.log('Parsed as EVM transaction on C-Chain');\n      }\n    } catch (e) {\n      txSource = 'PVM';\n      const pvmManager = FlareUtils.getManagerForVM('PVM');\n      tx = pvmManager.unpackTransaction(rawBuffer);\n      const blockchainId = tx.getBlockchainId();\n\n      if (blockchainId === network.blockchainID) {\n        console.log('Parsed as PVM transaction on P-Chain');\n      }\n    }\n\n    if (txSource === 'EVM') {\n      if (ExportInCTxBuilder.verifyTxType(tx._type)) {\n        const exportBuilder = this.getExportInCBuilder();\n        exportBuilder.initBuilder(tx as evmSerial.ExportTx, rawBuffer);\n        return exportBuilder;\n      }\n    } else if (txSource === 'PVM') {\n      if (ImportInPTxBuilder.verifyTxType(tx._type)) {\n        const importBuilder = this.getImportInPBuilder();\n        importBuilder.initBuilder(tx as pvmSerial.ImportTx, rawBuffer);\n        return importBuilder;\n      } else if (ExportInPTxBuilder.verifyTxType(tx._type)) {\n        const exportBuilder = this.getExportInPBuilder();\n        exportBuilder.initBuilder(tx as pvmSerial.ExportTx, rawBuffer);\n        return exportBuilder;\n      }\n    }\n    throw new NotSupported('Transaction type not supported');\n  }\n\n  /** @inheritdoc */\n  getTransferBuilder(): TransactionBuilder {\n    throw new NotSupported('Transfer is not supported in P Chain');\n  }\n\n  /**\n   * Export Cross chain transfer\n   */\n  getExportInPBuilder(): ExportInPTxBuilder {\n    return new ExportInPTxBuilder(this._coinConfig);\n  }\n\n  /**\n   * Import Cross chain transfer\n   */\n  getImportInPBuilder(): ImportInPTxBuilder {\n    return new ImportInPTxBuilder(this._coinConfig);\n  }\n\n  /**\n   * Import in C chain Cross chain transfer\n   */\n  getImportInCBuilder(): ImportInCTxBuilder {\n    return new ImportInCTxBuilder(this._coinConfig);\n  }\n\n  /**\n   * Export in C chain Cross chain transfer\n   */\n  getExportInCBuilder(): ExportInCTxBuilder {\n    return new ExportInCTxBuilder(this._coinConfig);\n  }\n\n  /** @inheritdoc */\n  getWalletInitializationBuilder(): TransactionBuilder {\n    throw new NotSupported('Wallet initialization is not needed');\n  }\n}\n"]}
@@ -50,6 +50,18 @@ export declare class Utils implements BaseUtils {
50
50
  * Creates a new signature object
51
51
  */
52
52
  createNewSig(sigHex: string): Signature;
53
+ /**
54
+ * Creates an empty signature with embedded address for signature slot identification.
55
+ * The address is embedded at position 90 (after the first 45 zero bytes).
56
+ * This allows the signing logic to determine which slot belongs to which address.
57
+ * @param addressHex The 20-byte address in hex format (40 chars, without 0x prefix)
58
+ */
59
+ createEmptySigWithAddress(addressHex: string): Signature;
60
+ /**
61
+ * Extracts the embedded address from an empty signature.
62
+ * Returns the address hex string (40 chars) or empty string if not found.
63
+ */
64
+ getAddressFromEmptySig(sig: string): string;
53
65
  /**
54
66
  * Computes SHA256 hash
55
67
  */
@@ -1 +1 @@
1
- {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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"}
@@ -201,6 +201,31 @@ class Utils {
201
201
  const buffer = buffer_1.Buffer.from(sigHex.padStart(130, '0'), 'hex');
202
202
  return new flarejs_1.Signature(buffer);
203
203
  }
204
+ /**
205
+ * Creates an empty signature with embedded address for signature slot identification.
206
+ * The address is embedded at position 90 (after the first 45 zero bytes).
207
+ * This allows the signing logic to determine which slot belongs to which address.
208
+ * @param addressHex The 20-byte address in hex format (40 chars, without 0x prefix)
209
+ */
210
+ createEmptySigWithAddress(addressHex) {
211
+ // First 45 bytes (90 hex chars) are zeros, followed by 20-byte address (40 hex chars)
212
+ const cleanAddr = this.removeHexPrefix(addressHex).toLowerCase();
213
+ const sigHex = '0'.repeat(90) + cleanAddr.padStart(40, '0');
214
+ const buffer = buffer_1.Buffer.from(sigHex, 'hex');
215
+ return new flarejs_1.Signature(buffer);
216
+ }
217
+ /**
218
+ * Extracts the embedded address from an empty signature.
219
+ * Returns the address hex string (40 chars) or empty string if not found.
220
+ */
221
+ getAddressFromEmptySig(sig) {
222
+ const cleanSig = this.removeHexPrefix(sig);
223
+ if (cleanSig.length >= 130) {
224
+ // Address is at position 90-130 (last 40 hex chars = 20 bytes)
225
+ return cleanSig.substring(90, 130).toLowerCase();
226
+ }
227
+ return '';
228
+ }
204
229
  /**
205
230
  * Computes SHA256 hash
206
231
  */
@@ -486,4 +511,4 @@ class Utils {
486
511
  exports.Utils = Utils;
487
512
  const utils = new Utils();
488
513
  exports.default = utils;
489
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Signature,\n  TransferableOutput,\n  TransferOutput,\n  TypeSymbols,\n  Id,\n  Credential,\n  utils as FlareUtils,\n} from '@flarenetwork/flarejs';\nimport {\n  BaseUtils,\n  Entry,\n  InvalidTransactionError,\n  isValidXprv,\n  isValidXpub,\n  NotImplementedError,\n  ParseTransactionError,\n} from '@bitgo-beta/sdk-core';\nimport { FlareNetwork } from '@bitgo-beta/statics';\nimport { Buffer } from 'buffer';\nimport { createHash } from 'crypto';\nimport { ecc } from '@bitgo-beta/secp256k1';\nimport { ADDRESS_SEPARATOR, Output, DeprecatedTx } from './iface';\nimport bs58 from 'bs58';\nimport { bech32 } from 'bech32';\n\nexport class Utils implements BaseUtils {\n  /**\n   * Check if addresses in wallet match UTXO output addresses\n   */\n  public includeIn(walletAddresses: string[], otxoOutputAddresses: string[]): boolean {\n    return walletAddresses.map((a) => otxoOutputAddresses.includes(a)).reduce((a, b) => a && b, true);\n  }\n\n  /**\n   * Validates a Flare address or array of addresses\n   * @param {string | string[]} address - address(es) to validate\n   * @returns {boolean} - validation result\n   */\n  isValidAddress(address: string | string[]): boolean {\n    const addressArr: string[] = Array.isArray(address) ? address : address.split('~');\n\n    for (const address of addressArr) {\n      if (!this.isValidAddressRegex(address)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  // Regex patterns\n  // export const ADDRESS_REGEX = /^(^P||NodeID)-[a-zA-Z0-9]+$/;\n  // export const HEX_REGEX = /^(0x){0,1}([0-9a-f])+$/i;\n\n  private isValidAddressRegex(address: string): boolean {\n    return /^(^P||NodeID)-[a-zA-Z0-9]+$/.test(address);\n  }\n\n  /**\n   * Validates a block ID\n   * @param {string} hash - block ID to validate\n   * @returns {boolean} - validation result\n   */\n  isValidBlockId(hash: string): boolean {\n    try {\n      const decoded = Buffer.from(hash, 'hex');\n      return decoded.length === 32;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Validates a public key\n   * @param {string} pub - public key to validate\n   * @returns {boolean} - validation result\n   */\n  isValidPublicKey(pub: string): boolean {\n    if (isValidXpub(pub)) return true;\n\n    let pubBuf: Buffer;\n    if (pub.length === 50) {\n      try {\n        pubBuf = Buffer.from(pub, 'hex');\n      } catch {\n        return false;\n      }\n    } else {\n      if (pub.length !== 66 && pub.length !== 130) return false;\n\n      const firstByte = pub.slice(0, 2);\n      if (pub.length === 130 && firstByte !== '04') return false;\n      if (pub.length === 66 && firstByte !== '02' && firstByte !== '03') return false;\n      if (!this.allHexChars(pub)) return false;\n\n      pubBuf = Buffer.from(pub, 'hex');\n    }\n\n    try {\n      ecc.isPoint(pubBuf);\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Validates a private key\n   * @param {string} prv - private key to validate\n   * @returns {boolean} - validation result\n   */\n  isValidPrivateKey(prv: string): boolean {\n    if (isValidXprv(prv)) return true;\n    if (prv.length !== 64 && prv.length !== 66) return false;\n    if (prv.length === 66 && prv.slice(64) !== '01') return false;\n    return this.allHexChars(prv);\n  }\n\n  /**\n   * Checks if a string contains only hex characters\n   */\n  allHexChars(str: string): boolean {\n    return /^(0x){0,1}([0-9a-f])+$/i.test(str);\n  }\n\n  /**\n   * Creates a signature using the Flare network parameters\n   * Returns a 65-byte signature (64 bytes signature + 1 byte recovery parameter)\n   */\n  createSignature(network: FlareNetwork, message: Buffer, prv: Buffer): Buffer {\n    const messageHash = this.sha256(message);\n    const signature = ecc.sign(messageHash, prv);\n\n    // Get the public key from the private key for recovery parameter determination\n    const publicKey = ecc.pointFromScalar(prv, true);\n    if (!publicKey) {\n      throw new Error('Failed to derive public key from private key');\n    }\n\n    // Try recovery with param 0 and 1 to find the correct one\n    let recoveryParam = 0;\n    for (let i = 0; i <= 1; i++) {\n      const recovered = ecc.recoverPublicKey(messageHash, signature, i, true);\n      if (recovered && Buffer.from(recovered).equals(Buffer.from(publicKey))) {\n        recoveryParam = i;\n        break;\n      }\n    }\n\n    // Append recovery parameter to create 65-byte signature\n    const sigWithRecovery = Buffer.alloc(65);\n    Buffer.from(signature).copy(sigWithRecovery, 0);\n    sigWithRecovery[64] = recoveryParam;\n\n    return sigWithRecovery;\n  }\n\n  /**\n   * Verifies a signature\n   */\n  verifySignature(network: FlareNetwork, message: Buffer, signature: Buffer, publicKey: Buffer): boolean {\n    try {\n      const messageHash = this.sha256(message);\n      return ecc.verify(signature, messageHash, publicKey);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Creates a new signature object\n   */\n  createNewSig(sigHex: string): Signature {\n    const buffer = Buffer.from(sigHex.padStart(130, '0'), 'hex');\n    return new Signature(buffer);\n  }\n\n  /**\n   * Computes SHA256 hash\n   */\n  sha256(buf: Uint8Array): Buffer {\n    return createHash('sha256').update(buf).digest();\n  }\n\n  /**\n   * Validates raw transaction format\n   */\n  validateRawTransaction(rawTransaction: string): void {\n    if (!rawTransaction) {\n      throw new InvalidTransactionError('Raw transaction is empty');\n    }\n    if (!this.allHexChars(rawTransaction)) {\n      throw new ParseTransactionError('Raw transaction is not hex string');\n    }\n  }\n\n  /**\n   * Checks if output is TransferableOutput type\n   */\n  isTransferableOutput(output: Output): output is TransferableOutput {\n    return output?._type === TypeSymbols.TransferableOutput;\n  }\n\n  /**\n   * Maps outputs to entry format\n   */\n  mapOutputToEntry(network: FlareNetwork): (Output) => Entry {\n    return (output: Output) => {\n      if (this.isTransferableOutput(output)) {\n        const outputAmount = output.amount();\n        const address = (output.output as TransferOutput)\n          .getOwners()\n          .map((a) => this.addressToString(network.hrp, network.alias, Buffer.from(a)))\n          .sort()\n          .join(ADDRESS_SEPARATOR);\n        return {\n          value: outputAmount.toString(),\n          address,\n        };\n      } else {\n        throw new Error('Invalid output type');\n      }\n    };\n  }\n\n  /**\n   * Removes 0x prefix from hex string\n   */\n  removeHexPrefix(hex: string): string {\n    return hex.startsWith('0x') ? hex.substring(2) : hex;\n  }\n\n  /**\n   * Converts output index to buffer\n   */\n  outputidxNumberToBuffer(outputidx: string): Buffer {\n    return Buffer.from(Number(outputidx).toString(16).padStart(8, '0'), 'hex');\n  }\n\n  /**\n   * Converts output index buffer to number string\n   */\n  outputidxBufferToNumber(outputidx: Buffer): string {\n    return parseInt(outputidx.toString('hex'), 16).toString();\n  }\n\n  // Required by BaseUtils interface but not implemented\n  isValidSignature(signature: string): boolean {\n    throw new NotImplementedError('isValidSignature not implemented');\n  }\n\n  isValidTransactionId(txId: string): boolean {\n    throw new NotImplementedError('isValidTransactionId not implemented');\n  }\n\n  /**\n   * Helper method to convert address components to string\n   */\n  public addressToString = (hrp: string, prefix: string, address: Buffer): string => {\n    // Convert the address bytes to 5-bit words for bech32 encoding\n    const words = bech32.toWords(address);\n    // Create the full bech32 address with format: P-{hrp}1{bech32_encoded_address}\n    return `${prefix}-${bech32.encode(hrp, words)}`;\n  };\n\n  /**\n   * Decodes a base58 string with checksum to a Buffer\n   */\n  public cb58Decode(str: string): Buffer {\n    const decoded = bs58.decode(str);\n    if (!this.validateChecksum(Buffer.from(decoded))) {\n      throw new Error('Invalid checksum');\n    }\n    return Buffer.from(decoded.slice(0, decoded.length - 4));\n  }\n\n  /**\n   * Validates a checksum on a Buffer and returns true if valid, false if not\n   */\n  private validateChecksum(buff: Buffer): boolean {\n    const hashSlice = buff.slice(buff.length - 4);\n    const calculatedHashSlice = createHash('sha256')\n      .update(buff.slice(0, buff.length - 4))\n      .digest()\n      .slice(28);\n    return hashSlice.toString('hex') === calculatedHashSlice.toString('hex');\n  }\n\n  /**\n   * Encodes a Buffer as a base58 string with checksum\n   */\n  public cb58Encode(bytes: Buffer): string {\n    const withChecksum = this.addChecksum(bytes);\n    return bs58.encode(withChecksum);\n  }\n\n  /**\n   * Adds a checksum to a Buffer and returns the concatenated result\n   */\n  private addChecksum(buff: Buffer): Buffer {\n    const hashSlice = createHash('sha256').update(buff).digest().slice(28);\n    return Buffer.concat([buff, hashSlice]);\n  }\n\n  // In utils.ts, add this method to the Utils class:\n\n  /**\n   * Parse an address string into a Buffer\n   * @param address - The address to parse\n   * @returns Buffer containing the parsed address\n   */\n  //TODO: need check and validate this method\n  public parseAddress = (address: string): Buffer => {\n    return this.stringToAddress(address);\n  };\n\n  public stringToAddress = (address: string, hrp?: string): Buffer => {\n    // Handle hex addresses\n    if (address.startsWith('0x')) {\n      return Buffer.from(address.slice(2), 'hex');\n    }\n\n    // Handle raw hex without 0x prefix\n    if (/^[0-9a-fA-F]{40}$/.test(address)) {\n      return Buffer.from(address, 'hex');\n    }\n\n    // Handle Bech32 addresses\n    const parts = address.trim().split('-');\n    if (parts.length < 2) {\n      throw new Error('Error - Valid address should include -');\n    }\n\n    const split = parts[1].lastIndexOf('1');\n    if (split < 0) {\n      throw new Error('Error - Valid bech32 address must include separator (1)');\n    }\n\n    const humanReadablePart = parts[1].slice(0, split);\n    if (humanReadablePart !== 'flare' && humanReadablePart !== 'costwo') {\n      throw new Error('Error - Invalid HRP');\n    }\n\n    return Buffer.from(bech32.fromWords(bech32.decode(parts[1]).words));\n  };\n\n  /**\n   * Check if tx is for the blockchainId\n   *\n   * @param {DeprecatedTx} tx\n   * @param {string} blockchainId\n   * @returns true if tx is for blockchainId\n   */\n  // TODO: remove DeprecatedTx usage\n  isTransactionOf(tx: DeprecatedTx, blockchainId: string): boolean {\n    // FlareJS equivalent - this would need proper CB58 encoding implementation\n    try {\n      const txRecord = tx as unknown as Record<string, unknown>;\n      const unsignedTx = (txRecord.getUnsignedTx as () => Record<string, unknown>)();\n      const transaction = (unsignedTx.getTransaction as () => Record<string, unknown>)();\n      const txBlockchainId = (transaction.getBlockchainID as () => unknown)();\n      return Buffer.from(txBlockchainId as string).toString('hex') === blockchainId;\n    } catch (error) {\n      return false;\n    }\n  }\n\n  flareIdString(value: string): Id {\n    return new Id(Buffer.from(value, 'hex'));\n  }\n\n  /**\n   * Extract credentials from raw transaction bytes.\n   * Signed transactions have credentials appended after the transaction body.\n   * This function handles both checking for credentials and extracting them.\n   *\n   * @param rawBytes - The full raw transaction bytes\n   * @param tx - The parsed transaction (must have toBytes method)\n   * @param vmType - The VM type ('EVM' or 'PVM') to get the correct codec\n   * @returns Object with hasCredentials flag and credentials array\n   */\n  extractCredentialsFromRawBytes(\n    rawBytes: Buffer,\n    tx: { toBytes(codec: unknown): Uint8Array },\n    vmType: 'EVM' | 'PVM' = 'EVM'\n  ): { hasCredentials: boolean; credentials: Credential[] } {\n    try {\n      // Get the size of the transaction without credentials using the default codec\n      const codec = FlareUtils.getManagerForVM(vmType).getDefaultCodec();\n      const txBytes = tx.toBytes(codec);\n      const txSize = txBytes.length;\n\n      // If raw bytes are not longer than tx bytes, there are no credentials\n      if (rawBytes.length <= txSize) {\n        return { hasCredentials: false, credentials: [] };\n      }\n\n      // Extract credential bytes (everything after the transaction)\n      const credentialBytes = rawBytes.slice(txSize);\n\n      // Parse credentials\n      // Format: [num_credentials: 4 bytes] [credentials...]\n      if (credentialBytes.length < 4) {\n        return { hasCredentials: false, credentials: [] };\n      }\n\n      const numCredentials = credentialBytes.readUInt32BE(0);\n\n      // Check if there are credentials in raw bytes (for hasCredentials flag)\n      const hasCredentials = numCredentials > 0;\n\n      if (numCredentials === 0) {\n        return { hasCredentials: false, credentials: [] };\n      }\n\n      const credentials: Credential[] = [];\n      let offset = 4;\n\n      for (let i = 0; i < numCredentials; i++) {\n        if (offset + 8 > credentialBytes.length) {\n          break;\n        }\n\n        // Read type ID (4 bytes) - Type ID 9 = secp256k1 credential\n        const typeId = credentialBytes.readUInt32BE(offset);\n        offset += 4;\n\n        // Validate credential type (9 = secp256k1)\n        if (typeId !== 9) {\n          continue; // Skip unsupported credential types\n        }\n\n        // Read number of signatures (4 bytes)\n        const numSigs = credentialBytes.readUInt32BE(offset);\n        offset += 4;\n\n        // Parse all signatures for this credential\n        const signatures: Signature[] = [];\n        for (let j = 0; j < numSigs; j++) {\n          if (offset + 65 > credentialBytes.length) {\n            break;\n          }\n          // Each signature is 65 bytes (64 bytes signature + 1 byte recovery)\n          const sigBytes = Buffer.from(credentialBytes.slice(offset, offset + 65));\n          signatures.push(new Signature(sigBytes));\n          offset += 65;\n        }\n\n        // Create credential with the parsed signatures\n        if (signatures.length > 0) {\n          credentials.push(new Credential(signatures));\n        }\n      }\n\n      return { hasCredentials, credentials };\n    } catch (e) {\n      // If parsing fails, return no credentials\n      return { hasCredentials: false, credentials: [] };\n    }\n  }\n\n  /**\n   * Parse credentials from raw bytes at a specific offset\n   * This is useful when the standard extraction fails due to serialization differences\n   * @param rawBytes Raw transaction bytes including credentials\n   * @param offset Byte offset where credentials start\n   * @returns Array of parsed credentials\n   */\n  parseCredentialsAtOffset(rawBytes: Buffer, offset: number): Credential[] {\n    try {\n      if (rawBytes.length <= offset + 4) {\n        return [];\n      }\n\n      const credentialBytes = rawBytes.slice(offset);\n      const numCredentials = credentialBytes.readUInt32BE(0);\n\n      if (numCredentials === 0) {\n        return [];\n      }\n\n      const credentials: Credential[] = [];\n      let pos = 4;\n\n      for (let i = 0; i < numCredentials; i++) {\n        if (pos + 8 > credentialBytes.length) {\n          break;\n        }\n\n        // Read type ID (4 bytes) - Type ID 9 = secp256k1 credential\n        const typeId = credentialBytes.readUInt32BE(pos);\n        pos += 4;\n\n        if (typeId !== 9) {\n          continue;\n        }\n\n        // Read number of signatures (4 bytes)\n        const numSigs = credentialBytes.readUInt32BE(pos);\n        pos += 4;\n\n        // Parse all signatures for this credential\n        const signatures: Signature[] = [];\n        for (let j = 0; j < numSigs; j++) {\n          if (pos + 65 > credentialBytes.length) {\n            break;\n          }\n          const sigBytes = Buffer.from(credentialBytes.slice(pos, pos + 65));\n          signatures.push(new Signature(sigBytes));\n          pos += 65;\n        }\n\n        if (signatures.length > 0) {\n          credentials.push(new Credential(signatures));\n        }\n      }\n\n      return credentials;\n    } catch {\n      return [];\n    }\n  }\n\n  /**\n   * FlareJS wrapper to recover signature\n   * @param network\n   * @param message\n   * @param signature\n   * @return recovered public key\n   */\n  recoverySignature(network: FlareNetwork, message: Buffer, signature: Buffer): Buffer {\n    try {\n      // Hash the message first - must match the hash used in signing\n      const messageHash = createHash('sha256').update(message).digest();\n\n      // Extract recovery parameter and signature\n      if (signature.length !== 65) {\n        throw new Error('Invalid signature length - expected 65 bytes (64 bytes signature + 1 byte recovery)');\n      }\n\n      const recoveryParam = signature[64];\n      const sigOnly = signature.slice(0, 64);\n\n      // Recover public key using the provided recovery parameter\n      const recovered = ecc.recoverPublicKey(messageHash, sigOnly, recoveryParam, true);\n      if (!recovered) {\n        throw new Error('Failed to recover public key');\n      }\n\n      return Buffer.from(recovered);\n    } catch (error) {\n      throw new Error(`Failed to recover signature: ${error}`);\n    }\n  }\n}\n\nconst utils = new Utils();\nexport default utils;\n"]}
514
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Signature,\n  TransferableOutput,\n  TransferOutput,\n  TypeSymbols,\n  Id,\n  Credential,\n  utils as FlareUtils,\n} from '@flarenetwork/flarejs';\nimport {\n  BaseUtils,\n  Entry,\n  InvalidTransactionError,\n  isValidXprv,\n  isValidXpub,\n  NotImplementedError,\n  ParseTransactionError,\n} from '@bitgo-beta/sdk-core';\nimport { FlareNetwork } from '@bitgo-beta/statics';\nimport { Buffer } from 'buffer';\nimport { createHash } from 'crypto';\nimport { ecc } from '@bitgo-beta/secp256k1';\nimport { ADDRESS_SEPARATOR, Output, DeprecatedTx } from './iface';\nimport bs58 from 'bs58';\nimport { bech32 } from 'bech32';\n\nexport class Utils implements BaseUtils {\n  /**\n   * Check if addresses in wallet match UTXO output addresses\n   */\n  public includeIn(walletAddresses: string[], otxoOutputAddresses: string[]): boolean {\n    return walletAddresses.map((a) => otxoOutputAddresses.includes(a)).reduce((a, b) => a && b, true);\n  }\n\n  /**\n   * Validates a Flare address or array of addresses\n   * @param {string | string[]} address - address(es) to validate\n   * @returns {boolean} - validation result\n   */\n  isValidAddress(address: string | string[]): boolean {\n    const addressArr: string[] = Array.isArray(address) ? address : address.split('~');\n\n    for (const address of addressArr) {\n      if (!this.isValidAddressRegex(address)) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  // Regex patterns\n  // export const ADDRESS_REGEX = /^(^P||NodeID)-[a-zA-Z0-9]+$/;\n  // export const HEX_REGEX = /^(0x){0,1}([0-9a-f])+$/i;\n\n  private isValidAddressRegex(address: string): boolean {\n    return /^(^P||NodeID)-[a-zA-Z0-9]+$/.test(address);\n  }\n\n  /**\n   * Validates a block ID\n   * @param {string} hash - block ID to validate\n   * @returns {boolean} - validation result\n   */\n  isValidBlockId(hash: string): boolean {\n    try {\n      const decoded = Buffer.from(hash, 'hex');\n      return decoded.length === 32;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Validates a public key\n   * @param {string} pub - public key to validate\n   * @returns {boolean} - validation result\n   */\n  isValidPublicKey(pub: string): boolean {\n    if (isValidXpub(pub)) return true;\n\n    let pubBuf: Buffer;\n    if (pub.length === 50) {\n      try {\n        pubBuf = Buffer.from(pub, 'hex');\n      } catch {\n        return false;\n      }\n    } else {\n      if (pub.length !== 66 && pub.length !== 130) return false;\n\n      const firstByte = pub.slice(0, 2);\n      if (pub.length === 130 && firstByte !== '04') return false;\n      if (pub.length === 66 && firstByte !== '02' && firstByte !== '03') return false;\n      if (!this.allHexChars(pub)) return false;\n\n      pubBuf = Buffer.from(pub, 'hex');\n    }\n\n    try {\n      ecc.isPoint(pubBuf);\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Validates a private key\n   * @param {string} prv - private key to validate\n   * @returns {boolean} - validation result\n   */\n  isValidPrivateKey(prv: string): boolean {\n    if (isValidXprv(prv)) return true;\n    if (prv.length !== 64 && prv.length !== 66) return false;\n    if (prv.length === 66 && prv.slice(64) !== '01') return false;\n    return this.allHexChars(prv);\n  }\n\n  /**\n   * Checks if a string contains only hex characters\n   */\n  allHexChars(str: string): boolean {\n    return /^(0x){0,1}([0-9a-f])+$/i.test(str);\n  }\n\n  /**\n   * Creates a signature using the Flare network parameters\n   * Returns a 65-byte signature (64 bytes signature + 1 byte recovery parameter)\n   */\n  createSignature(network: FlareNetwork, message: Buffer, prv: Buffer): Buffer {\n    const messageHash = this.sha256(message);\n    const signature = ecc.sign(messageHash, prv);\n\n    // Get the public key from the private key for recovery parameter determination\n    const publicKey = ecc.pointFromScalar(prv, true);\n    if (!publicKey) {\n      throw new Error('Failed to derive public key from private key');\n    }\n\n    // Try recovery with param 0 and 1 to find the correct one\n    let recoveryParam = 0;\n    for (let i = 0; i <= 1; i++) {\n      const recovered = ecc.recoverPublicKey(messageHash, signature, i, true);\n      if (recovered && Buffer.from(recovered).equals(Buffer.from(publicKey))) {\n        recoveryParam = i;\n        break;\n      }\n    }\n\n    // Append recovery parameter to create 65-byte signature\n    const sigWithRecovery = Buffer.alloc(65);\n    Buffer.from(signature).copy(sigWithRecovery, 0);\n    sigWithRecovery[64] = recoveryParam;\n\n    return sigWithRecovery;\n  }\n\n  /**\n   * Verifies a signature\n   */\n  verifySignature(network: FlareNetwork, message: Buffer, signature: Buffer, publicKey: Buffer): boolean {\n    try {\n      const messageHash = this.sha256(message);\n      return ecc.verify(signature, messageHash, publicKey);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Creates a new signature object\n   */\n  createNewSig(sigHex: string): Signature {\n    const buffer = Buffer.from(sigHex.padStart(130, '0'), 'hex');\n    return new Signature(buffer);\n  }\n\n  /**\n   * Creates an empty signature with embedded address for signature slot identification.\n   * The address is embedded at position 90 (after the first 45 zero bytes).\n   * This allows the signing logic to determine which slot belongs to which address.\n   * @param addressHex The 20-byte address in hex format (40 chars, without 0x prefix)\n   */\n  createEmptySigWithAddress(addressHex: string): Signature {\n    // First 45 bytes (90 hex chars) are zeros, followed by 20-byte address (40 hex chars)\n    const cleanAddr = this.removeHexPrefix(addressHex).toLowerCase();\n    const sigHex = '0'.repeat(90) + cleanAddr.padStart(40, '0');\n    const buffer = Buffer.from(sigHex, 'hex');\n    return new Signature(buffer);\n  }\n\n  /**\n   * Extracts the embedded address from an empty signature.\n   * Returns the address hex string (40 chars) or empty string if not found.\n   */\n  getAddressFromEmptySig(sig: string): string {\n    const cleanSig = this.removeHexPrefix(sig);\n    if (cleanSig.length >= 130) {\n      // Address is at position 90-130 (last 40 hex chars = 20 bytes)\n      return cleanSig.substring(90, 130).toLowerCase();\n    }\n    return '';\n  }\n\n  /**\n   * Computes SHA256 hash\n   */\n  sha256(buf: Uint8Array): Buffer {\n    return createHash('sha256').update(buf).digest();\n  }\n\n  /**\n   * Validates raw transaction format\n   */\n  validateRawTransaction(rawTransaction: string): void {\n    if (!rawTransaction) {\n      throw new InvalidTransactionError('Raw transaction is empty');\n    }\n    if (!this.allHexChars(rawTransaction)) {\n      throw new ParseTransactionError('Raw transaction is not hex string');\n    }\n  }\n\n  /**\n   * Checks if output is TransferableOutput type\n   */\n  isTransferableOutput(output: Output): output is TransferableOutput {\n    return output?._type === TypeSymbols.TransferableOutput;\n  }\n\n  /**\n   * Maps outputs to entry format\n   */\n  mapOutputToEntry(network: FlareNetwork): (Output) => Entry {\n    return (output: Output) => {\n      if (this.isTransferableOutput(output)) {\n        const outputAmount = output.amount();\n        const address = (output.output as TransferOutput)\n          .getOwners()\n          .map((a) => this.addressToString(network.hrp, network.alias, Buffer.from(a)))\n          .sort()\n          .join(ADDRESS_SEPARATOR);\n        return {\n          value: outputAmount.toString(),\n          address,\n        };\n      } else {\n        throw new Error('Invalid output type');\n      }\n    };\n  }\n\n  /**\n   * Removes 0x prefix from hex string\n   */\n  removeHexPrefix(hex: string): string {\n    return hex.startsWith('0x') ? hex.substring(2) : hex;\n  }\n\n  /**\n   * Converts output index to buffer\n   */\n  outputidxNumberToBuffer(outputidx: string): Buffer {\n    return Buffer.from(Number(outputidx).toString(16).padStart(8, '0'), 'hex');\n  }\n\n  /**\n   * Converts output index buffer to number string\n   */\n  outputidxBufferToNumber(outputidx: Buffer): string {\n    return parseInt(outputidx.toString('hex'), 16).toString();\n  }\n\n  // Required by BaseUtils interface but not implemented\n  isValidSignature(signature: string): boolean {\n    throw new NotImplementedError('isValidSignature not implemented');\n  }\n\n  isValidTransactionId(txId: string): boolean {\n    throw new NotImplementedError('isValidTransactionId not implemented');\n  }\n\n  /**\n   * Helper method to convert address components to string\n   */\n  public addressToString = (hrp: string, prefix: string, address: Buffer): string => {\n    // Convert the address bytes to 5-bit words for bech32 encoding\n    const words = bech32.toWords(address);\n    // Create the full bech32 address with format: P-{hrp}1{bech32_encoded_address}\n    return `${prefix}-${bech32.encode(hrp, words)}`;\n  };\n\n  /**\n   * Decodes a base58 string with checksum to a Buffer\n   */\n  public cb58Decode(str: string): Buffer {\n    const decoded = bs58.decode(str);\n    if (!this.validateChecksum(Buffer.from(decoded))) {\n      throw new Error('Invalid checksum');\n    }\n    return Buffer.from(decoded.slice(0, decoded.length - 4));\n  }\n\n  /**\n   * Validates a checksum on a Buffer and returns true if valid, false if not\n   */\n  private validateChecksum(buff: Buffer): boolean {\n    const hashSlice = buff.slice(buff.length - 4);\n    const calculatedHashSlice = createHash('sha256')\n      .update(buff.slice(0, buff.length - 4))\n      .digest()\n      .slice(28);\n    return hashSlice.toString('hex') === calculatedHashSlice.toString('hex');\n  }\n\n  /**\n   * Encodes a Buffer as a base58 string with checksum\n   */\n  public cb58Encode(bytes: Buffer): string {\n    const withChecksum = this.addChecksum(bytes);\n    return bs58.encode(withChecksum);\n  }\n\n  /**\n   * Adds a checksum to a Buffer and returns the concatenated result\n   */\n  private addChecksum(buff: Buffer): Buffer {\n    const hashSlice = createHash('sha256').update(buff).digest().slice(28);\n    return Buffer.concat([buff, hashSlice]);\n  }\n\n  // In utils.ts, add this method to the Utils class:\n\n  /**\n   * Parse an address string into a Buffer\n   * @param address - The address to parse\n   * @returns Buffer containing the parsed address\n   */\n  //TODO: need check and validate this method\n  public parseAddress = (address: string): Buffer => {\n    return this.stringToAddress(address);\n  };\n\n  public stringToAddress = (address: string, hrp?: string): Buffer => {\n    // Handle hex addresses\n    if (address.startsWith('0x')) {\n      return Buffer.from(address.slice(2), 'hex');\n    }\n\n    // Handle raw hex without 0x prefix\n    if (/^[0-9a-fA-F]{40}$/.test(address)) {\n      return Buffer.from(address, 'hex');\n    }\n\n    // Handle Bech32 addresses\n    const parts = address.trim().split('-');\n    if (parts.length < 2) {\n      throw new Error('Error - Valid address should include -');\n    }\n\n    const split = parts[1].lastIndexOf('1');\n    if (split < 0) {\n      throw new Error('Error - Valid bech32 address must include separator (1)');\n    }\n\n    const humanReadablePart = parts[1].slice(0, split);\n    if (humanReadablePart !== 'flare' && humanReadablePart !== 'costwo') {\n      throw new Error('Error - Invalid HRP');\n    }\n\n    return Buffer.from(bech32.fromWords(bech32.decode(parts[1]).words));\n  };\n\n  /**\n   * Check if tx is for the blockchainId\n   *\n   * @param {DeprecatedTx} tx\n   * @param {string} blockchainId\n   * @returns true if tx is for blockchainId\n   */\n  // TODO: remove DeprecatedTx usage\n  isTransactionOf(tx: DeprecatedTx, blockchainId: string): boolean {\n    // FlareJS equivalent - this would need proper CB58 encoding implementation\n    try {\n      const txRecord = tx as unknown as Record<string, unknown>;\n      const unsignedTx = (txRecord.getUnsignedTx as () => Record<string, unknown>)();\n      const transaction = (unsignedTx.getTransaction as () => Record<string, unknown>)();\n      const txBlockchainId = (transaction.getBlockchainID as () => unknown)();\n      return Buffer.from(txBlockchainId as string).toString('hex') === blockchainId;\n    } catch (error) {\n      return false;\n    }\n  }\n\n  flareIdString(value: string): Id {\n    return new Id(Buffer.from(value, 'hex'));\n  }\n\n  /**\n   * Extract credentials from raw transaction bytes.\n   * Signed transactions have credentials appended after the transaction body.\n   * This function handles both checking for credentials and extracting them.\n   *\n   * @param rawBytes - The full raw transaction bytes\n   * @param tx - The parsed transaction (must have toBytes method)\n   * @param vmType - The VM type ('EVM' or 'PVM') to get the correct codec\n   * @returns Object with hasCredentials flag and credentials array\n   */\n  extractCredentialsFromRawBytes(\n    rawBytes: Buffer,\n    tx: { toBytes(codec: unknown): Uint8Array },\n    vmType: 'EVM' | 'PVM' = 'EVM'\n  ): { hasCredentials: boolean; credentials: Credential[] } {\n    try {\n      // Get the size of the transaction without credentials using the default codec\n      const codec = FlareUtils.getManagerForVM(vmType).getDefaultCodec();\n      const txBytes = tx.toBytes(codec);\n      const txSize = txBytes.length;\n\n      // If raw bytes are not longer than tx bytes, there are no credentials\n      if (rawBytes.length <= txSize) {\n        return { hasCredentials: false, credentials: [] };\n      }\n\n      // Extract credential bytes (everything after the transaction)\n      const credentialBytes = rawBytes.slice(txSize);\n\n      // Parse credentials\n      // Format: [num_credentials: 4 bytes] [credentials...]\n      if (credentialBytes.length < 4) {\n        return { hasCredentials: false, credentials: [] };\n      }\n\n      const numCredentials = credentialBytes.readUInt32BE(0);\n\n      // Check if there are credentials in raw bytes (for hasCredentials flag)\n      const hasCredentials = numCredentials > 0;\n\n      if (numCredentials === 0) {\n        return { hasCredentials: false, credentials: [] };\n      }\n\n      const credentials: Credential[] = [];\n      let offset = 4;\n\n      for (let i = 0; i < numCredentials; i++) {\n        if (offset + 8 > credentialBytes.length) {\n          break;\n        }\n\n        // Read type ID (4 bytes) - Type ID 9 = secp256k1 credential\n        const typeId = credentialBytes.readUInt32BE(offset);\n        offset += 4;\n\n        // Validate credential type (9 = secp256k1)\n        if (typeId !== 9) {\n          continue; // Skip unsupported credential types\n        }\n\n        // Read number of signatures (4 bytes)\n        const numSigs = credentialBytes.readUInt32BE(offset);\n        offset += 4;\n\n        // Parse all signatures for this credential\n        const signatures: Signature[] = [];\n        for (let j = 0; j < numSigs; j++) {\n          if (offset + 65 > credentialBytes.length) {\n            break;\n          }\n          // Each signature is 65 bytes (64 bytes signature + 1 byte recovery)\n          const sigBytes = Buffer.from(credentialBytes.slice(offset, offset + 65));\n          signatures.push(new Signature(sigBytes));\n          offset += 65;\n        }\n\n        // Create credential with the parsed signatures\n        if (signatures.length > 0) {\n          credentials.push(new Credential(signatures));\n        }\n      }\n\n      return { hasCredentials, credentials };\n    } catch (e) {\n      // If parsing fails, return no credentials\n      return { hasCredentials: false, credentials: [] };\n    }\n  }\n\n  /**\n   * Parse credentials from raw bytes at a specific offset\n   * This is useful when the standard extraction fails due to serialization differences\n   * @param rawBytes Raw transaction bytes including credentials\n   * @param offset Byte offset where credentials start\n   * @returns Array of parsed credentials\n   */\n  parseCredentialsAtOffset(rawBytes: Buffer, offset: number): Credential[] {\n    try {\n      if (rawBytes.length <= offset + 4) {\n        return [];\n      }\n\n      const credentialBytes = rawBytes.slice(offset);\n      const numCredentials = credentialBytes.readUInt32BE(0);\n\n      if (numCredentials === 0) {\n        return [];\n      }\n\n      const credentials: Credential[] = [];\n      let pos = 4;\n\n      for (let i = 0; i < numCredentials; i++) {\n        if (pos + 8 > credentialBytes.length) {\n          break;\n        }\n\n        // Read type ID (4 bytes) - Type ID 9 = secp256k1 credential\n        const typeId = credentialBytes.readUInt32BE(pos);\n        pos += 4;\n\n        if (typeId !== 9) {\n          continue;\n        }\n\n        // Read number of signatures (4 bytes)\n        const numSigs = credentialBytes.readUInt32BE(pos);\n        pos += 4;\n\n        // Parse all signatures for this credential\n        const signatures: Signature[] = [];\n        for (let j = 0; j < numSigs; j++) {\n          if (pos + 65 > credentialBytes.length) {\n            break;\n          }\n          const sigBytes = Buffer.from(credentialBytes.slice(pos, pos + 65));\n          signatures.push(new Signature(sigBytes));\n          pos += 65;\n        }\n\n        if (signatures.length > 0) {\n          credentials.push(new Credential(signatures));\n        }\n      }\n\n      return credentials;\n    } catch {\n      return [];\n    }\n  }\n\n  /**\n   * FlareJS wrapper to recover signature\n   * @param network\n   * @param message\n   * @param signature\n   * @return recovered public key\n   */\n  recoverySignature(network: FlareNetwork, message: Buffer, signature: Buffer): Buffer {\n    try {\n      // Hash the message first - must match the hash used in signing\n      const messageHash = createHash('sha256').update(message).digest();\n\n      // Extract recovery parameter and signature\n      if (signature.length !== 65) {\n        throw new Error('Invalid signature length - expected 65 bytes (64 bytes signature + 1 byte recovery)');\n      }\n\n      const recoveryParam = signature[64];\n      const sigOnly = signature.slice(0, 64);\n\n      // Recover public key using the provided recovery parameter\n      const recovered = ecc.recoverPublicKey(messageHash, sigOnly, recoveryParam, true);\n      if (!recovered) {\n        throw new Error('Failed to recover public key');\n      }\n\n      return Buffer.from(recovered);\n    } catch (error) {\n      throw new Error(`Failed to recover signature: ${error}`);\n    }\n  }\n}\n\nconst utils = new Utils();\nexport default utils;\n"]}
@@ -0,0 +1,69 @@
1
+ export declare const EXPORT_IN_P: {
2
+ txhash: string;
3
+ unsignedHex: string;
4
+ halfSignedSignature: string;
5
+ halfSigntxHex: string;
6
+ fullSigntxHex: string;
7
+ fullSignedSignature: string;
8
+ outputs: {
9
+ outputID: number;
10
+ amount: string;
11
+ txid: string;
12
+ outputidx: string;
13
+ addresses: string[];
14
+ threshold: number;
15
+ }[];
16
+ amount: string;
17
+ pAddresses: string[];
18
+ privateKeys: string[];
19
+ sourceChainId: string;
20
+ threshold: number;
21
+ fee: string;
22
+ locktime: number;
23
+ INVALID_CHAIN_ID: string;
24
+ VALID_C_CHAIN_ID: string;
25
+ };
26
+ export declare const EXPORT_IN_P_TWO_UTXOS: {
27
+ txhash: string;
28
+ unsignedHex: string;
29
+ halfSigntxHex: string;
30
+ fullSigntxHex: string;
31
+ outputs: {
32
+ outputID: number;
33
+ amount: string;
34
+ txid: string;
35
+ outputidx: string;
36
+ addresses: string[];
37
+ threshold: number;
38
+ }[];
39
+ amount: string;
40
+ pAddresses: string[];
41
+ privateKeys: string[];
42
+ sourceChainId: string;
43
+ threshold: number;
44
+ fee: string;
45
+ locktime: number;
46
+ expectedChange: string;
47
+ };
48
+ export declare const EXPORT_IN_P_NO_CHANGE: {
49
+ txhash: string;
50
+ unsignedHex: string;
51
+ halfSigntxHex: string;
52
+ fullSigntxHex: string;
53
+ outputs: {
54
+ outputID: number;
55
+ amount: string;
56
+ txid: string;
57
+ outputidx: string;
58
+ addresses: string[];
59
+ threshold: number;
60
+ }[];
61
+ amount: string;
62
+ pAddresses: string[];
63
+ privateKeys: string[];
64
+ sourceChainId: string;
65
+ threshold: number;
66
+ fee: string;
67
+ locktime: number;
68
+ };
69
+ //# sourceMappingURL=exportInP.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"exportInP.d.ts","sourceRoot":"","sources":["../../../../test/resources/transactionData/exportInP.ts"],"names":[],"mappings":"AACA,eAAO,MAAM,WAAW;;;;;;;;;;;;;;;;;;;;;;;;CA4CvB,CAAC;AAOF,eAAO,MAAM,qBAAqB;;;;;;;;;;;;;;;;;;;;;CAmDjC,CAAC;AAQF,eAAO,MAAM,qBAAqB;;;;;;;;;;;;;;;;;;;;CAsCjC,CAAC"}