PHP WebShell
Текущая директория: /opt/BitGoJS/modules/sdk-coin-avaxp/dist/src/lib
Просмотр файла: atomicTransactionBuilder.js
"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.AtomicTransactionBuilder = void 0;
const avalanche_1 = require("avalanche");
const utils_1 = __importDefault(require("./utils"));
const deprecatedTransactionBuilder_1 = require("./deprecatedTransactionBuilder");
const platformvm_1 = require("avalanche/dist/apis/platformvm");
const sdk_core_1 = require("@bitgo/sdk-core");
const iface_1 = require("./iface");
/**
* Cross-chain transactions (export and import) are atomic operations.
*/
class AtomicTransactionBuilder extends deprecatedTransactionBuilder_1.DeprecatedTransactionBuilder {
constructor(_coinConfig) {
super(_coinConfig);
this.transaction._fee.fee = this.fixedFee;
}
/**
* The internal chain is the one set for the coin in coinConfig.network. The external chain is the other chain involved.
* The external chain id is the source on import and the destination on export.
*
* @param {string} chainId - id of the external chain
*/
externalChainId(chainId) {
const newTargetChainId = typeof chainId === 'string' ? utils_1.default.cb58Decode(chainId) : avalanche_1.Buffer.from(chainId);
this.validateChainId(newTargetChainId);
this._externalChainId = newTargetChainId;
return this;
}
/**
* Fee is fix for AVM atomic tx.
*
* @returns network.txFee
* @protected
*/
get fixedFee() {
return this.transaction._network.txFee;
}
// region utxo engine
/**
* Threshold must be 2 and since output always get reordered we want to make sure we can always add signatures in the correct location
* To find the correct location for the signature, we use the output's addresses to create the signatureIdx in the order that we desire
* 0: user key, 1: hsm key, 2: recovery key
* @protected
*/
createInputOutput(amount) {
const inputs = [];
const outputs = [];
// amount spent so far
let currentTotal = new avalanche_1.BN(0);
// delegating and validating have no fees
const totalTarget = amount.clone();
const credentials = [];
/*
A = user key
B = hsm key
C = backup key
bitgoAddresses = bitgo addresses [ A, B, C ]
utxo.addresses = IMS addresses [ B, C, A ]
utxo.addressesIndex = [ 2, 0, 1 ]
we pick 0, 1 for non-recovery
we pick 1, 2 for recovery
*/
this.transaction._utxos.forEach((utxo) => {
// in WP, output.addressesIndex is empty, so fill it
if (!utxo.addressesIndex || utxo.addressesIndex.length === 0) {
const utxoAddresses = utxo.addresses.map((a) => utils_1.default.parseAddress(a));
utxo.addressesIndex = this.transaction._fromAddresses.map((a) => utxoAddresses.findIndex((u) => a.equals(u)));
}
// in OVC, output.addressesIndex is defined correctly from the previous iteration
});
// validate the utxos
this.transaction._utxos.forEach((utxo) => {
if (!utxo) {
throw new sdk_core_1.BuildTransactionError('Utxo is undefined');
}
// addressesIndex should never have a mismatch
if (utxo.addressesIndex?.includes(-1)) {
throw new sdk_core_1.BuildTransactionError('Addresses are inconsistent: ' + utxo.txid);
}
if (utxo.threshold !== this.transaction._threshold) {
throw new sdk_core_1.BuildTransactionError('Threshold is inconsistent');
}
});
this.transaction._utxos.forEach((utxo, i) => {
if (utxo.outputID === iface_1.SECP256K1_Transfer_Output) {
const txidBuf = utils_1.default.cb58Decode(utxo.txid);
const amt = new avalanche_1.BN(utxo.amount);
const outputidx = utils_1.default.outputidxNumberToBuffer(utxo.outputidx);
const addressesIndex = utxo.addressesIndex ?? [];
// either user (0) or recovery (2)
const firstIndex = this.recoverSigner ? 2 : 0;
const bitgoIndex = 1;
currentTotal = currentTotal.add(amt);
const secpTransferInput = new platformvm_1.SECPTransferInput(amt);
// if user/backup > bitgo
if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {
secpTransferInput.addSignatureIdx(addressesIndex[bitgoIndex], this.transaction._fromAddresses[bitgoIndex]);
secpTransferInput.addSignatureIdx(addressesIndex[firstIndex], this.transaction._fromAddresses[firstIndex]);
credentials.push((0, platformvm_1.SelectCredentialClass)(secpTransferInput.getCredentialID(), // 9
['', this.transaction._fromAddresses[firstIndex].toString('hex')].map(utils_1.default.createSig)));
}
else {
secpTransferInput.addSignatureIdx(addressesIndex[firstIndex], this.transaction._fromAddresses[firstIndex]);
secpTransferInput.addSignatureIdx(addressesIndex[bitgoIndex], this.transaction._fromAddresses[bitgoIndex]);
credentials.push((0, platformvm_1.SelectCredentialClass)(secpTransferInput.getCredentialID(), [this.transaction._fromAddresses[firstIndex].toString('hex'), ''].map(utils_1.default.createSig)));
}
const input = new platformvm_1.TransferableInput(txidBuf, outputidx, this.transaction._assetId, secpTransferInput);
inputs.push(input);
}
});
if (currentTotal.lt(totalTarget)) {
throw new sdk_core_1.BuildTransactionError(`Utxo outputs get ${currentTotal.toString()} and ${totalTarget.toString()} is required`);
}
else if (currentTotal.gt(totalTarget)) {
outputs.push(new platformvm_1.TransferableOutput(this.transaction._assetId, new platformvm_1.SECPTransferOutput(currentTotal.sub(totalTarget), this.transaction._fromAddresses, this.transaction._locktime, this.transaction._threshold)));
}
return {
inputs,
outputs,
credentials,
};
}
}
exports.AtomicTransactionBuilder = AtomicTransactionBuilder;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"atomicTransactionBuilder.js","sourceRoot":"","sources":["../../../src/lib/atomicTransactionBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseCoin as CoinConfig } from '@bitgo/statics';\nimport { BN, Buffer as BufferAvax } from 'avalanche';\nimport utils from './utils';\nimport { DeprecatedTransactionBuilder } from './deprecatedTransactionBuilder';\nimport {\n  SECPTransferInput,\n  SECPTransferOutput,\n  SelectCredentialClass,\n  TransferableInput,\n  TransferableOutput,\n} from 'avalanche/dist/apis/platformvm';\nimport { Credential } from 'avalanche/dist/common';\nimport { BuildTransactionError } from '@bitgo/sdk-core';\nimport { SECP256K1_Transfer_Output } from './iface';\n\n/**\n * Cross-chain transactions (export and import) are atomic operations.\n */\nexport abstract class AtomicTransactionBuilder extends DeprecatedTransactionBuilder {\n  protected _externalChainId: BufferAvax;\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this.transaction._fee.fee = this.fixedFee;\n  }\n\n  /**\n   * The internal chain is the one set for the coin in coinConfig.network. The external chain is the other chain involved.\n   * The external chain id is the source on import and the destination on export.\n   *\n   * @param {string} chainId - id of the external chain\n   */\n  externalChainId(chainId: string | Buffer): this {\n    const newTargetChainId = typeof chainId === 'string' ? utils.cb58Decode(chainId) : BufferAvax.from(chainId);\n    this.validateChainId(newTargetChainId);\n    this._externalChainId = newTargetChainId;\n    return this;\n  }\n\n  /**\n   * Fee is fix for AVM atomic tx.\n   *\n   * @returns network.txFee\n   * @protected\n   */\n  protected get fixedFee(): string {\n    return this.transaction._network.txFee;\n  }\n\n  // region utxo engine\n  /**\n   * Threshold must be 2 and since output always get reordered we want to make sure we can always add signatures in the correct location\n   * To find the correct location for the signature, we use the output's addresses to create the signatureIdx in the order that we desire\n   * 0: user key, 1: hsm key, 2: recovery key\n   * @protected\n   */\n  protected createInputOutput(amount: BN): {\n    inputs: TransferableInput[];\n    outputs: TransferableOutput[];\n    credentials: Credential[];\n  } {\n    const inputs: TransferableInput[] = [];\n    const outputs: TransferableOutput[] = [];\n\n    // amount spent so far\n    let currentTotal: BN = new BN(0);\n\n    // delegating and validating have no fees\n    const totalTarget = amount.clone();\n\n    const credentials: Credential[] = [];\n\n    /*\n    A = user key\n    B = hsm key\n    C = backup key\n    bitgoAddresses = bitgo addresses [ A, B, C ]\n    utxo.addresses = IMS addresses [ B, C, A ]\n    utxo.addressesIndex = [ 2, 0, 1 ]\n    we pick 0, 1 for non-recovery\n    we pick 1, 2 for recovery\n    */\n    this.transaction._utxos.forEach((utxo) => {\n      // in WP, output.addressesIndex is empty, so fill it\n      if (!utxo.addressesIndex || utxo.addressesIndex.length === 0) {\n        const utxoAddresses: BufferAvax[] = utxo.addresses.map((a) => utils.parseAddress(a));\n        utxo.addressesIndex = this.transaction._fromAddresses.map((a) => utxoAddresses.findIndex((u) => a.equals(u)));\n      }\n      // in OVC, output.addressesIndex is defined correctly from the previous iteration\n    });\n\n    // validate the utxos\n    this.transaction._utxos.forEach((utxo) => {\n      if (!utxo) {\n        throw new BuildTransactionError('Utxo is undefined');\n      }\n      // addressesIndex should never have a mismatch\n      if (utxo.addressesIndex?.includes(-1)) {\n        throw new BuildTransactionError('Addresses are inconsistent: ' + utxo.txid);\n      }\n      if (utxo.threshold !== this.transaction._threshold) {\n        throw new BuildTransactionError('Threshold is inconsistent');\n      }\n    });\n\n    this.transaction._utxos.forEach((utxo, i) => {\n      if (utxo.outputID === SECP256K1_Transfer_Output) {\n        const txidBuf = utils.cb58Decode(utxo.txid);\n        const amt: BN = new BN(utxo.amount);\n        const outputidx = utils.outputidxNumberToBuffer(utxo.outputidx);\n        const addressesIndex = utxo.addressesIndex ?? [];\n\n        // either user (0) or recovery (2)\n        const firstIndex = this.recoverSigner ? 2 : 0;\n        const bitgoIndex = 1;\n        currentTotal = currentTotal.add(amt);\n\n        const secpTransferInput = new SECPTransferInput(amt);\n\n        // if user/backup > bitgo\n        if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {\n          secpTransferInput.addSignatureIdx(addressesIndex[bitgoIndex], this.transaction._fromAddresses[bitgoIndex]);\n          secpTransferInput.addSignatureIdx(addressesIndex[firstIndex], this.transaction._fromAddresses[firstIndex]);\n          credentials.push(\n            SelectCredentialClass(\n              secpTransferInput.getCredentialID(), // 9\n              ['', this.transaction._fromAddresses[firstIndex].toString('hex')].map(utils.createSig)\n            )\n          );\n        } else {\n          secpTransferInput.addSignatureIdx(addressesIndex[firstIndex], this.transaction._fromAddresses[firstIndex]);\n          secpTransferInput.addSignatureIdx(addressesIndex[bitgoIndex], this.transaction._fromAddresses[bitgoIndex]);\n          credentials.push(\n            SelectCredentialClass(\n              secpTransferInput.getCredentialID(),\n              [this.transaction._fromAddresses[firstIndex].toString('hex'), ''].map(utils.createSig)\n            )\n          );\n        }\n\n        const input: TransferableInput = new TransferableInput(\n          txidBuf,\n          outputidx,\n          this.transaction._assetId,\n          secpTransferInput\n        );\n        inputs.push(input);\n      }\n    });\n\n    if (currentTotal.lt(totalTarget)) {\n      throw new BuildTransactionError(\n        `Utxo outputs get ${currentTotal.toString()} and ${totalTarget.toString()} is required`\n      );\n    } else if (currentTotal.gt(totalTarget)) {\n      outputs.push(\n        new TransferableOutput(\n          this.transaction._assetId,\n          new SECPTransferOutput(\n            currentTotal.sub(totalTarget),\n            this.transaction._fromAddresses,\n            this.transaction._locktime,\n            this.transaction._threshold\n          )\n        )\n      );\n    }\n    return {\n      inputs,\n      outputs,\n      credentials,\n    };\n  }\n\n  // endregion\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!