@rosen-bridge/rosen-extractor 12.0.0 → 12.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,19 @@
1
1
  # @rosen-bridge/rosen-extractor
2
2
 
3
+ ## 12.0.2
4
+
5
+ ### Patch Changes
6
+
7
+ - Add store raw data param to Handshake
8
+
9
+ ## 12.0.1
10
+
11
+ ### Patch Changes
12
+
13
+ - Require Handshake lock UTXOs to use covenant type 0
14
+ - Update dependencies
15
+ - @rosen-bridge/tokens@6.0.1
16
+
3
17
  ## 12.0.0
4
18
 
5
19
  ### Major Changes
@@ -6,7 +6,7 @@ import { HandshakeTxOutput } from './types';
6
6
  export declare class HandshakeRosenExtractor extends AbstractRosenDataExtractor<string> {
7
7
  readonly chain = "handshake";
8
8
  protected lockAddressHash: string;
9
- constructor(lockAddress: string, tokens: TokenMap, logger?: AbstractLogger);
9
+ constructor(lockAddress: string, tokens: TokenMap, logger?: AbstractLogger, storeRawData?: boolean);
10
10
  /**
11
11
  * extracts RosenData from given lock transaction in HandshakeTx format
12
12
  * @param serializedTransaction stringified transaction in HandshakeTx format
@@ -1 +1 @@
1
- {"version":3,"file":"handshakeRosenExtractor.d.ts","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/handshakeRosenExtractor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAE/D,OAAO,EAAE,QAAQ,EAAE,MAAM,sBAAsB,CAAC;AAGhD,OAAO,0BAA0B,MAAM,wCAAwC,CAAC;AAChF,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAEnE,OAAO,EAAe,iBAAiB,EAAsB,MAAM,SAAS,CAAC;AAG7E,qBAAa,uBAAwB,SAAQ,0BAA0B,CAAC,MAAM,CAAC;IAC7E,QAAQ,CAAC,KAAK,eAAmB;IACjC,SAAS,CAAC,eAAe,EAAE,MAAM,CAAC;gBAEtB,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,CAAC,EAAE,cAAc;IAK1E;;;OAGG;IACH,WAAW,GAAI,uBAAuB,MAAM,KAAG,SAAS,GAAG,SAAS,CA4FlE;IAEF;;;;OAIG;IACH,sBAAsB,GACpB,KAAK,iBAAiB,EACtB,SAAS,MAAM,KACd,mBAAmB,GAAG,SAAS,CAahC;CACH"}
1
+ {"version":3,"file":"handshakeRosenExtractor.d.ts","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/handshakeRosenExtractor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAE/D,OAAO,EAAE,QAAQ,EAAE,MAAM,sBAAsB,CAAC;AAGhD,OAAO,0BAA0B,MAAM,wCAAwC,CAAC;AAChF,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAEnE,OAAO,EAAe,iBAAiB,EAAsB,MAAM,SAAS,CAAC;AAO7E,qBAAa,uBAAwB,SAAQ,0BAA0B,CAAC,MAAM,CAAC;IAC7E,QAAQ,CAAC,KAAK,eAAmB;IACjC,SAAS,CAAC,eAAe,EAAE,MAAM,CAAC;gBAGhC,WAAW,EAAE,MAAM,EACnB,MAAM,EAAE,QAAQ,EAChB,MAAM,CAAC,EAAE,cAAc,EACvB,YAAY,UAAO;IAMrB;;;OAGG;IACH,WAAW,GAAI,uBAAuB,MAAM,KAAG,SAAS,GAAG,SAAS,CA8FlE;IAEF;;;;OAIG;IACH,sBAAsB,GACpB,KAAK,iBAAiB,EACtB,SAAS,MAAM,KACd,mBAAmB,GAAG,SAAS,CAahC;CACH"}
@@ -2,12 +2,12 @@ import JsonBigInt from '@rosen-bridge/json-bigint';
2
2
  import { parseRosenData } from '../../utils';
3
3
  import AbstractRosenDataExtractor from '../abstract/abstractRosenDataExtractor';
4
4
  import { HANDSHAKE_CHAIN, HANDSHAKE_NATIVE_TOKEN } from '../const';
5
- import { addressToHash, extractDataFromOutputs } from './utils';
5
+ import { addressToHash, extractDataFromOutputs, hasNoneCovenant, } from './utils';
6
6
  export class HandshakeRosenExtractor extends AbstractRosenDataExtractor {
7
7
  chain = HANDSHAKE_CHAIN;
8
8
  lockAddressHash;
9
- constructor(lockAddress, tokens, logger) {
10
- super(lockAddress, tokens, logger);
9
+ constructor(lockAddress, tokens, logger, storeRawData = true) {
10
+ super(lockAddress, tokens, logger, storeRawData);
11
11
  this.lockAddressHash = addressToHash(lockAddress);
12
12
  }
13
13
  /**
@@ -30,7 +30,8 @@ export class HandshakeRosenExtractor extends AbstractRosenDataExtractor {
30
30
  return undefined;
31
31
  }
32
32
  // Find lock output and position
33
- const lockOutputIndex = outputs.findIndex((output) => output.address?.hash === this.lockAddressHash);
33
+ const lockOutputIndex = outputs.findIndex((output) => output.address?.hash === this.lockAddressHash &&
34
+ hasNoneCovenant(output));
34
35
  if (lockOutputIndex === -1) {
35
36
  this.logger.debug(baseError + `: Lock output not found`);
36
37
  return undefined;
@@ -101,4 +102,4 @@ export class HandshakeRosenExtractor extends AbstractRosenDataExtractor {
101
102
  return undefined;
102
103
  };
103
104
  }
104
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handshakeRosenExtractor.js","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/handshakeRosenExtractor.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport JsonBigInt from '@rosen-bridge/json-bigint';\nimport { TokenMap } from '@rosen-bridge/tokens';\n\nimport { parseRosenData } from '../../utils';\nimport AbstractRosenDataExtractor from '../abstract/abstractRosenDataExtractor';\nimport { RosenData, TokenTransformation } from '../abstract/types';\nimport { HANDSHAKE_CHAIN, HANDSHAKE_NATIVE_TOKEN } from '../const';\nimport { HandshakeTx, HandshakeTxOutput, HandshakeRosenData } from './types';\nimport { addressToHash, extractDataFromOutputs } from './utils';\n\nexport class HandshakeRosenExtractor extends AbstractRosenDataExtractor<string> {\n  readonly chain = HANDSHAKE_CHAIN;\n  protected lockAddressHash: string;\n\n  constructor(lockAddress: string, tokens: TokenMap, logger?: AbstractLogger) {\n    super(lockAddress, tokens, logger);\n    this.lockAddressHash = addressToHash(lockAddress);\n  }\n\n  /**\n   * extracts RosenData from given lock transaction in HandshakeTx format\n   * @param serializedTransaction stringified transaction in HandshakeTx format\n   */\n  extractData = (serializedTransaction: string): RosenData | undefined => {\n    let transaction: HandshakeTx;\n    try {\n      transaction = JsonBigInt.parse(serializedTransaction);\n    } catch (e) {\n      throw new Error(\n        `Failed to parse transaction json to HandshakeTx format while extracting rosen data: ${e}`,\n      );\n    }\n\n    const baseError = `No rosen data found for tx [${transaction.id}]`;\n    try {\n      const outputs = transaction.outputs;\n      if (outputs.length < 2) {\n        this.logger.debug(baseError + `: Insufficient number of outputs`);\n        return undefined;\n      }\n\n      // Find lock output and position\n      const lockOutputIndex = outputs.findIndex(\n        (output) => output.address?.hash === this.lockAddressHash,\n      );\n\n      if (lockOutputIndex === -1) {\n        this.logger.debug(baseError + `: Lock output not found`);\n        return undefined;\n      }\n\n      const lockOutput = outputs[lockOutputIndex];\n\n      // Extract data from outputs using utility function\n      const reconstructedData = extractDataFromOutputs(\n        outputs,\n        lockOutputIndex,\n      );\n\n      if (!reconstructedData) {\n        this.logger.debug(baseError + `: No data chunks found`);\n        return undefined;\n      }\n\n      // Parse the reconstructed data\n      let rosenData: HandshakeRosenData | undefined;\n      try {\n        rosenData = parseRosenData(reconstructedData);\n        this.logger.debug(\n          `Successfully extracted Rosen data for [${rosenData.toChain}]`,\n        );\n      } catch (e) {\n        this.logger.debug(\n          baseError + `: Failed to parse reconstructed data: ${e}`,\n        );\n        return undefined;\n      }\n\n      // Find asset transformation using the lock output\n      const assetTransformation = this.getAssetTransformation(\n        lockOutput,\n        rosenData.toChain,\n      );\n\n      if (!assetTransformation) {\n        this.logger.debug(\n          baseError + `: Failed to find rosen asset transformation`,\n        );\n        return undefined;\n      }\n\n      const fromAddress = `box:${transaction.inputs[0].txId}.${transaction.inputs[0].index}`;\n      return {\n        toChain: rosenData.toChain,\n        toAddress: rosenData.toAddress,\n        bridgeFee: rosenData.bridgeFee,\n        networkFee: rosenData.networkFee,\n        fromAddress: fromAddress,\n        sourceChainTokenId: assetTransformation.from,\n        amount: assetTransformation.amount,\n        targetChainTokenId: assetTransformation.to,\n        sourceTxId: transaction.id,\n        rawData: outputs\n          .map((output) => `${output.address?.hash}:${output.value}`)\n          .join(','),\n      };\n    } catch (e) {\n      this.logger.debug(\n        `An error occurred while getting Handshake rosen data: ${e}`,\n      );\n      if (e instanceof Error && e.stack) {\n        this.logger.debug(e.stack);\n      }\n    }\n    return undefined;\n  };\n\n  /**\n   * extracts and builds token transformation from UTXO and tokenMap\n   * @param box transaction output\n   * @param toChain event target chain\n   */\n  getAssetTransformation = (\n    box: HandshakeTxOutput,\n    toChain: string,\n  ): TokenTransformation | undefined => {\n    const wrappedHns = this.tokens.search(HANDSHAKE_CHAIN, {\n      tokenId: HANDSHAKE_NATIVE_TOKEN,\n    });\n\n    if (wrappedHns.length > 0 && Object.hasOwn(wrappedHns[0], toChain)) {\n      return {\n        from: HANDSHAKE_NATIVE_TOKEN,\n        to: this.tokens.getID(wrappedHns[0], toChain),\n        amount: box.value.toString(),\n      };\n    }\n    return undefined;\n  };\n}\n"]}
105
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handshakeRosenExtractor.js","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/handshakeRosenExtractor.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport JsonBigInt from '@rosen-bridge/json-bigint';\nimport { TokenMap } from '@rosen-bridge/tokens';\n\nimport { parseRosenData } from '../../utils';\nimport AbstractRosenDataExtractor from '../abstract/abstractRosenDataExtractor';\nimport { RosenData, TokenTransformation } from '../abstract/types';\nimport { HANDSHAKE_CHAIN, HANDSHAKE_NATIVE_TOKEN } from '../const';\nimport { HandshakeTx, HandshakeTxOutput, HandshakeRosenData } from './types';\nimport {\n  addressToHash,\n  extractDataFromOutputs,\n  hasNoneCovenant,\n} from './utils';\n\nexport class HandshakeRosenExtractor extends AbstractRosenDataExtractor<string> {\n  readonly chain = HANDSHAKE_CHAIN;\n  protected lockAddressHash: string;\n\n  constructor(\n    lockAddress: string,\n    tokens: TokenMap,\n    logger?: AbstractLogger,\n    storeRawData = true,\n  ) {\n    super(lockAddress, tokens, logger, storeRawData);\n    this.lockAddressHash = addressToHash(lockAddress);\n  }\n\n  /**\n   * extracts RosenData from given lock transaction in HandshakeTx format\n   * @param serializedTransaction stringified transaction in HandshakeTx format\n   */\n  extractData = (serializedTransaction: string): RosenData | undefined => {\n    let transaction: HandshakeTx;\n    try {\n      transaction = JsonBigInt.parse(serializedTransaction);\n    } catch (e) {\n      throw new Error(\n        `Failed to parse transaction json to HandshakeTx format while extracting rosen data: ${e}`,\n      );\n    }\n\n    const baseError = `No rosen data found for tx [${transaction.id}]`;\n    try {\n      const outputs = transaction.outputs;\n      if (outputs.length < 2) {\n        this.logger.debug(baseError + `: Insufficient number of outputs`);\n        return undefined;\n      }\n\n      // Find lock output and position\n      const lockOutputIndex = outputs.findIndex(\n        (output) =>\n          output.address?.hash === this.lockAddressHash &&\n          hasNoneCovenant(output),\n      );\n\n      if (lockOutputIndex === -1) {\n        this.logger.debug(baseError + `: Lock output not found`);\n        return undefined;\n      }\n\n      const lockOutput = outputs[lockOutputIndex];\n\n      // Extract data from outputs using utility function\n      const reconstructedData = extractDataFromOutputs(\n        outputs,\n        lockOutputIndex,\n      );\n\n      if (!reconstructedData) {\n        this.logger.debug(baseError + `: No data chunks found`);\n        return undefined;\n      }\n\n      // Parse the reconstructed data\n      let rosenData: HandshakeRosenData | undefined;\n      try {\n        rosenData = parseRosenData(reconstructedData);\n        this.logger.debug(\n          `Successfully extracted Rosen data for [${rosenData.toChain}]`,\n        );\n      } catch (e) {\n        this.logger.debug(\n          baseError + `: Failed to parse reconstructed data: ${e}`,\n        );\n        return undefined;\n      }\n\n      // Find asset transformation using the lock output\n      const assetTransformation = this.getAssetTransformation(\n        lockOutput,\n        rosenData.toChain,\n      );\n\n      if (!assetTransformation) {\n        this.logger.debug(\n          baseError + `: Failed to find rosen asset transformation`,\n        );\n        return undefined;\n      }\n\n      const fromAddress = `box:${transaction.inputs[0].txId}.${transaction.inputs[0].index}`;\n      return {\n        toChain: rosenData.toChain,\n        toAddress: rosenData.toAddress,\n        bridgeFee: rosenData.bridgeFee,\n        networkFee: rosenData.networkFee,\n        fromAddress: fromAddress,\n        sourceChainTokenId: assetTransformation.from,\n        amount: assetTransformation.amount,\n        targetChainTokenId: assetTransformation.to,\n        sourceTxId: transaction.id,\n        rawData: outputs\n          .map((output) => `${output.address?.hash}:${output.value}`)\n          .join(','),\n      };\n    } catch (e) {\n      this.logger.debug(\n        `An error occurred while getting Handshake rosen data: ${e}`,\n      );\n      if (e instanceof Error && e.stack) {\n        this.logger.debug(e.stack);\n      }\n    }\n    return undefined;\n  };\n\n  /**\n   * extracts and builds token transformation from UTXO and tokenMap\n   * @param box transaction output\n   * @param toChain event target chain\n   */\n  getAssetTransformation = (\n    box: HandshakeTxOutput,\n    toChain: string,\n  ): TokenTransformation | undefined => {\n    const wrappedHns = this.tokens.search(HANDSHAKE_CHAIN, {\n      tokenId: HANDSHAKE_NATIVE_TOKEN,\n    });\n\n    if (wrappedHns.length > 0 && Object.hasOwn(wrappedHns[0], toChain)) {\n      return {\n        from: HANDSHAKE_NATIVE_TOKEN,\n        to: this.tokens.getID(wrappedHns[0], toChain),\n        amount: box.value.toString(),\n      };\n    }\n    return undefined;\n  };\n}\n"]}
@@ -6,7 +6,7 @@ import { HandshakeRpcTransaction, HandshakeRpcTxOutput } from './types';
6
6
  export declare class HandshakeRpcRosenExtractor extends AbstractRosenDataExtractor<HandshakeRpcTransaction> {
7
7
  readonly chain = "handshake";
8
8
  protected lockAddressHash: string;
9
- constructor(lockAddress: string, tokens: TokenMap, logger?: AbstractLogger);
9
+ constructor(lockAddress: string, tokens: TokenMap, logger?: AbstractLogger, storeRawData?: boolean);
10
10
  /**
11
11
  * extracts RosenData from given lock transaction in Rpc format
12
12
  * @param transaction the lock transaction in Rpc format
@@ -1 +1 @@
1
- {"version":3,"file":"handshakeRpcRosenExtractor.d.ts","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/handshakeRpcRosenExtractor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAC/D,OAAO,EAAE,QAAQ,EAAE,MAAM,sBAAsB,CAAC;AAGhD,OAAO,0BAA0B,MAAM,wCAAwC,CAAC;AAChF,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAEnE,OAAO,EACL,uBAAuB,EACvB,oBAAoB,EAErB,MAAM,SAAS,CAAC;AAOjB,qBAAa,0BAA2B,SAAQ,0BAA0B,CAAC,uBAAuB,CAAC;IACjG,QAAQ,CAAC,KAAK,eAAmB;IACjC,SAAS,CAAC,eAAe,EAAE,MAAM,CAAC;gBAEtB,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,CAAC,EAAE,cAAc;IAK1E;;;OAGG;IACH,WAAW,GACT,aAAa,uBAAuB,KACnC,SAAS,GAAG,SAAS,CA0FtB;IAEF;;;;OAIG;IACH,sBAAsB,GACpB,KAAK,oBAAoB,EACzB,SAAS,MAAM,KACd,mBAAmB,GAAG,SAAS,CAgBhC;CACH"}
1
+ {"version":3,"file":"handshakeRpcRosenExtractor.d.ts","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/handshakeRpcRosenExtractor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAC/D,OAAO,EAAE,QAAQ,EAAE,MAAM,sBAAsB,CAAC;AAGhD,OAAO,0BAA0B,MAAM,wCAAwC,CAAC;AAChF,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAEnE,OAAO,EACL,uBAAuB,EACvB,oBAAoB,EAErB,MAAM,SAAS,CAAC;AAQjB,qBAAa,0BAA2B,SAAQ,0BAA0B,CAAC,uBAAuB,CAAC;IACjG,QAAQ,CAAC,KAAK,eAAmB;IACjC,SAAS,CAAC,eAAe,EAAE,MAAM,CAAC;gBAGhC,WAAW,EAAE,MAAM,EACnB,MAAM,EAAE,QAAQ,EAChB,MAAM,CAAC,EAAE,cAAc,EACvB,YAAY,UAAO;IAMrB;;;OAGG;IACH,WAAW,GACT,aAAa,uBAAuB,KACnC,SAAS,GAAG,SAAS,CA4FtB;IAEF;;;;OAIG;IACH,sBAAsB,GACpB,KAAK,oBAAoB,EACzB,SAAS,MAAM,KACd,mBAAmB,GAAG,SAAS,CAgBhC;CACH"}
@@ -1,12 +1,12 @@
1
1
  import { parseRosenData } from '../../utils';
2
2
  import AbstractRosenDataExtractor from '../abstract/abstractRosenDataExtractor';
3
3
  import { HANDSHAKE_CHAIN, HANDSHAKE_NATIVE_TOKEN } from '../const';
4
- import { addressToHash, convertHnsToDollarydoos, extractDataFromOutputs, } from './utils';
4
+ import { addressToHash, convertHnsToDollarydoos, extractDataFromOutputs, hasNoneCovenant, } from './utils';
5
5
  export class HandshakeRpcRosenExtractor extends AbstractRosenDataExtractor {
6
6
  chain = HANDSHAKE_CHAIN;
7
7
  lockAddressHash;
8
- constructor(lockAddress, tokens, logger) {
9
- super(lockAddress, tokens, logger);
8
+ constructor(lockAddress, tokens, logger, storeRawData = true) {
9
+ super(lockAddress, tokens, logger, storeRawData);
10
10
  this.lockAddressHash = addressToHash(lockAddress);
11
11
  }
12
12
  /**
@@ -22,7 +22,8 @@ export class HandshakeRpcRosenExtractor extends AbstractRosenDataExtractor {
22
22
  return undefined;
23
23
  }
24
24
  // Find lock output first (need to use original RPC output for value)
25
- const lockOutputIndex = outputs.findIndex((output) => output.address?.hash === this.lockAddressHash);
25
+ const lockOutputIndex = outputs.findIndex((output) => output.address?.hash === this.lockAddressHash &&
26
+ hasNoneCovenant(output));
26
27
  if (lockOutputIndex === -1) {
27
28
  this.logger.debug(baseError + `: Lock output not found`);
28
29
  return undefined;
@@ -103,4 +104,4 @@ export class HandshakeRpcRosenExtractor extends AbstractRosenDataExtractor {
103
104
  return undefined;
104
105
  };
105
106
  }
106
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handshakeRpcRosenExtractor.js","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/handshakeRpcRosenExtractor.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport { TokenMap } from '@rosen-bridge/tokens';\n\nimport { parseRosenData } from '../../utils';\nimport AbstractRosenDataExtractor from '../abstract/abstractRosenDataExtractor';\nimport { RosenData, TokenTransformation } from '../abstract/types';\nimport { HANDSHAKE_CHAIN, HANDSHAKE_NATIVE_TOKEN } from '../const';\nimport {\n  HandshakeRpcTransaction,\n  HandshakeRpcTxOutput,\n  HandshakeRosenData,\n} from './types';\nimport {\n  addressToHash,\n  convertHnsToDollarydoos,\n  extractDataFromOutputs,\n} from './utils';\n\nexport class HandshakeRpcRosenExtractor extends AbstractRosenDataExtractor<HandshakeRpcTransaction> {\n  readonly chain = HANDSHAKE_CHAIN;\n  protected lockAddressHash: string;\n\n  constructor(lockAddress: string, tokens: TokenMap, logger?: AbstractLogger) {\n    super(lockAddress, tokens, logger);\n    this.lockAddressHash = addressToHash(lockAddress);\n  }\n\n  /**\n   * extracts RosenData from given lock transaction in Rpc format\n   * @param transaction the lock transaction in Rpc format\n   */\n  extractData = (\n    transaction: HandshakeRpcTransaction,\n  ): RosenData | undefined => {\n    const baseError = `No rosen data found for tx [${transaction.txid}]`;\n    try {\n      const outputs = transaction.vout;\n      if (outputs.length < 2) {\n        this.logger.debug(baseError + `: Insufficient number of outputs`);\n        return undefined;\n      }\n\n      // Find lock output first (need to use original RPC output for value)\n      const lockOutputIndex = outputs.findIndex(\n        (output) => output.address?.hash === this.lockAddressHash,\n      );\n\n      if (lockOutputIndex === -1) {\n        this.logger.debug(baseError + `: Lock output not found`);\n        return undefined;\n      }\n\n      const lockOutputRpc = outputs[lockOutputIndex];\n\n      // Convert RPC outputs to standard format (HNS to dollarydoos)\n      const convertedOutputs = outputs.map((output) => {\n        const dollarydoos = convertHnsToDollarydoos(output.value);\n        return {\n          value: BigInt(dollarydoos),\n          address: output.address,\n        };\n      });\n\n      // Extract data from outputs using utility function\n      const reconstructedData = extractDataFromOutputs(\n        convertedOutputs,\n        lockOutputIndex,\n      );\n\n      if (!reconstructedData) {\n        this.logger.debug(baseError + `: No data chunks found`);\n        return undefined;\n      }\n\n      // Parse the reconstructed data\n      let rosenData: HandshakeRosenData | undefined;\n      try {\n        rosenData = parseRosenData(reconstructedData);\n        this.logger.debug(\n          `Successfully extracted Rosen data for [${rosenData.toChain}]`,\n        );\n      } catch (e) {\n        this.logger.debug(baseError + `: Failed to parse extracted data: ${e}`);\n        return undefined;\n      }\n\n      // Find asset transformation using the lock output\n      const assetTransformation = this.getAssetTransformation(\n        lockOutputRpc,\n        rosenData.toChain,\n      );\n\n      if (!assetTransformation) {\n        this.logger.debug(\n          baseError + `: Failed to find rosen asset transformation`,\n        );\n        return undefined;\n      }\n\n      const fromAddress = `box:${transaction.vin[0].txid}.${transaction.vin[0].vout}`;\n      return {\n        toChain: rosenData.toChain,\n        toAddress: rosenData.toAddress,\n        bridgeFee: rosenData.bridgeFee,\n        networkFee: rosenData.networkFee,\n        fromAddress: fromAddress,\n        sourceChainTokenId: assetTransformation.from,\n        amount: assetTransformation.amount,\n        targetChainTokenId: assetTransformation.to,\n        sourceTxId: transaction.txid,\n        rawData: outputs\n          .map((output) => `${output.address?.hash}:${output.value}`)\n          .join(','),\n      };\n    } catch (e) {\n      this.logger.debug(\n        `An error occurred while getting Handshake rosen data from Rpc: ${e}`,\n      );\n      if (e instanceof Error && e.stack) {\n        this.logger.debug(e.stack);\n      }\n    }\n    return undefined;\n  };\n\n  /**\n   * extracts and builds token transformation from UTXO and tokenMap\n   * @param box transaction output\n   * @param toChain event target chain\n   */\n  getAssetTransformation = (\n    box: HandshakeRpcTxOutput,\n    toChain: string,\n  ): TokenTransformation | undefined => {\n    const wrappedHns = this.tokens.search(HANDSHAKE_CHAIN, {\n      tokenId: HANDSHAKE_NATIVE_TOKEN,\n    });\n\n    if (wrappedHns.length > 0 && Object.hasOwn(wrappedHns[0], toChain)) {\n      // Convert HNS to dollarydoos\n      const dollarydoos = convertHnsToDollarydoos(box.value);\n\n      return {\n        from: HANDSHAKE_NATIVE_TOKEN,\n        to: this.tokens.getID(wrappedHns[0], toChain),\n        amount: dollarydoos,\n      };\n    }\n    return undefined;\n  };\n}\n"]}
107
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handshakeRpcRosenExtractor.js","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/handshakeRpcRosenExtractor.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport { TokenMap } from '@rosen-bridge/tokens';\n\nimport { parseRosenData } from '../../utils';\nimport AbstractRosenDataExtractor from '../abstract/abstractRosenDataExtractor';\nimport { RosenData, TokenTransformation } from '../abstract/types';\nimport { HANDSHAKE_CHAIN, HANDSHAKE_NATIVE_TOKEN } from '../const';\nimport {\n  HandshakeRpcTransaction,\n  HandshakeRpcTxOutput,\n  HandshakeRosenData,\n} from './types';\nimport {\n  addressToHash,\n  convertHnsToDollarydoos,\n  extractDataFromOutputs,\n  hasNoneCovenant,\n} from './utils';\n\nexport class HandshakeRpcRosenExtractor extends AbstractRosenDataExtractor<HandshakeRpcTransaction> {\n  readonly chain = HANDSHAKE_CHAIN;\n  protected lockAddressHash: string;\n\n  constructor(\n    lockAddress: string,\n    tokens: TokenMap,\n    logger?: AbstractLogger,\n    storeRawData = true,\n  ) {\n    super(lockAddress, tokens, logger, storeRawData);\n    this.lockAddressHash = addressToHash(lockAddress);\n  }\n\n  /**\n   * extracts RosenData from given lock transaction in Rpc format\n   * @param transaction the lock transaction in Rpc format\n   */\n  extractData = (\n    transaction: HandshakeRpcTransaction,\n  ): RosenData | undefined => {\n    const baseError = `No rosen data found for tx [${transaction.txid}]`;\n    try {\n      const outputs = transaction.vout;\n      if (outputs.length < 2) {\n        this.logger.debug(baseError + `: Insufficient number of outputs`);\n        return undefined;\n      }\n\n      // Find lock output first (need to use original RPC output for value)\n      const lockOutputIndex = outputs.findIndex(\n        (output) =>\n          output.address?.hash === this.lockAddressHash &&\n          hasNoneCovenant(output),\n      );\n\n      if (lockOutputIndex === -1) {\n        this.logger.debug(baseError + `: Lock output not found`);\n        return undefined;\n      }\n\n      const lockOutputRpc = outputs[lockOutputIndex];\n\n      // Convert RPC outputs to standard format (HNS to dollarydoos)\n      const convertedOutputs = outputs.map((output) => {\n        const dollarydoos = convertHnsToDollarydoos(output.value);\n        return {\n          value: BigInt(dollarydoos),\n          address: output.address,\n        };\n      });\n\n      // Extract data from outputs using utility function\n      const reconstructedData = extractDataFromOutputs(\n        convertedOutputs,\n        lockOutputIndex,\n      );\n\n      if (!reconstructedData) {\n        this.logger.debug(baseError + `: No data chunks found`);\n        return undefined;\n      }\n\n      // Parse the reconstructed data\n      let rosenData: HandshakeRosenData | undefined;\n      try {\n        rosenData = parseRosenData(reconstructedData);\n        this.logger.debug(\n          `Successfully extracted Rosen data for [${rosenData.toChain}]`,\n        );\n      } catch (e) {\n        this.logger.debug(baseError + `: Failed to parse extracted data: ${e}`);\n        return undefined;\n      }\n\n      // Find asset transformation using the lock output\n      const assetTransformation = this.getAssetTransformation(\n        lockOutputRpc,\n        rosenData.toChain,\n      );\n\n      if (!assetTransformation) {\n        this.logger.debug(\n          baseError + `: Failed to find rosen asset transformation`,\n        );\n        return undefined;\n      }\n\n      const fromAddress = `box:${transaction.vin[0].txid}.${transaction.vin[0].vout}`;\n      return {\n        toChain: rosenData.toChain,\n        toAddress: rosenData.toAddress,\n        bridgeFee: rosenData.bridgeFee,\n        networkFee: rosenData.networkFee,\n        fromAddress: fromAddress,\n        sourceChainTokenId: assetTransformation.from,\n        amount: assetTransformation.amount,\n        targetChainTokenId: assetTransformation.to,\n        sourceTxId: transaction.txid,\n        rawData: outputs\n          .map((output) => `${output.address?.hash}:${output.value}`)\n          .join(','),\n      };\n    } catch (e) {\n      this.logger.debug(\n        `An error occurred while getting Handshake rosen data from Rpc: ${e}`,\n      );\n      if (e instanceof Error && e.stack) {\n        this.logger.debug(e.stack);\n      }\n    }\n    return undefined;\n  };\n\n  /**\n   * extracts and builds token transformation from UTXO and tokenMap\n   * @param box transaction output\n   * @param toChain event target chain\n   */\n  getAssetTransformation = (\n    box: HandshakeRpcTxOutput,\n    toChain: string,\n  ): TokenTransformation | undefined => {\n    const wrappedHns = this.tokens.search(HANDSHAKE_CHAIN, {\n      tokenId: HANDSHAKE_NATIVE_TOKEN,\n    });\n\n    if (wrappedHns.length > 0 && Object.hasOwn(wrappedHns[0], toChain)) {\n      // Convert HNS to dollarydoos\n      const dollarydoos = convertHnsToDollarydoos(box.value);\n\n      return {\n        from: HANDSHAKE_NATIVE_TOKEN,\n        to: this.tokens.getID(wrappedHns[0], toChain),\n        amount: dollarydoos,\n      };\n    }\n    return undefined;\n  };\n}\n"]}
@@ -1,10 +1,16 @@
1
- import { DataExtractionOutput } from './types';
1
+ import { DataExtractionOutput, HandshakeRpcTxOutput, HandshakeTxOutput } from './types';
2
2
  /**
3
3
  * Extracts the hash from a Handshake address using bitcoinjs-lib
4
4
  * @param addr The Handshake address to convert
5
5
  * @returns The address hash as a hex string
6
6
  */
7
7
  export declare const addressToHash: (addr: string) => string;
8
+ /**
9
+ * Checks if a Handshake output uses the NONE covenant
10
+ * @param output The output to validate
11
+ * @returns true when covenant type is 0
12
+ */
13
+ export declare const hasNoneCovenant: (output: HandshakeTxOutput | HandshakeRpcTxOutput) => boolean;
8
14
  /**
9
15
  * Extracts data chunks from transaction outputs using value-based ordering
10
16
  * Data is encoded in P2WPKH address hashes (version 0, values 1000+)
@@ -1 +1 @@
1
- {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/utils.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,oBAAoB,EAAE,MAAM,SAAS,CAAC;AAE/C;;;;GAIG;AACH,eAAO,MAAM,aAAa,GAAI,MAAM,MAAM,KAAG,MAQ5C,CAAC;AAEF;;;;;;GAMG;AACH,eAAO,MAAM,sBAAsB,GACjC,SAAS,oBAAoB,EAAE,EAC/B,kBAAkB,MAAM,KACvB,MAAM,GAAG,SA2CX,CAAC;AAEF;;;;;GAKG;AACH,eAAO,MAAM,uBAAuB,GAAI,OAAO,MAAM,KAAG,MAIvD,CAAC"}
1
+ {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/utils.ts"],"names":[],"mappings":"AAGA,OAAO,EACL,oBAAoB,EACpB,oBAAoB,EACpB,iBAAiB,EAClB,MAAM,SAAS,CAAC;AAEjB;;;;GAIG;AACH,eAAO,MAAM,aAAa,GAAI,MAAM,MAAM,KAAG,MAQ5C,CAAC;AAEF;;;;GAIG;AACH,eAAO,MAAM,eAAe,GAC1B,QAAQ,iBAAiB,GAAG,oBAAoB,KAC/C,OAMF,CAAC;AAEF;;;;;;GAMG;AACH,eAAO,MAAM,sBAAsB,GACjC,SAAS,oBAAoB,EAAE,EAC/B,kBAAkB,MAAM,KACvB,MAAM,GAAG,SA2CX,CAAC;AAEF;;;;;GAKG;AACH,eAAO,MAAM,uBAAuB,GAAI,OAAO,MAAM,KAAG,MAIvD,CAAC"}
@@ -11,6 +11,16 @@ export const addressToHash = (addr) => {
11
11
  // We want just the hash part (skip first 2 bytes: OP_0 and length)
12
12
  return outputScript.subarray(2).toString('hex');
13
13
  };
14
+ /**
15
+ * Checks if a Handshake output uses the NONE covenant
16
+ * @param output The output to validate
17
+ * @returns true when covenant type is 0
18
+ */
19
+ export const hasNoneCovenant = (output) => {
20
+ const covenantType = output.covenant.type;
21
+ return ((typeof covenantType === 'bigint' ? Number(covenantType) : covenantType) ===
22
+ 0);
23
+ };
14
24
  /**
15
25
  * Extracts data chunks from transaction outputs using value-based ordering
16
26
  * Data is encoded in P2WPKH address hashes (version 0, values 1000+)
@@ -64,4 +74,4 @@ export const convertHnsToDollarydoos = (value) => {
64
74
  const part1 = ((parts[1] ?? '') + '0'.repeat(6)).substring(0, 6);
65
75
  return (parts[0] === '0' ? '' : parts[0]) + part1;
66
76
  };
67
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9saWIvZ2V0Um9zZW5EYXRhL2hhbmRzaGFrZS91dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssVUFBVSxNQUFNLGVBQWUsQ0FBQztBQUU1QyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsY0FBYyxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBR2hFOzs7O0dBSUc7QUFDSCxNQUFNLENBQUMsTUFBTSxhQUFhLEdBQUcsQ0FBQyxJQUFZLEVBQVUsRUFBRTtJQUNwRCxNQUFNLFlBQVksR0FBRyxVQUFVLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FDcEQsSUFBSSxFQUNKLGlCQUFpQixDQUNsQixDQUFDO0lBQ0YsNERBQTREO0lBQzVELG1FQUFtRTtJQUNuRSxPQUFPLFlBQVksQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDO0FBQ2xELENBQUMsQ0FBQztBQUVGOzs7Ozs7R0FNRztBQUNILE1BQU0sQ0FBQyxNQUFNLHNCQUFzQixHQUFHLENBQ3BDLE9BQStCLEVBQy9CLGdCQUF3QixFQUNKLEVBQUU7SUFDdEIsTUFBTSxlQUFlLEdBQTJDLEVBQUUsQ0FBQztJQUVuRSxtQ0FBbUM7SUFDbkMsMkVBQTJFO0lBQzNFLEtBQUssSUFBSSxVQUFVLEdBQUcsQ0FBQyxFQUFFLFVBQVUsR0FBRyxDQUFDLEVBQUUsVUFBVSxFQUFFLEVBQUUsQ0FBQztRQUN0RCxLQUFLLElBQUksV0FBVyxHQUFHLENBQUMsRUFBRSxXQUFXLEdBQUcsZ0JBQWdCLEVBQUUsV0FBVyxFQUFFLEVBQUUsQ0FBQztZQUN4RSxNQUFNLE1BQU0sR0FBRyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUM7WUFFcEMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLElBQUksQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLElBQUk7Z0JBQUUsU0FBUztZQUV0RCxNQUFNLEtBQUssR0FBRyxNQUFNLENBQUMsS0FBSyxDQUFDO1lBQzNCLE1BQU0sUUFBUSxHQUFHLE1BQU0sQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDO1lBQ3JDLE1BQU0sT0FBTyxHQUNYLE9BQU8sTUFBTSxDQUFDLE9BQU8sQ0FBQyxPQUFPLEtBQUssUUFBUTtnQkFDeEMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQztnQkFDaEMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDO1lBRTdCLGtEQUFrRDtZQUNsRCxJQUNFLEtBQUssS0FBSyxNQUFNLENBQUMsY0FBYyxDQUFDLEdBQUcsTUFBTSxDQUFDLFVBQVUsQ0FBQztnQkFDckQsT0FBTyxLQUFLLENBQUM7Z0JBQ2IsUUFBUSxDQUFDLE1BQU0sS0FBSyxFQUFFLEVBQ3RCLENBQUM7Z0JBQ0QsZUFBZSxDQUFDLElBQUksQ0FBQztvQkFDbkIsS0FBSyxFQUFFLFVBQVU7b0JBQ2pCLElBQUksRUFBRSxRQUFRO2lCQUNmLENBQUMsQ0FBQztnQkFDSCxNQUFNLENBQUMscUJBQXFCO1lBQzlCLENBQUM7UUFDSCxDQUFDO0lBQ0gsQ0FBQztJQUVELDhDQUE4QztJQUM5QyxNQUFNLGlCQUFpQixHQUNyQixlQUFlLENBQUMsTUFBTSxHQUFHLENBQUM7UUFDeEIsQ0FBQyxDQUFDLGVBQWU7YUFDWixJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsS0FBSyxHQUFHLENBQUMsQ0FBQyxLQUFLLENBQUM7YUFDakMsR0FBRyxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDO2FBQzFCLElBQUksQ0FBQyxFQUFFLENBQUM7UUFDYixDQUFDLENBQUMsU0FBUyxDQUFDO0lBRWhCLE9BQU8saUJBQWlCLENBQUM7QUFDM0IsQ0FBQyxDQUFDO0FBRUY7Ozs7O0dBS0c7QUFDSCxNQUFNLENBQUMsTUFBTSx1QkFBdUIsR0FBRyxDQUFDLEtBQWEsRUFBVSxFQUFFO0lBQy9ELE1BQU0sS0FBSyxHQUFHLEtBQUssQ0FBQyxRQUFRLEVBQUUsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDMUMsTUFBTSxLQUFLLEdBQUcsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsSUFBSSxFQUFFLENBQUMsR0FBRyxHQUFHLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQztJQUNqRSxPQUFPLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxLQUFLLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsR0FBRyxLQUFLLENBQUM7QUFDcEQsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgYml0Y29pbkxpYiBmcm9tICdiaXRjb2luanMtbGliJztcblxuaW1wb3J0IHsgSEFORFNIQUtFX05FVFdPUkssIE1JTl9VVFhPX1ZBTFVFIH0gZnJvbSAnLi9jb25zdGFudHMnO1xuaW1wb3J0IHsgRGF0YUV4dHJhY3Rpb25PdXRwdXQgfSBmcm9tICcuL3R5cGVzJztcblxuLyoqXG4gKiBFeHRyYWN0cyB0aGUgaGFzaCBmcm9tIGEgSGFuZHNoYWtlIGFkZHJlc3MgdXNpbmcgYml0Y29pbmpzLWxpYlxuICogQHBhcmFtIGFkZHIgVGhlIEhhbmRzaGFrZSBhZGRyZXNzIHRvIGNvbnZlcnRcbiAqIEByZXR1cm5zIFRoZSBhZGRyZXNzIGhhc2ggYXMgYSBoZXggc3RyaW5nXG4gKi9cbmV4cG9ydCBjb25zdCBhZGRyZXNzVG9IYXNoID0gKGFkZHI6IHN0cmluZyk6IHN0cmluZyA9PiB7XG4gIGNvbnN0IG91dHB1dFNjcmlwdCA9IGJpdGNvaW5MaWIuYWRkcmVzcy50b091dHB1dFNjcmlwdChcbiAgICBhZGRyLFxuICAgIEhBTkRTSEFLRV9ORVRXT1JLLFxuICApO1xuICAvLyBPdXRwdXQgc2NyaXB0IGZvcm1hdCBmb3Igd2l0bmVzcyB2MDogT1BfMCA8bGVuZ3RoPiA8aGFzaD5cbiAgLy8gV2Ugd2FudCBqdXN0IHRoZSBoYXNoIHBhcnQgKHNraXAgZmlyc3QgMiBieXRlczogT1BfMCBhbmQgbGVuZ3RoKVxuICByZXR1cm4gb3V0cHV0U2NyaXB0LnN1YmFycmF5KDIpLnRvU3RyaW5nKCdoZXgnKTtcbn07XG5cbi8qKlxuICogRXh0cmFjdHMgZGF0YSBjaHVua3MgZnJvbSB0cmFuc2FjdGlvbiBvdXRwdXRzIHVzaW5nIHZhbHVlLWJhc2VkIG9yZGVyaW5nXG4gKiBEYXRhIGlzIGVuY29kZWQgaW4gUDJXUEtIIGFkZHJlc3MgaGFzaGVzICh2ZXJzaW9uIDAsIHZhbHVlcyAxMDAwKylcbiAqIEBwYXJhbSBvdXRwdXRzIC0gYXJyYXkgb2Ygb3V0cHV0cyB3aXRoIHZhbHVlLCBhZGRyZXNzLmhhc2gsIGFkZHJlc3MudmVyc2lvblxuICogQHBhcmFtIGxvY2tBZGRyZXNzSW5kZXggLSBpbmRleCBvZiB0aGUgbG9jayBhZGRyZXNzIG91dHB1dFxuICogQHJldHVybnMgcmVjb25zdHJ1Y3RlZERhdGEgYXMgc3RyaW5nIG9yIHVuZGVmaW5lZFxuICovXG5leHBvcnQgY29uc3QgZXh0cmFjdERhdGFGcm9tT3V0cHV0cyA9IChcbiAgb3V0cHV0czogRGF0YUV4dHJhY3Rpb25PdXRwdXRbXSxcbiAgbG9ja0FkZHJlc3NJbmRleDogbnVtYmVyLFxuKTogc3RyaW5nIHwgdW5kZWZpbmVkID0+IHtcbiAgY29uc3QgZXh0cmFjdGVkQ2h1bmtzOiBBcnJheTx7IGluZGV4OiBudW1iZXI7IGRhdGE6IHN0cmluZyB9PiA9IFtdO1xuXG4gIC8vIEV4dHJhY3QgZGF0YSBjaHVua3Mgc3RlcCBieSBzdGVwXG4gIC8vIFNlYXJjaCBvdXRwdXRzIGJlZm9yZSBsb2NrIGFkZHJlc3MgKGRhdGEgY2h1bmtzIGNvbWUgYmVmb3JlIGxvY2sgb3V0cHV0KVxuICBmb3IgKGxldCBjaHVua0luZGV4ID0gMDsgY2h1bmtJbmRleCA8IDQ7IGNodW5rSW5kZXgrKykge1xuICAgIGZvciAobGV0IG91dHB1dEluZGV4ID0gMDsgb3V0cHV0SW5kZXggPCBsb2NrQWRkcmVzc0luZGV4OyBvdXRwdXRJbmRleCsrKSB7XG4gICAgICBjb25zdCBvdXRwdXQgPSBvdXRwdXRzW291dHB1dEluZGV4XTtcblxuICAgICAgaWYgKCFvdXRwdXQuYWRkcmVzcyB8fCAhb3V0cHV0LmFkZHJlc3MuaGFzaCkgY29udGludWU7XG5cbiAgICAgIGNvbnN0IHZhbHVlID0gb3V0cHV0LnZhbHVlO1xuICAgICAgY29uc3QgYWRkckhhc2ggPSBvdXRwdXQuYWRkcmVzcy5oYXNoO1xuICAgICAgY29uc3QgdmVyc2lvbiA9XG4gICAgICAgIHR5cGVvZiBvdXRwdXQuYWRkcmVzcy52ZXJzaW9uID09PSAnYmlnaW50J1xuICAgICAgICAgID8gTnVtYmVyKG91dHB1dC5hZGRyZXNzLnZlcnNpb24pXG4gICAgICAgICAgOiBvdXRwdXQuYWRkcmVzcy52ZXJzaW9uO1xuXG4gICAgICAvLyBDaGVjayBpZiB0aGlzIG91dHB1dCBtYXRjaGVzIHRoZSBleHBlY3RlZCBjaHVua1xuICAgICAgaWYgKFxuICAgICAgICB2YWx1ZSA9PT0gQmlnSW50KE1JTl9VVFhPX1ZBTFVFKSArIEJpZ0ludChjaHVua0luZGV4KSAmJlxuICAgICAgICB2ZXJzaW9uID09PSAwICYmXG4gICAgICAgIGFkZHJIYXNoLmxlbmd0aCA9PT0gNDBcbiAgICAgICkge1xuICAgICAgICBleHRyYWN0ZWRDaHVua3MucHVzaCh7XG4gICAgICAgICAgaW5kZXg6IGNodW5rSW5kZXgsXG4gICAgICAgICAgZGF0YTogYWRkckhhc2gsXG4gICAgICAgIH0pO1xuICAgICAgICBicmVhazsgLy8gTW92ZSB0byBuZXh0IGNodW5rXG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgLy8gUmVjb25zdHJ1Y3QgZGF0YSBieSBzb3J0aW5nIGNodW5rcyBieSBpbmRleFxuICBjb25zdCByZWNvbnN0cnVjdGVkRGF0YSA9XG4gICAgZXh0cmFjdGVkQ2h1bmtzLmxlbmd0aCA+IDBcbiAgICAgID8gZXh0cmFjdGVkQ2h1bmtzXG4gICAgICAgICAgLnNvcnQoKGEsIGIpID0+IGEuaW5kZXggLSBiLmluZGV4KVxuICAgICAgICAgIC5tYXAoKGNodW5rKSA9PiBjaHVuay5kYXRhKVxuICAgICAgICAgIC5qb2luKCcnKVxuICAgICAgOiB1bmRlZmluZWQ7XG5cbiAgcmV0dXJuIHJlY29uc3RydWN0ZWREYXRhO1xufTtcblxuLyoqXG4gKiBDb252ZXJ0cyBITlMgdmFsdWUgdG8gZG9sbGFyeWRvb3MgKHNhdG9zaGktZXF1aXZhbGVudCkgdXNpbmcgc3RyaW5nLWJhc2VkIGFyaXRobWV0aWNcbiAqIFRoaXMgYXZvaWRzIGZsb2F0aW5nLXBvaW50IHByZWNpc2lvbiBlcnJvcnMgd2hlbiBjb252ZXJ0aW5nIGZyb20gSE5TIGRlY2ltYWwgZm9ybWF0XG4gKiBAcGFyYW0gdmFsdWUgVGhlIEhOUyB2YWx1ZSBhcyBhIG51bWJlciAoZS5nLiwgMC4wMDEsIDEuNSlcbiAqIEByZXR1cm5zIFRoZSB2YWx1ZSBpbiBkb2xsYXJ5ZG9vcyBhcyBhIHN0cmluZyAoZS5nLiwgXCIxMDAwXCIsIFwiMTUwMDAwMFwiKVxuICovXG5leHBvcnQgY29uc3QgY29udmVydEhuc1RvRG9sbGFyeWRvb3MgPSAodmFsdWU6IG51bWJlcik6IHN0cmluZyA9PiB7XG4gIGNvbnN0IHBhcnRzID0gdmFsdWUudG9TdHJpbmcoKS5zcGxpdCgnLicpO1xuICBjb25zdCBwYXJ0MSA9ICgocGFydHNbMV0gPz8gJycpICsgJzAnLnJlcGVhdCg2KSkuc3Vic3RyaW5nKDAsIDYpO1xuICByZXR1cm4gKHBhcnRzWzBdID09PSAnMCcgPyAnJyA6IHBhcnRzWzBdKSArIHBhcnQxO1xufTtcbiJdfQ==
77
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../lib/getRosenData/handshake/utils.ts"],"names":[],"mappings":"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","sourcesContent":["import * as bitcoinLib from 'bitcoinjs-lib';\n\nimport { HANDSHAKE_NETWORK, MIN_UTXO_VALUE } from './constants';\nimport {\n  DataExtractionOutput,\n  HandshakeRpcTxOutput,\n  HandshakeTxOutput,\n} from './types';\n\n/**\n * Extracts the hash from a Handshake address using bitcoinjs-lib\n * @param addr The Handshake address to convert\n * @returns The address hash as a hex string\n */\nexport const addressToHash = (addr: string): string => {\n  const outputScript = bitcoinLib.address.toOutputScript(\n    addr,\n    HANDSHAKE_NETWORK,\n  );\n  // Output script format for witness v0: OP_0 <length> <hash>\n  // We want just the hash part (skip first 2 bytes: OP_0 and length)\n  return outputScript.subarray(2).toString('hex');\n};\n\n/**\n * Checks if a Handshake output uses the NONE covenant\n * @param output The output to validate\n * @returns true when covenant type is 0\n */\nexport const hasNoneCovenant = (\n  output: HandshakeTxOutput | HandshakeRpcTxOutput,\n): boolean => {\n  const covenantType = output.covenant.type as number | bigint;\n  return (\n    (typeof covenantType === 'bigint' ? Number(covenantType) : covenantType) ===\n    0\n  );\n};\n\n/**\n * Extracts data chunks from transaction outputs using value-based ordering\n * Data is encoded in P2WPKH address hashes (version 0, values 1000+)\n * @param outputs - array of outputs with value, address.hash, address.version\n * @param lockAddressIndex - index of the lock address output\n * @returns reconstructedData as string or undefined\n */\nexport const extractDataFromOutputs = (\n  outputs: DataExtractionOutput[],\n  lockAddressIndex: number,\n): string | undefined => {\n  const extractedChunks: Array<{ index: number; data: string }> = [];\n\n  // Extract data chunks step by step\n  // Search outputs before lock address (data chunks come before lock output)\n  for (let chunkIndex = 0; chunkIndex < 4; chunkIndex++) {\n    for (let outputIndex = 0; outputIndex < lockAddressIndex; outputIndex++) {\n      const output = outputs[outputIndex];\n\n      if (!output.address || !output.address.hash) continue;\n\n      const value = output.value;\n      const addrHash = output.address.hash;\n      const version =\n        typeof output.address.version === 'bigint'\n          ? Number(output.address.version)\n          : output.address.version;\n\n      // Check if this output matches the expected chunk\n      if (\n        value === BigInt(MIN_UTXO_VALUE) + BigInt(chunkIndex) &&\n        version === 0 &&\n        addrHash.length === 40\n      ) {\n        extractedChunks.push({\n          index: chunkIndex,\n          data: addrHash,\n        });\n        break; // Move to next chunk\n      }\n    }\n  }\n\n  // Reconstruct data by sorting chunks by index\n  const reconstructedData =\n    extractedChunks.length > 0\n      ? extractedChunks\n          .sort((a, b) => a.index - b.index)\n          .map((chunk) => chunk.data)\n          .join('')\n      : undefined;\n\n  return reconstructedData;\n};\n\n/**\n * Converts HNS value to dollarydoos (satoshi-equivalent) using string-based arithmetic\n * This avoids floating-point precision errors when converting from HNS decimal format\n * @param value The HNS value as a number (e.g., 0.001, 1.5)\n * @returns The value in dollarydoos as a string (e.g., \"1000\", \"1500000\")\n */\nexport const convertHnsToDollarydoos = (value: number): string => {\n  const parts = value.toString().split('.');\n  const part1 = ((parts[1] ?? '') + '0'.repeat(6)).substring(0, 6);\n  return (parts[0] === '0' ? '' : parts[0]) + part1;\n};\n"]}