PHP WebShell

Текущая директория: /opt/BitGoJS/modules/sdk-core/dist/src/bitgo/pendingApproval

Просмотр файла: pendingApproval.js

"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    var desc = Object.getOwnPropertyDescriptor(m, k);
    if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
      desc = { enumerable: true, get: function() { return m[k]; } };
    }
    Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
    Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
    o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
    var ownKeys = function(o) {
        ownKeys = Object.getOwnPropertyNames || function (o) {
            var ar = [];
            for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
            return ar;
        };
        return ownKeys(o);
    };
    return function (mod) {
        if (mod && mod.__esModule) return mod;
        var result = {};
        if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
        __setModuleDefault(result, mod);
        return result;
    };
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.PendingApproval = void 0;
/**
 * @prettier
 */
const _ = __importStar(require("lodash"));
const utxolib = __importStar(require("@bitgo/utxo-lib"));
const pendingApproval_1 = require("../pendingApproval");
const utils_1 = require("../utils");
const BuildParams_1 = require("../wallet/BuildParams");
const eddsa_1 = __importDefault(require("../utils/tss/eddsa"));
const ecdsa_1 = require("../utils/tss/ecdsa");
const common_1 = require("../tss/common");
const assert_1 = __importDefault(require("assert"));
class PendingApproval {
    constructor(bitgo, baseCoin, pendingApprovalData, wallet) {
        this.bitgo = bitgo;
        this.baseCoin = baseCoin;
        this.wallet = wallet;
        if (this.baseCoin.supportsTss()) {
            if (this.baseCoin.getMPCAlgorithm() === 'ecdsa') {
                if (this.wallet?.multisigTypeVersion() === 'MPCv2') {
                    this.tssUtils = new ecdsa_1.EcdsaMPCv2Utils(this.bitgo, this.baseCoin, wallet);
                }
                else {
                    this.tssUtils = new ecdsa_1.EcdsaUtils(this.bitgo, this.baseCoin, wallet);
                }
            }
            else {
                this.tssUtils = new eddsa_1.default(this.bitgo, this.baseCoin, wallet);
            }
        }
        this._pendingApproval = pendingApprovalData;
    }
    /**
     * Get the id for this PendingApproval
     */
    id() {
        return this._pendingApproval.id;
    }
    toJSON() {
        return this._pendingApproval;
    }
    /**
     * Get the owner type (wallet or enterprise)
     * Pending approvals can be approved or modified by different scopes (depending on how they were created)
     * If a pending approval is owned by a wallet, then it can be approved by administrators of the wallet
     * If a pending approval is owned by an enterprise, then it can be approved by administrators of the enterprise
     */
    ownerType() {
        if (this._pendingApproval.wallet) {
            return pendingApproval_1.OwnerType.WALLET;
        }
        else if (this._pendingApproval.enterprise) {
            return pendingApproval_1.OwnerType.ENTERPRISE;
        }
        else {
            throw new Error('unexpected pending approval owner: neither wallet nor enterprise was present');
        }
    }
    /**
     * Get the id of the wallet which is associated with this PendingApproval
     */
    walletId() {
        return this._pendingApproval.wallet;
    }
    /**
     * Get the enterprise ID that is associated with this PendingApproval
     */
    enterpriseId() {
        return this._pendingApproval.enterprise;
    }
    /**
     * Get the state of this PendingApproval
     */
    state() {
        return this._pendingApproval.state;
    }
    /**
     * Get the id of the user that performed the action resulting in this PendingApproval
     */
    creator() {
        return this._pendingApproval.creator;
    }
    /**
     * Get the type of the pending approval (what it approves)
     */
    type() {
        if (!this._pendingApproval.info) {
            throw new Error('pending approval info is not available');
        }
        return this._pendingApproval.info.type;
    }
    /**
     * Get information about this PendingApproval
     */
    info() {
        return this._pendingApproval.info;
    }
    /**
     * Get the number of approvals that are required for this PendingApproval to be approved.
     * Defaults to 1 if approvalsRequired doesn't exist on the object
     */
    approvalsRequired() {
        return this._pendingApproval.approvalsRequired || 1;
    }
    /**
     * Generate a url for this PendingApproval for making requests to the server.
     * @param extra
     */
    url(extra = '') {
        return this.baseCoin.url('/pendingapprovals/' + this.id() + extra);
    }
    /**
     * Refetches this PendingApproval from the server and returns it.
     *
     * Note that this mutates the PendingApproval object in place.
     * @param params
     */
    async get(params = {}) {
        this._pendingApproval = await this.bitgo.get(this.url()).result();
        return this;
    }
    /**
     * Sets this PendingApproval to an approved state
     */
    async approve(params = {}) {
        params.previewPendingTxs = true;
        params.pendingApprovalId = this.id();
        const canRecreateTransaction = this.canRecreateTransaction(params);
        const reqId = new utils_1.RequestTracer();
        this.bitgo.setRequestTracer(reqId);
        await this.populateWallet();
        try {
            const transaction = await this.preApprove(params, reqId);
            const approvalParams = { state: 'approved', otp: params.otp };
            if (transaction) {
                // if the transaction already has a half signed property, we take that directly
                approvalParams.halfSigned = transaction.halfSigned || transaction;
            }
            const response = await this.bitgo.put(this.url()).send(approvalParams).result();
            // if the response comes with an error, means that the transaction triggered another condition
            if (response.hasOwnProperty('error') && response.hasOwnProperty('pendingApproval')) {
                return response;
            }
            this._pendingApproval = response;
            await this.postApprove(params, reqId);
            return this._pendingApproval;
        }
        catch (e) {
            if (!canRecreateTransaction &&
                (e.message.indexOf('could not find unspent output for input') !== -1 ||
                    e.message.indexOf('transaction conflicts with an existing transaction in the send queue') !== -1)) {
                throw new Error('unspents expired, wallet passphrase or xprv required to recreate transaction');
            }
            throw e;
        }
    }
    /**
     * Sets this PendingApproval to a rejected state
     * @param params
     */
    async reject(params = {}) {
        return await this.bitgo.put(this.url()).send({ state: 'rejected' }).result();
    }
    /**
     * Alias for PendingApproval.reject()
     *
     * @deprecated
     * @param params
     */
    async cancel(params = {}) {
        return await this.reject(params);
    }
    /**
     * Recreate and sign TSS transaction
     * @param {ApproveOptions} params needed to get txs and use the walletPassphrase to tss sign
     * @param {RequestTracer} reqId id tracer.
     */
    async recreateAndSignTSSTransaction(params, reqId) {
        const { walletPassphrase } = params;
        const txRequestId = this._pendingApproval.txRequestId;
        if (!this.wallet) {
            throw new Error('Wallet not found');
        }
        if (!walletPassphrase) {
            throw new Error('walletPassphrase not found');
        }
        if (!txRequestId) {
            throw new Error('txRequestId not found');
        }
        const decryptedPrv = await this.wallet.getPrv({ walletPassphrase });
        const txRequest = await this.tssUtils.recreateTxRequest(txRequestId, decryptedPrv, reqId);
        if (txRequest.apiVersion === 'lite') {
            if (!txRequest.unsignedTxs || txRequest.unsignedTxs.length === 0) {
                throw new Error('Unexpected error, no transactions found in txRequest.');
            }
            return {
                txHex: txRequest.unsignedTxs[0].serializedTxHex,
            };
        }
        else {
            if (!txRequest.transactions || txRequest.transactions.length === 0) {
                throw new Error('Unexpected error, no transactions found in txRequest.');
            }
            return {
                txHex: txRequest.transactions[0].unsignedTx.serializedTxHex,
            };
        }
    }
    /**
     * Recreate a transaction for a pending approval to respond to updated network conditions
     * @param params
     * @param reqId
     */
    async recreateAndSignTransaction(params = {}, reqId) {
        // this method only makes sense with existing transaction requests
        const transactionRequest = this.info().transactionRequest;
        if (_.isUndefined(transactionRequest)) {
            throw new Error('cannot recreate transaction without transaction request');
        }
        if (_.isUndefined(this.wallet)) {
            throw new Error('cannot recreate transaction without wallet');
        }
        const originalPrebuild = transactionRequest.coinSpecific[this.baseCoin.type];
        const recipients = transactionRequest.recipients;
        let prebuildParams = _.extend({}, params, { recipients: recipients }, transactionRequest.buildParams);
        if (!_.isUndefined(originalPrebuild.hopTransaction)) {
            prebuildParams.hop = true;
        }
        const reqTracer = reqId || new utils_1.RequestTracer();
        if (transactionRequest.buildParams && transactionRequest.buildParams.type === 'consolidate') {
            // consolidate tag is in the build params - this is a consolidation transaction, so
            // it needs to be rebuilt using the special consolidation build route
            this.bitgo.setRequestTracer(reqTracer);
            prebuildParams.prebuildTx = await this.bitgo
                .post(this.wallet.url(`/consolidateUnspents`))
                .send(BuildParams_1.BuildParams.encode(prebuildParams))
                .result();
            delete prebuildParams.recipients;
        }
        prebuildParams = _.extend({}, prebuildParams, { reqId: reqId });
        const signedTransaction = await this.wallet.prebuildAndSignTransaction(prebuildParams);
        // compare PAYGo fees
        const originalParsedTransaction = (await this.baseCoin.parseTransaction({
            txParams: prebuildParams,
            wallet: this.wallet,
            txPrebuild: originalPrebuild,
        }));
        const recreatedParsedTransaction = (await this.baseCoin.parseTransaction({
            txParams: prebuildParams,
            wallet: this.wallet,
            txPrebuild: signedTransaction,
        }));
        if (_.isUndefined(recreatedParsedTransaction.implicitExternalSpendAmount)) {
            return signedTransaction;
        }
        if (typeof recreatedParsedTransaction.implicitExternalSpendAmount !== 'bigint' &&
            !_.isFinite(recreatedParsedTransaction.implicitExternalSpendAmount)) {
            throw new Error('implicit external spend amount could not be determined');
        }
        if (!_.isUndefined(originalParsedTransaction.implicitExternalSpendAmount) &&
            recreatedParsedTransaction.implicitExternalSpendAmount > originalParsedTransaction.implicitExternalSpendAmount) {
            throw new Error('recreated transaction is using a higher pay-as-you-go-fee');
        }
        return signedTransaction;
    }
    /*
     * Cold wallets cannot recreate transactions if the only thing provided is the wallet passphrase
     *
     * The transaction can be recreated if either
     * – there is an xprv
     * – there is a walletPassphrase and the wallet is not cold (because if it's cold, the passphrase is of little use)
     *
     * Therefore, if neither of these is true, the transaction cannot be recreated, which is reflected in the if
     * statement below.
     *
     * Lightning transactions cannot be recreated.
     */
    canRecreateTransaction(params) {
        const isColdWallet = !!_.get(this.wallet, '_wallet.isCold');
        const isOFCWallet = this.baseCoin.getFamily() === 'ofc'; // Off-chain transactions don't need to be rebuilt
        const isLightningWallet = this.baseCoin.getFamily() === 'lnbtc';
        if (isLightningWallet) {
            return false;
        }
        if (!params.xprv && !(params.walletPassphrase && !isColdWallet && !isOFCWallet)) {
            return false;
        }
        // If there are no recipients, then the transaction cannot be recreated
        const recipients = this.info()?.transactionRequest?.buildParams?.recipients || [];
        const type = this.info()?.transactionRequest?.buildParams?.type;
        // We only want to not recreate transactions with no recipients if it is a UTXO coin.
        // eslint-disable-next-line @typescript-eslint/no-explicit-any
        return !(utxolib.isValidNetwork(this.baseCoin.network) &&
            recipients.length === 0 &&
            type !== 'consolidate');
    }
    /*
     * Internal helper function to get the serialized transaction which is being approved.
     * If this PA is of type 'transactionRequest' this function will try to rebuild and resign the transaction
     * @param {ApproveOptions} params
     * @param {boolean} canRecreateTransaction -
     * @param {RequestTracer} reqId id tracer
     */
    async preApprove(params = {}, reqId) {
        // TransactionRequestLite or Multisig tx's must sign before pending approval is approved
        // Re-signed tx is provided to the pending approval api
        if (this.type() === pendingApproval_1.Type.TRANSACTION_REQUEST) {
            /*
             * If this is a request for approving a transaction, depending on whether this user has a private key to the wallet
             * (some admins may not have the spend permission), the transaction could either be rebroadcast as is, or it could
             * be reconstructed. It is preferable to reconstruct a tx in order to adhere to the latest network conditions
             * such as newer unspents, different fees, or a higher sequence id
             */
            if (params.tx) {
                // the approval tx was reconstructed and explicitly specified - pass it through
                return {
                    txHex: params.tx,
                };
            }
            // this user may not have spending privileges or a passphrase may not have been passed in
            if (!this.canRecreateTransaction(params)) {
                // If this is a TransactionRequest, then the txRequest already has the unsigned transaction
                if (this._pendingApproval.txRequestId) {
                    return undefined;
                }
                // If this is a MultiSig, then we need to fetch the half signed tx to propagate to the approval API
                const transaction = _.get(this.info(), `transactionRequest.coinSpecific.${this.baseCoin.type}`);
                if (!_.isObject(transaction)) {
                    throw new Error('there is neither an original transaction object nor can a new one be recreated');
                }
                return transaction;
            }
            if (this._pendingApproval.txRequestId) {
                return await this.recreateAndSignTSSTransaction(params, reqId);
            }
            return await this.recreateAndSignTransaction(params, reqId);
        }
    }
    /**
     * Internal helper function to perform any post-approval actions.
     * If type is 'transactionRequestFull', this will sign the txRequestFull if possible
     * @param params
     * @param reqId
     * @private
     */
    async postApprove(params = {}, reqId) {
        switch (this.type()) {
            case pendingApproval_1.Type.TRANSACTION_REQUEST_FULL:
                if (this._pendingApproval.state === pendingApproval_1.State.APPROVED) {
                    // After we approve a lightning transaction, we should proceed with submitting the payment
                    if (this.baseCoin.getFamily() === 'lnbtc') {
                        (0, assert_1.default)(this._pendingApproval.txRequestId, 'Missing txRequestId');
                        // this.populateWallet is called before this so we should be good here
                        (0, assert_1.default)(this.wallet?.id(), 'Missing wallet id');
                        await (0, common_1.sendTxRequest)(this.bitgo, this.wallet?.id(), this._pendingApproval.txRequestId, utils_1.RequestType.tx, reqId);
                    }
                    else if (this.canRecreateTransaction(params) && this.baseCoin.supportsTss()) {
                        await this.recreateAndSignTSSTransaction(params, reqId);
                    }
                }
        }
    }
    /**
     * Helper function to ensure that self.wallet is set
     */
    async populateWallet() {
        if (this.wallet) {
            return;
        }
        // TODO(WP-1341): consolidate/simplify this logic
        switch (this.type()) {
            case pendingApproval_1.Type.TRANSACTION_REQUEST:
                const transactionRequest = this.info().transactionRequest;
                if (_.isUndefined(transactionRequest)) {
                    throw new Error('missing required object property transactionRequest');
                }
                const updatedWallet = await this.baseCoin.wallets().get({ id: transactionRequest.sourceWallet });
                if (_.isUndefined(updatedWallet)) {
                    throw new Error('unexpected - unable to get wallet using sourcewallet');
                }
                this.wallet = updatedWallet;
                if (this.wallet.id() !== transactionRequest.sourceWallet) {
                    throw new Error('unexpected source wallet for pending approval');
                }
                break;
            case pendingApproval_1.Type.TRANSACTION_REQUEST_FULL:
                const walletId = this.walletId();
                if (!walletId) {
                    throw new Error('Unexpected error, pendingApproval.wallet is expected to be defined!');
                }
                this.wallet = await this.baseCoin.wallets().get({ id: this.walletId() });
                if (!this.wallet) {
                    throw new Error('unexpected - unable to get wallet using pendingApproval.wallet');
                }
                break;
        }
        return;
    }
}
exports.PendingApproval = PendingApproval;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pendingApproval.js","sourceRoot":"","sources":["../../../../src/bitgo/pendingApproval/pendingApproval.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;;GAEG;AACH,0CAA4B;AAC5B,yDAA2C;AAG3C,wDAQ4B;AAC5B,oCAAsD;AAEtD,uDAAoD;AAGpD,+DAA4C;AAC5C,8CAAiE;AAGjE,0CAA8C;AAC9C,oDAA4B;AAa5B,MAAa,eAAe;IAO1B,YAAY,KAAgB,EAAE,QAAmB,EAAE,mBAAwC,EAAE,MAAgB;QAC3G,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QAErB,IAAI,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE,EAAE,CAAC;YAChC,IAAI,IAAI,CAAC,QAAQ,CAAC,eAAe,EAAE,KAAK,OAAO,EAAE,CAAC;gBAChD,IAAI,IAAI,CAAC,MAAM,EAAE,mBAAmB,EAAE,KAAK,OAAO,EAAE,CAAC;oBACnD,IAAI,CAAC,QAAQ,GAAG,IAAI,uBAAe,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;gBACzE,CAAC;qBAAM,CAAC;oBACN,IAAI,CAAC,QAAQ,GAAG,IAAI,kBAAU,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;gBACpE,CAAC;YACH,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,QAAQ,GAAG,IAAI,eAAU,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YACpE,CAAC;QACH,CAAC;QAED,IAAI,CAAC,gBAAgB,GAAG,mBAAmB,CAAC;IAC9C,CAAC;IAED;;OAEG;IACH,EAAE;QACA,OAAO,IAAI,CAAC,gBAAgB,CAAC,EAAE,CAAC;IAClC,CAAC;IAED,MAAM;QACJ,OAAO,IAAI,CAAC,gBAAgB,CAAC;IAC/B,CAAC;IAED;;;;;OAKG;IACH,SAAS;QACP,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,EAAE,CAAC;YACjC,OAAO,2BAAS,CAAC,MAAM,CAAC;QAC1B,CAAC;aAAM,IAAI,IAAI,CAAC,gBAAgB,CAAC,UAAU,EAAE,CAAC;YAC5C,OAAO,2BAAS,CAAC,UAAU,CAAC;QAC9B,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,KAAK,CAAC,8EAA8E,CAAC,CAAC;QAClG,CAAC;IACH,CAAC;IAED;;OAEG;IACH,QAAQ;QACN,OAAO,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC;IACtC,CAAC;IAED;;OAEG;IACH,YAAY;QACV,OAAO,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC;IAC1C,CAAC;IAED;;OAEG;IACH,KAAK;QACH,OAAO,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC;IACrC,CAAC;IAED;;OAEG;IACH,OAAO;QACL,OAAO,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC;IACvC,CAAC;IAED;;OAEG;IACH,IAAI;QACF,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,IAAI,EAAE,CAAC;YAChC,MAAM,IAAI,KAAK,CAAC,wCAAwC,CAAC,CAAC;QAC5D,CAAC;QAED,OAAO,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC;IACzC,CAAC;IAED;;OAEG;IACH,IAAI;QACF,OAAO,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC;IACpC,CAAC;IAED;;;OAGG;IACH,iBAAiB;QACf,OAAO,IAAI,CAAC,gBAAgB,CAAC,iBAAiB,IAAI,CAAC,CAAC;IACtD,CAAC;IAED;;;OAGG;IACH,GAAG,CAAC,KAAK,GAAG,EAAE;QACZ,OAAO,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,oBAAoB,GAAG,IAAI,CAAC,EAAE,EAAE,GAAG,KAAK,CAAC,CAAC;IACrE,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,GAAG,CAAC,SAAgC,EAAE;QAC1C,IAAI,CAAC,gBAAgB,GAAG,MAAM,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC;QAClE,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,OAAO,CAAC,SAAyB,EAAE;QACvC,MAAM,CAAC,iBAAiB,GAAG,IAAI,CAAC;QAChC,MAAM,CAAC,iBAAiB,GAAG,IAAI,CAAC,EAAE,EAAE,CAAC;QACrC,MAAM,sBAAsB,GAAG,IAAI,CAAC,sBAAsB,CAAC,MAAM,CAAC,CAAC;QACnE,MAAM,KAAK,GAAG,IAAI,qBAAa,EAAE,CAAC;QAClC,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;QACnC,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC;QAE5B,IAAI,CAAC;YACH,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;YAEzD,MAAM,cAAc,GAAsC,EAAE,KAAK,EAAE,UAAU,EAAE,GAAG,EAAE,MAAM,CAAC,GAAG,EAAE,CAAC;YACjG,IAAI,WAAW,EAAE,CAAC;gBAChB,+EAA+E;gBAC/E,cAAc,CAAC,UAAU,GAAG,WAAW,CAAC,UAAU,IAAI,WAAW,CAAC;YACpE,CAAC;YACD,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,MAAM,EAAE,CAAC;YAEhF,8FAA8F;YAC9F,IAAI,QAAQ,CAAC,cAAc,CAAC,OAAO,CAAC,IAAI,QAAQ,CAAC,cAAc,CAAC,iBAAiB,CAAC,EAAE,CAAC;gBACnF,OAAO,QAAQ,CAAC;YAClB,CAAC;YAED,IAAI,CAAC,gBAAgB,GAAG,QAAQ,CAAC;YACjC,MAAM,IAAI,CAAC,WAAW,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;YAEtC,OAAO,IAAI,CAAC,gBAAgB,CAAC;QAC/B,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,IACE,CAAC,sBAAsB;gBACvB,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,yCAAyC,CAAC,KAAK,CAAC,CAAC;oBAClE,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,sEAAsE,CAAC,KAAK,CAAC,CAAC,CAAC,EACnG,CAAC;gBACD,MAAM,IAAI,KAAK,CAAC,8EAA8E,CAAC,CAAC;YAClG,CAAC;YACD,MAAM,CAAC,CAAC;QACV,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,MAAM,CAAC,SAAgC,EAAE;QAC7C,OAAO,MAAM,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC;IAC/E,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,MAAM,CAAC,SAAgC,EAAE;QAC7C,OAAO,MAAM,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IACnC,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,6BAA6B,CAAC,MAAsB,EAAE,KAAqB;QAC/E,MAAM,EAAE,gBAAgB,EAAE,GAAG,MAAM,CAAC;QACpC,MAAM,WAAW,GAAG,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC;QAEtD,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;YACjB,MAAM,IAAI,KAAK,CAAC,kBAAkB,CAAC,CAAC;QACtC,CAAC;QAED,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACtB,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAC;QAChD,CAAC;QAED,IAAI,CAAC,WAAW,EAAE,CAAC;YACjB,MAAM,IAAI,KAAK,CAAC,uBAAuB,CAAC,CAAC;QAC3C,CAAC;QAED,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,gBAAgB,EAAE,CAAC,CAAC;QACpE,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,QAAS,CAAC,iBAAiB,CAAC,WAAW,EAAE,YAAY,EAAE,KAAK,CAAC,CAAC;QAC3F,IAAI,SAAS,CAAC,UAAU,KAAK,MAAM,EAAE,CAAC;YACpC,IAAI,CAAC,SAAS,CAAC,WAAW,IAAI,SAAS,CAAC,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACjE,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;YAC3E,CAAC;YACD,OAAO;gBACL,KAAK,EAAE,SAAS,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,eAAe;aAChD,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,SAAS,CAAC,YAAY,IAAI,SAAS,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACnE,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;YAC3E,CAAC;YACD,OAAO;gBACL,KAAK,EAAE,SAAS,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,eAAe;aAC5D,CAAC;QACJ,CAAC;IACH,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,0BAA0B,CAAC,SAAc,EAAE,EAAE,KAAsB;QACvE,kEAAkE;QAClE,MAAM,kBAAkB,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC,kBAAkB,CAAC;QAC1D,IAAI,CAAC,CAAC,WAAW,CAAC,kBAAkB,CAAC,EAAE,CAAC;YACtC,MAAM,IAAI,KAAK,CAAC,yDAAyD,CAAC,CAAC;QAC7E,CAAC;QAED,IAAI,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;YAC/B,MAAM,IAAI,KAAK,CAAC,4CAA4C,CAAC,CAAC;QAChE,CAAC;QAED,MAAM,gBAAgB,GAAG,kBAAkB,CAAC,YAAY,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QAE7E,MAAM,UAAU,GAAG,kBAAkB,CAAC,UAAU,CAAC;QACjD,IAAI,cAAc,GAAG,CAAC,CAAC,MAAM,CAAC,EAAE,EAAE,MAAM,EAAE,EAAE,UAAU,EAAE,UAAU,EAAE,EAAE,kBAAkB,CAAC,WAAW,CAAC,CAAC;QAEtG,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,gBAAgB,CAAC,cAAc,CAAC,EAAE,CAAC;YACpD,cAAc,CAAC,GAAG,GAAG,IAAI,CAAC;QAC5B,CAAC;QAED,MAAM,SAAS,GAAG,KAAK,IAAI,IAAI,qBAAa,EAAE,CAAC;QAC/C,IAAI,kBAAkB,CAAC,WAAW,IAAI,kBAAkB,CAAC,WAAW,CAAC,IAAI,KAAK,aAAa,EAAE,CAAC;YAC5F,mFAAmF;YACnF,qEAAqE;YACrE,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC,SAAS,CAAC,CAAC;YACvC,cAAc,CAAC,UAAU,GAAG,MAAM,IAAI,CAAC,KAAK;iBACzC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,sBAAsB,CAAC,CAAC;iBAC7C,IAAI,CAAC,yBAAW,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC;iBACxC,MAAM,EAAE,CAAC;YACZ,OAAO,cAAc,CAAC,UAAU,CAAC;QACnC,CAAC;QAED,cAAc,GAAG,CAAC,CAAC,MAAM,CAAC,EAAE,EAAE,cAAc,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC;QAChE,MAAM,iBAAiB,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,0BAA0B,CAAC,cAAc,CAAC,CAAC;QACvF,qBAAqB;QACrB,MAAM,yBAAyB,GAAG,CAAC,MAAM,IAAI,CAAC,QAAQ,CAAC,gBAAgB,CAAC;YACtE,QAAQ,EAAE,cAAc;YACxB,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,UAAU,EAAE,gBAAgB;SAC7B,CAAC,CAAQ,CAAC;QACX,MAAM,0BAA0B,GAAG,CAAC,MAAM,IAAI,CAAC,QAAQ,CAAC,gBAAgB,CAAC;YACvE,QAAQ,EAAE,cAAc;YACxB,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,UAAU,EAAE,iBAAiB;SAC9B,CAAC,CAAQ,CAAC;QAEX,IAAI,CAAC,CAAC,WAAW,CAAC,0BAA0B,CAAC,2BAA2B,CAAC,EAAE,CAAC;YAC1E,OAAO,iBAAiB,CAAC;QAC3B,CAAC;QAED,IACE,OAAO,0BAA0B,CAAC,2BAA2B,KAAK,QAAQ;YAC1E,CAAC,CAAC,CAAC,QAAQ,CAAC,0BAA0B,CAAC,2BAA2B,CAAC,EACnE,CAAC;YACD,MAAM,IAAI,KAAK,CAAC,wDAAwD,CAAC,CAAC;QAC5E,CAAC;QACD,IACE,CAAC,CAAC,CAAC,WAAW,CAAC,yBAAyB,CAAC,2BAA2B,CAAC;YACrE,0BAA0B,CAAC,2BAA2B,GAAG,yBAAyB,CAAC,2BAA2B,EAC9G,CAAC;YACD,MAAM,IAAI,KAAK,CAAC,2DAA2D,CAAC,CAAC;QAC/E,CAAC;QACD,OAAO,iBAAiB,CAAC;IAC3B,CAAC;IAED;;;;;;;;;;;OAWG;IACK,sBAAsB,CAAC,MAAsB;QACnD,MAAM,YAAY,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;QAC5D,MAAM,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC,SAAS,EAAE,KAAK,KAAK,CAAC,CAAC,kDAAkD;QAC3G,MAAM,iBAAiB,GAAG,IAAI,CAAC,QAAQ,CAAC,SAAS,EAAE,KAAK,OAAO,CAAC;QAChE,IAAI,iBAAiB,EAAE,CAAC;YACtB,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IAAI,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,CAAC,MAAM,CAAC,gBAAgB,IAAI,CAAC,YAAY,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC;YAChF,OAAO,KAAK,CAAC;QACf,CAAC;QAED,uEAAuE;QACvE,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,EAAE,EAAE,kBAAkB,EAAE,WAAW,EAAE,UAAU,IAAI,EAAE,CAAC;QAClF,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,EAAE,EAAE,kBAAkB,EAAE,WAAW,EAAE,IAAI,CAAC;QAEhE,qFAAqF;QACrF,8DAA8D;QAC9D,OAAO,CAAC,CACN,OAAO,CAAC,cAAc,CAAE,IAAI,CAAC,QAAgB,CAAC,OAAO,CAAC;YACtD,UAAU,CAAC,MAAM,KAAK,CAAC;YACvB,IAAI,KAAK,aAAa,CACvB,CAAC;IACJ,CAAC;IAED;;;;;;OAMG;IACK,KAAK,CAAC,UAAU,CAAC,SAAyB,EAAE,EAAE,KAAqB;QACzE,wFAAwF;QACxF,uDAAuD;QACvD,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,sBAAI,CAAC,mBAAmB,EAAE,CAAC;YAC7C;;;;;eAKG;YACH,IAAI,MAAM,CAAC,EAAE,EAAE,CAAC;gBACd,+EAA+E;gBAC/E,OAAO;oBACL,KAAK,EAAE,MAAM,CAAC,EAAE;iBACjB,CAAC;YACJ,CAAC;YAED,yFAAyF;YACzF,IAAI,CAAC,IAAI,CAAC,sBAAsB,CAAC,MAAM,CAAC,EAAE,CAAC;gBACzC,2FAA2F;gBAC3F,IAAI,IAAI,CAAC,gBAAgB,CAAC,WAAW,EAAE,CAAC;oBACtC,OAAO,SAAS,CAAC;gBACnB,CAAC;gBACD,mGAAmG;gBACnG,MAAM,WAAW,GAAG,CAAC,CAAC,GAAG,CACvB,IAAI,CAAC,IAAI,EAAE,EACX,mCAAmC,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CACpC,CAAC;gBACtB,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;oBAC7B,MAAM,IAAI,KAAK,CAAC,gFAAgF,CAAC,CAAC;gBACpG,CAAC;gBACD,OAAO,WAAW,CAAC;YACrB,CAAC;YAED,IAAI,IAAI,CAAC,gBAAgB,CAAC,WAAW,EAAE,CAAC;gBACtC,OAAO,MAAM,IAAI,CAAC,6BAA6B,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;YACjE,CAAC;YACD,OAAO,MAAM,IAAI,CAAC,0BAA0B,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAC9D,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACK,KAAK,CAAC,WAAW,CAAC,SAAyB,EAAE,EAAE,KAAqB;QAC1E,QAAQ,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC;YACpB,KAAK,sBAAI,CAAC,wBAAwB;gBAChC,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,uBAAK,CAAC,QAAQ,EAAE,CAAC;oBACnD,0FAA0F;oBAC1F,IAAI,IAAI,CAAC,QAAQ,CAAC,SAAS,EAAE,KAAK,OAAO,EAAE,CAAC;wBAC1C,IAAA,gBAAM,EAAC,IAAI,CAAC,gBAAgB,CAAC,WAAW,EAAE,qBAAqB,CAAC,CAAC;wBACjE,sEAAsE;wBACtE,IAAA,gBAAM,EAAC,IAAI,CAAC,MAAM,EAAE,EAAE,EAAE,EAAE,mBAAmB,CAAC,CAAC;wBAC/C,MAAM,IAAA,sBAAa,EACjB,IAAI,CAAC,KAAK,EACV,IAAI,CAAC,MAAM,EAAE,EAAE,EAAY,EAC3B,IAAI,CAAC,gBAAgB,CAAC,WAAW,EACjC,mBAAW,CAAC,EAAE,EACd,KAAK,CACN,CAAC;oBACJ,CAAC;yBAAM,IAAI,IAAI,CAAC,sBAAsB,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE,EAAE,CAAC;wBAC9E,MAAM,IAAI,CAAC,6BAA6B,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;oBAC1D,CAAC;gBACH,CAAC;QACL,CAAC;IACH,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,cAAc;QAC1B,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,OAAO;QACT,CAAC;QACD,iDAAiD;QACjD,QAAQ,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC;YACpB,KAAK,sBAAI,CAAC,mBAAmB;gBAC3B,MAAM,kBAAkB,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC,kBAAkB,CAAC;gBAC1D,IAAI,CAAC,CAAC,WAAW,CAAC,kBAAkB,CAAC,EAAE,CAAC;oBACtC,MAAM,IAAI,KAAK,CAAC,qDAAqD,CAAC,CAAC;gBACzE,CAAC;gBAED,MAAM,aAAa,GAAY,MAAM,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,EAAE,EAAE,EAAE,kBAAkB,CAAC,YAAY,EAAE,CAAC,CAAC;gBAE1G,IAAI,CAAC,CAAC,WAAW,CAAC,aAAa,CAAC,EAAE,CAAC;oBACjC,MAAM,IAAI,KAAK,CAAC,sDAAsD,CAAC,CAAC;gBAC1E,CAAC;gBAED,IAAI,CAAC,MAAM,GAAG,aAAa,CAAC;gBAE5B,IAAI,IAAI,CAAC,MAAM,CAAC,EAAE,EAAE,KAAK,kBAAkB,CAAC,YAAY,EAAE,CAAC;oBACzD,MAAM,IAAI,KAAK,CAAC,+CAA+C,CAAC,CAAC;gBACnE,CAAC;gBACD,MAAM;YACR,KAAK,sBAAI,CAAC,wBAAwB;gBAChC,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACjC,IAAI,CAAC,QAAQ,EAAE,CAAC;oBACd,MAAM,IAAI,KAAK,CAAC,qEAAqE,CAAC,CAAC;gBACzF,CAAC;gBACD,IAAI,CAAC,MAAM,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,EAAE,EAAE,EAAE,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;gBACzE,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;oBACjB,MAAM,IAAI,KAAK,CAAC,gEAAgE,CAAC,CAAC;gBACpF,CAAC;gBACD,MAAM;QACV,CAAC;QACD,OAAO;IACT,CAAC;CACF;AApcD,0CAocC","sourcesContent":["/**\n * @prettier\n */\nimport * as _ from 'lodash';\nimport * as utxolib from '@bitgo/utxo-lib';\nimport { IBaseCoin } from '../baseCoin';\nimport { BitGoBase } from '../bitgoBase';\nimport {\n  ApproveOptions,\n  IPendingApproval,\n  OwnerType,\n  PendingApprovalData,\n  PendingApprovalInfo,\n  State,\n  Type,\n} from '../pendingApproval';\nimport { RequestTracer, RequestType } from '../utils';\nimport { IWallet } from '../wallet';\nimport { BuildParams } from '../wallet/BuildParams';\nimport { IRequestTracer } from '../../api';\nimport BaseTssUtils from '../utils/tss/baseTSSUtils';\nimport EddsaUtils from '../utils/tss/eddsa';\nimport { EcdsaMPCv2Utils, EcdsaUtils } from '../utils/tss/ecdsa';\nimport { KeyShare as EcdsaKeyShare } from '../utils/tss/ecdsa/types';\nimport { KeyShare as EddsaKeyShare } from '../utils/tss/eddsa/types';\nimport { sendTxRequest } from '../tss/common';\nimport assert from 'assert';\n\ntype PreApproveResult = {\n  txHex: string;\n  halfSigned?: string;\n};\n\ntype ApprovePendingApprovalRequestBody = {\n  state: 'approved';\n  otp: string | undefined;\n  halfSigned?: string | Omit<PreApproveResult, 'halfSigned'>;\n};\n\nexport class PendingApproval implements IPendingApproval {\n  private readonly bitgo: BitGoBase;\n  private readonly baseCoin: IBaseCoin;\n  private tssUtils?: BaseTssUtils<EcdsaKeyShare | EddsaKeyShare>;\n  private wallet?: IWallet;\n  private _pendingApproval: PendingApprovalData;\n\n  constructor(bitgo: BitGoBase, baseCoin: IBaseCoin, pendingApprovalData: PendingApprovalData, wallet?: IWallet) {\n    this.bitgo = bitgo;\n    this.baseCoin = baseCoin;\n    this.wallet = wallet;\n\n    if (this.baseCoin.supportsTss()) {\n      if (this.baseCoin.getMPCAlgorithm() === 'ecdsa') {\n        if (this.wallet?.multisigTypeVersion() === 'MPCv2') {\n          this.tssUtils = new EcdsaMPCv2Utils(this.bitgo, this.baseCoin, wallet);\n        } else {\n          this.tssUtils = new EcdsaUtils(this.bitgo, this.baseCoin, wallet);\n        }\n      } else {\n        this.tssUtils = new EddsaUtils(this.bitgo, this.baseCoin, wallet);\n      }\n    }\n\n    this._pendingApproval = pendingApprovalData;\n  }\n\n  /**\n   * Get the id for this PendingApproval\n   */\n  id(): string {\n    return this._pendingApproval.id;\n  }\n\n  toJSON(): PendingApprovalData {\n    return this._pendingApproval;\n  }\n\n  /**\n   * Get the owner type (wallet or enterprise)\n   * Pending approvals can be approved or modified by different scopes (depending on how they were created)\n   * If a pending approval is owned by a wallet, then it can be approved by administrators of the wallet\n   * If a pending approval is owned by an enterprise, then it can be approved by administrators of the enterprise\n   */\n  ownerType(): OwnerType {\n    if (this._pendingApproval.wallet) {\n      return OwnerType.WALLET;\n    } else if (this._pendingApproval.enterprise) {\n      return OwnerType.ENTERPRISE;\n    } else {\n      throw new Error('unexpected pending approval owner: neither wallet nor enterprise was present');\n    }\n  }\n\n  /**\n   * Get the id of the wallet which is associated with this PendingApproval\n   */\n  walletId(): string | undefined {\n    return this._pendingApproval.wallet;\n  }\n\n  /**\n   * Get the enterprise ID that is associated with this PendingApproval\n   */\n  enterpriseId(): string | undefined {\n    return this._pendingApproval.enterprise;\n  }\n\n  /**\n   * Get the state of this PendingApproval\n   */\n  state(): State {\n    return this._pendingApproval.state;\n  }\n\n  /**\n   * Get the id of the user that performed the action resulting in this PendingApproval\n   */\n  creator(): string {\n    return this._pendingApproval.creator;\n  }\n\n  /**\n   * Get the type of the pending approval (what it approves)\n   */\n  type(): Type {\n    if (!this._pendingApproval.info) {\n      throw new Error('pending approval info is not available');\n    }\n\n    return this._pendingApproval.info.type;\n  }\n\n  /**\n   * Get information about this PendingApproval\n   */\n  info(): PendingApprovalInfo {\n    return this._pendingApproval.info;\n  }\n\n  /**\n   * Get the number of approvals that are required for this PendingApproval to be approved.\n   * Defaults to 1 if approvalsRequired doesn't exist on the object\n   */\n  approvalsRequired(): number {\n    return this._pendingApproval.approvalsRequired || 1;\n  }\n\n  /**\n   * Generate a url for this PendingApproval for making requests to the server.\n   * @param extra\n   */\n  url(extra = ''): string {\n    return this.baseCoin.url('/pendingapprovals/' + this.id() + extra);\n  }\n\n  /**\n   * Refetches this PendingApproval from the server and returns it.\n   *\n   * Note that this mutates the PendingApproval object in place.\n   * @param params\n   */\n  async get(params: Record<string, never> = {}): Promise<PendingApproval> {\n    this._pendingApproval = await this.bitgo.get(this.url()).result();\n    return this;\n  }\n\n  /**\n   * Sets this PendingApproval to an approved state\n   */\n  async approve(params: ApproveOptions = {}): Promise<any> {\n    params.previewPendingTxs = true;\n    params.pendingApprovalId = this.id();\n    const canRecreateTransaction = this.canRecreateTransaction(params);\n    const reqId = new RequestTracer();\n    this.bitgo.setRequestTracer(reqId);\n    await this.populateWallet();\n\n    try {\n      const transaction = await this.preApprove(params, reqId);\n\n      const approvalParams: ApprovePendingApprovalRequestBody = { state: 'approved', otp: params.otp };\n      if (transaction) {\n        // if the transaction already has a half signed property, we take that directly\n        approvalParams.halfSigned = transaction.halfSigned || transaction;\n      }\n      const response = await this.bitgo.put(this.url()).send(approvalParams).result();\n\n      // if the response comes with an error, means that the transaction triggered another condition\n      if (response.hasOwnProperty('error') && response.hasOwnProperty('pendingApproval')) {\n        return response;\n      }\n\n      this._pendingApproval = response;\n      await this.postApprove(params, reqId);\n\n      return this._pendingApproval;\n    } catch (e) {\n      if (\n        !canRecreateTransaction &&\n        (e.message.indexOf('could not find unspent output for input') !== -1 ||\n          e.message.indexOf('transaction conflicts with an existing transaction in the send queue') !== -1)\n      ) {\n        throw new Error('unspents expired, wallet passphrase or xprv required to recreate transaction');\n      }\n      throw e;\n    }\n  }\n\n  /**\n   * Sets this PendingApproval to a rejected state\n   * @param params\n   */\n  async reject(params: Record<string, never> = {}): Promise<any> {\n    return await this.bitgo.put(this.url()).send({ state: 'rejected' }).result();\n  }\n\n  /**\n   * Alias for PendingApproval.reject()\n   *\n   * @deprecated\n   * @param params\n   */\n  async cancel(params: Record<string, never> = {}): Promise<any> {\n    return await this.reject(params);\n  }\n\n  /**\n   * Recreate and sign TSS transaction\n   * @param {ApproveOptions} params needed to get txs and use the walletPassphrase to tss sign\n   * @param {RequestTracer} reqId id tracer.\n   */\n  async recreateAndSignTSSTransaction(params: ApproveOptions, reqId: IRequestTracer): Promise<{ txHex: string }> {\n    const { walletPassphrase } = params;\n    const txRequestId = this._pendingApproval.txRequestId;\n\n    if (!this.wallet) {\n      throw new Error('Wallet not found');\n    }\n\n    if (!walletPassphrase) {\n      throw new Error('walletPassphrase not found');\n    }\n\n    if (!txRequestId) {\n      throw new Error('txRequestId not found');\n    }\n\n    const decryptedPrv = await this.wallet.getPrv({ walletPassphrase });\n    const txRequest = await this.tssUtils!.recreateTxRequest(txRequestId, decryptedPrv, reqId);\n    if (txRequest.apiVersion === 'lite') {\n      if (!txRequest.unsignedTxs || txRequest.unsignedTxs.length === 0) {\n        throw new Error('Unexpected error, no transactions found in txRequest.');\n      }\n      return {\n        txHex: txRequest.unsignedTxs[0].serializedTxHex,\n      };\n    } else {\n      if (!txRequest.transactions || txRequest.transactions.length === 0) {\n        throw new Error('Unexpected error, no transactions found in txRequest.');\n      }\n      return {\n        txHex: txRequest.transactions[0].unsignedTx.serializedTxHex,\n      };\n    }\n  }\n\n  /**\n   * Recreate a transaction for a pending approval to respond to updated network conditions\n   * @param params\n   * @param reqId\n   */\n  async recreateAndSignTransaction(params: any = {}, reqId?: IRequestTracer): Promise<any> {\n    // this method only makes sense with existing transaction requests\n    const transactionRequest = this.info().transactionRequest;\n    if (_.isUndefined(transactionRequest)) {\n      throw new Error('cannot recreate transaction without transaction request');\n    }\n\n    if (_.isUndefined(this.wallet)) {\n      throw new Error('cannot recreate transaction without wallet');\n    }\n\n    const originalPrebuild = transactionRequest.coinSpecific[this.baseCoin.type];\n\n    const recipients = transactionRequest.recipients;\n    let prebuildParams = _.extend({}, params, { recipients: recipients }, transactionRequest.buildParams);\n\n    if (!_.isUndefined(originalPrebuild.hopTransaction)) {\n      prebuildParams.hop = true;\n    }\n\n    const reqTracer = reqId || new RequestTracer();\n    if (transactionRequest.buildParams && transactionRequest.buildParams.type === 'consolidate') {\n      // consolidate tag is in the build params - this is a consolidation transaction, so\n      // it needs to be rebuilt using the special consolidation build route\n      this.bitgo.setRequestTracer(reqTracer);\n      prebuildParams.prebuildTx = await this.bitgo\n        .post(this.wallet.url(`/consolidateUnspents`))\n        .send(BuildParams.encode(prebuildParams))\n        .result();\n      delete prebuildParams.recipients;\n    }\n\n    prebuildParams = _.extend({}, prebuildParams, { reqId: reqId });\n    const signedTransaction = await this.wallet.prebuildAndSignTransaction(prebuildParams);\n    // compare PAYGo fees\n    const originalParsedTransaction = (await this.baseCoin.parseTransaction({\n      txParams: prebuildParams,\n      wallet: this.wallet,\n      txPrebuild: originalPrebuild,\n    })) as any;\n    const recreatedParsedTransaction = (await this.baseCoin.parseTransaction({\n      txParams: prebuildParams,\n      wallet: this.wallet,\n      txPrebuild: signedTransaction,\n    })) as any;\n\n    if (_.isUndefined(recreatedParsedTransaction.implicitExternalSpendAmount)) {\n      return signedTransaction;\n    }\n\n    if (\n      typeof recreatedParsedTransaction.implicitExternalSpendAmount !== 'bigint' &&\n      !_.isFinite(recreatedParsedTransaction.implicitExternalSpendAmount)\n    ) {\n      throw new Error('implicit external spend amount could not be determined');\n    }\n    if (\n      !_.isUndefined(originalParsedTransaction.implicitExternalSpendAmount) &&\n      recreatedParsedTransaction.implicitExternalSpendAmount > originalParsedTransaction.implicitExternalSpendAmount\n    ) {\n      throw new Error('recreated transaction is using a higher pay-as-you-go-fee');\n    }\n    return signedTransaction;\n  }\n\n  /*\n   * Cold wallets cannot recreate transactions if the only thing provided is the wallet passphrase\n   *\n   * The transaction can be recreated if either\n   * – there is an xprv\n   * – there is a walletPassphrase and the wallet is not cold (because if it's cold, the passphrase is of little use)\n   *\n   * Therefore, if neither of these is true, the transaction cannot be recreated, which is reflected in the if\n   * statement below.\n   *\n   * Lightning transactions cannot be recreated.\n   */\n  private canRecreateTransaction(params: ApproveOptions): boolean {\n    const isColdWallet = !!_.get(this.wallet, '_wallet.isCold');\n    const isOFCWallet = this.baseCoin.getFamily() === 'ofc'; // Off-chain transactions don't need to be rebuilt\n    const isLightningWallet = this.baseCoin.getFamily() === 'lnbtc';\n    if (isLightningWallet) {\n      return false;\n    }\n\n    if (!params.xprv && !(params.walletPassphrase && !isColdWallet && !isOFCWallet)) {\n      return false;\n    }\n\n    // If there are no recipients, then the transaction cannot be recreated\n    const recipients = this.info()?.transactionRequest?.buildParams?.recipients || [];\n    const type = this.info()?.transactionRequest?.buildParams?.type;\n\n    // We only want to not recreate transactions with no recipients if it is a UTXO coin.\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    return !(\n      utxolib.isValidNetwork((this.baseCoin as any).network) &&\n      recipients.length === 0 &&\n      type !== 'consolidate'\n    );\n  }\n\n  /*\n   * Internal helper function to get the serialized transaction which is being approved.\n   * If this PA is of type 'transactionRequest' this function will try to rebuild and resign the transaction\n   * @param {ApproveOptions} params\n   * @param {boolean} canRecreateTransaction -\n   * @param {RequestTracer} reqId id tracer\n   */\n  private async preApprove(params: ApproveOptions = {}, reqId: IRequestTracer): Promise<PreApproveResult | undefined> {\n    // TransactionRequestLite or Multisig tx's must sign before pending approval is approved\n    // Re-signed tx is provided to the pending approval api\n    if (this.type() === Type.TRANSACTION_REQUEST) {\n      /*\n       * If this is a request for approving a transaction, depending on whether this user has a private key to the wallet\n       * (some admins may not have the spend permission), the transaction could either be rebroadcast as is, or it could\n       * be reconstructed. It is preferable to reconstruct a tx in order to adhere to the latest network conditions\n       * such as newer unspents, different fees, or a higher sequence id\n       */\n      if (params.tx) {\n        // the approval tx was reconstructed and explicitly specified - pass it through\n        return {\n          txHex: params.tx,\n        };\n      }\n\n      // this user may not have spending privileges or a passphrase may not have been passed in\n      if (!this.canRecreateTransaction(params)) {\n        // If this is a TransactionRequest, then the txRequest already has the unsigned transaction\n        if (this._pendingApproval.txRequestId) {\n          return undefined;\n        }\n        // If this is a MultiSig, then we need to fetch the half signed tx to propagate to the approval API\n        const transaction = _.get(\n          this.info(),\n          `transactionRequest.coinSpecific.${this.baseCoin.type}`\n        ) as PreApproveResult;\n        if (!_.isObject(transaction)) {\n          throw new Error('there is neither an original transaction object nor can a new one be recreated');\n        }\n        return transaction;\n      }\n\n      if (this._pendingApproval.txRequestId) {\n        return await this.recreateAndSignTSSTransaction(params, reqId);\n      }\n      return await this.recreateAndSignTransaction(params, reqId);\n    }\n  }\n\n  /**\n   * Internal helper function to perform any post-approval actions.\n   * If type is 'transactionRequestFull', this will sign the txRequestFull if possible\n   * @param params\n   * @param reqId\n   * @private\n   */\n  private async postApprove(params: ApproveOptions = {}, reqId: IRequestTracer): Promise<void> {\n    switch (this.type()) {\n      case Type.TRANSACTION_REQUEST_FULL:\n        if (this._pendingApproval.state === State.APPROVED) {\n          // After we approve a lightning transaction, we should proceed with submitting the payment\n          if (this.baseCoin.getFamily() === 'lnbtc') {\n            assert(this._pendingApproval.txRequestId, 'Missing txRequestId');\n            // this.populateWallet is called before this so we should be good here\n            assert(this.wallet?.id(), 'Missing wallet id');\n            await sendTxRequest(\n              this.bitgo,\n              this.wallet?.id() as string,\n              this._pendingApproval.txRequestId,\n              RequestType.tx,\n              reqId\n            );\n          } else if (this.canRecreateTransaction(params) && this.baseCoin.supportsTss()) {\n            await this.recreateAndSignTSSTransaction(params, reqId);\n          }\n        }\n    }\n  }\n\n  /**\n   * Helper function to ensure that self.wallet is set\n   */\n  private async populateWallet(): Promise<undefined> {\n    if (this.wallet) {\n      return;\n    }\n    // TODO(WP-1341): consolidate/simplify this logic\n    switch (this.type()) {\n      case Type.TRANSACTION_REQUEST:\n        const transactionRequest = this.info().transactionRequest;\n        if (_.isUndefined(transactionRequest)) {\n          throw new Error('missing required object property transactionRequest');\n        }\n\n        const updatedWallet: IWallet = await this.baseCoin.wallets().get({ id: transactionRequest.sourceWallet });\n\n        if (_.isUndefined(updatedWallet)) {\n          throw new Error('unexpected - unable to get wallet using sourcewallet');\n        }\n\n        this.wallet = updatedWallet;\n\n        if (this.wallet.id() !== transactionRequest.sourceWallet) {\n          throw new Error('unexpected source wallet for pending approval');\n        }\n        break;\n      case Type.TRANSACTION_REQUEST_FULL:\n        const walletId = this.walletId();\n        if (!walletId) {\n          throw new Error('Unexpected error, pendingApproval.wallet is expected to be defined!');\n        }\n        this.wallet = await this.baseCoin.wallets().get({ id: this.walletId() });\n        if (!this.wallet) {\n          throw new Error('unexpected - unable to get wallet using pendingApproval.wallet');\n        }\n        break;\n    }\n    return;\n  }\n}\n"]}

Выполнить команду


Для локальной разработки. Не используйте в интернете!