PHP WebShell

Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/sdk-core/dist/src/bitgo/wallet

Просмотр файла: wallets.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.Wallets = void 0;
exports.isWalletWithKeychains = isWalletWithKeychains;
/**
 * @prettier
 */
const assert_1 = __importDefault(require("assert"));
const bignumber_js_1 = require("bignumber.js");
const utxo_lib_1 = require("@bitgo/utxo-lib");
const _ = __importStar(require("lodash"));
const statics_1 = require("@bitgo/statics");
const api_1 = require("../../api");
const common = __importStar(require("../../common"));
const ecdh_1 = require("../ecdh");
const keychain_1 = require("../keychain");
const utils_1 = require("../utils");
const iWallets_1 = require("./iWallets");
const wallet_1 = require("./wallet");
/**
 * Check if a wallet is a WalletWithKeychains
 */
function isWalletWithKeychains(wallet) {
    return wallet.responseType === 'WalletWithKeychains';
}
class Wallets {
    constructor(bitgo, baseCoin) {
        this.bitgo = bitgo;
        this.baseCoin = baseCoin;
    }
    /**
     * Get a wallet by ID (proxy for getWallet)
     * @param params
     */
    async get(params = {}) {
        return this.getWallet(params);
    }
    /**
     * List a user's wallets
     * @param params
     * @returns {*}
     */
    async list(params = {}) {
        if (params.skip && params.prevId) {
            throw new Error('cannot specify both skip and prevId');
        }
        const body = (await this.bitgo.get(this.baseCoin.url('/wallet')).query(params).result());
        body.wallets = body.wallets.map((w) => new wallet_1.Wallet(this.bitgo, this.baseCoin, w));
        return body;
    }
    /**
     * add
     * Add a new wallet (advanced mode).
     * This allows you to manually submit the keys, type, m and n of the wallet
     * Parameters include:
     *    "label": label of the wallet to be shown in UI
     *    "m": number of keys required to unlock wallet (2)
     *    "n": number of keys available on the wallet (3)
     *    "keys": array of keychain ids
     */
    async add(params) {
        params = params || {};
        common.validateParams(params, [], ['label', 'enterprise', 'type']);
        if (typeof params.label !== 'string') {
            throw new Error('missing required string parameter label');
        }
        // no need to pass keys for (single) custodial wallets
        if (params.type !== 'custodial') {
            if (Array.isArray(params.keys) === false || !_.isNumber(params.m) || !_.isNumber(params.n)) {
                throw new Error('invalid argument');
            }
            // TODO: support more types of multisig
            if (!this.baseCoin.isValidMofNSetup(params)) {
                throw new Error('unsupported multi-sig type');
            }
        }
        if (params.gasPrice && !_.isNumber(params.gasPrice)) {
            throw new Error('invalid argument for gasPrice - number expected');
        }
        if (params.walletVersion) {
            if (!_.isNumber(params.walletVersion)) {
                throw new Error('invalid argument for walletVersion - number expected');
            }
            if (params.multisigType === 'tss' && this.baseCoin.getMPCAlgorithm() === 'ecdsa' && params.walletVersion === 3) {
                const tssSettings = await this.bitgo
                    .get(this.bitgo.microservicesUrl('/api/v2/tss/settings'))
                    .result();
                const multisigTypeVersion = tssSettings.coinSettings[this.baseCoin.getFamily()]?.walletCreationSettings?.multiSigTypeVersion;
                if (multisigTypeVersion === 'MPCv2') {
                    params.walletVersion = 5;
                }
            }
        }
        if (params.tags && Array.isArray(params.tags) === false) {
            throw new Error('invalid argument for tags - array expected');
        }
        if (params.clientFlags && Array.isArray(params.clientFlags) === false) {
            throw new Error('invalid argument for clientFlags - array expected');
        }
        if (params.isCold && !_.isBoolean(params.isCold)) {
            throw new Error('invalid argument for isCold - boolean expected');
        }
        if (params.isCustodial && !_.isBoolean(params.isCustodial)) {
            throw new Error('invalid argument for isCustodial - boolean expected');
        }
        if (params.address && (!_.isString(params.address) || !this.baseCoin.isValidAddress(params.address))) {
            throw new Error('invalid argument for address - valid address string expected');
        }
        const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(params).result();
        return {
            wallet: new wallet_1.Wallet(this.bitgo, this.baseCoin, newWallet),
        };
    }
    async generateLightningWallet(params) {
        const reqId = new utils_1.RequestTracer();
        this.bitgo.setRequestTracer(reqId);
        const { label, passphrase, enterprise, passcodeEncryptionCode, subType } = params;
        // TODO BTC-1899: only userAuth key is required for custodial lightning wallet. all 3 keys are required for self custodial lightning.
        // to avoid changing the platform for custodial flow, let us all 3 keys both wallet types.
        const keychainPromises = [undefined, 'userAuth', 'nodeAuth'].map((purpose) => {
            return async () => {
                const keychain = this.baseCoin.keychains().create();
                const keychainParams = {
                    pub: keychain.pub,
                    encryptedPrv: this.bitgo.encrypt({ password: passphrase, input: keychain.prv }),
                    originalPasscodeEncryptionCode: purpose === undefined ? passcodeEncryptionCode : undefined,
                    coinSpecific: purpose === undefined ? undefined : { [this.baseCoin.getChain()]: { purpose } },
                    keyType: 'independent',
                    source: 'user',
                };
                return await this.baseCoin.keychains().add(keychainParams);
            };
        });
        const { userKeychain, userAuthKeychain, nodeAuthKeychain } = await (0, utils_1.promiseProps)({
            userKeychain: keychainPromises[0](),
            userAuthKeychain: keychainPromises[1](),
            nodeAuthKeychain: keychainPromises[2](),
        });
        const walletParams = {
            label,
            m: 1,
            n: 1,
            type: 'hot',
            subType,
            enterprise,
            keys: [userKeychain.id],
            coinSpecific: { [this.baseCoin.getChain()]: { keys: [userAuthKeychain.id, nodeAuthKeychain.id] } },
        };
        const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(walletParams).result();
        const wallet = new wallet_1.Wallet(this.bitgo, this.baseCoin, newWallet);
        return {
            wallet,
            userKeychain,
            userAuthKeychain,
            nodeAuthKeychain,
            responseType: 'LightningWalletWithKeychains',
        };
    }
    /**
     * Generate a new wallet
     * 1. Creates the user keychain locally on the client, and encrypts it with the provided passphrase
     * 2. If no pub was provided, creates the backup keychain locally on the client, and encrypts it with the provided passphrase
     * 3. Uploads the encrypted user and backup keychains to BitGo
     * 4. Creates the BitGo key on the service
     * 5. Creates the wallet on BitGo with the 3 public keys above
     * @param params
     * @param params.label Label for the wallet
     * @param params.passphrase Passphrase to be used to encrypt the user and backup keychains
     * @param params.userKey User xpub
     * @param params.backupXpub Backup xpub
     * @param params.backupXpubProvider
     * @param params.enterprise the enterpriseId
     * @param params.disableTransactionNotifications
     * @param params.passcodeEncryptionCode optional this is a recovery code that can be used to decrypt the original passphrase in a recovery case.
     *                                      The user must generate and keep the encrypted original passphrase safe while this code is stored on BitGo
     * @param params.coldDerivationSeed optional seed for SMC wallets
     * @param params.gasPrice
     * @param params.disableKRSEmail
     * @param params.walletVersion
     * @param params.multisigType optional multisig type, 'onchain' or 'tss' or 'blsdkg'; if absent, we will defer to the coin's default type
     * @param params.isDistributedCustody optional parameter for creating bitgo key. This is only necessary if you want to create
     *                                    a distributed custody wallet. If provided, you must have the enterprise license and pass in
     *                                    `params.enterprise` into `generateWallet` as well.
     * @param params.type optional wallet type, 'hot' or 'cold' or 'custodial'; if absent, we will defer to 'hot'
     * @param params.bitgoKeyId optional bitgo key id for SMC TSS wallets
     * @param params.commonKeychain optional common keychain for SMC TSS wallets
     *
     * @returns {*}
     */
    async generateWallet(params = {}) {
        // Assign the default multiSig type value based on the coin
        if (!params.multisigType) {
            params.multisigType = this.baseCoin.getDefaultMultisigType();
        }
        if (this.baseCoin.getFamily() === 'lnbtc') {
            const options = (0, utils_1.decodeOrElse)(iWallets_1.GenerateLightningWalletOptionsCodec.name, iWallets_1.GenerateLightningWalletOptionsCodec, params, (errors) => {
                throw new Error(`error(s) parsing generate lightning wallet request params: ${errors}`);
            });
            const walletData = await this.generateLightningWallet(options);
            walletData.encryptedWalletPassphrase = this.bitgo.encrypt({
                input: options.passphrase,
                password: options.passcodeEncryptionCode,
            });
            return walletData;
        }
        common.validateParams(params, ['label'], ['passphrase', 'userKey', 'backupXpub']);
        if (typeof params.label !== 'string') {
            throw new Error('missing required string parameter label');
        }
        const { type = 'hot', label, passphrase, enterprise, isDistributedCustody } = params;
        const isTss = params.multisigType === 'tss' && this.baseCoin.supportsTss();
        const canEncrypt = !!passphrase && typeof passphrase === 'string';
        const walletParams = {
            label: label,
            m: 2,
            n: 3,
            keys: [],
            type: !!params.userKey && params.multisigType !== 'onchain' ? 'cold' : type,
        };
        if (!_.isUndefined(params.passcodeEncryptionCode)) {
            if (!_.isString(params.passcodeEncryptionCode)) {
                throw new Error('passcodeEncryptionCode must be a string');
            }
        }
        if (!_.isUndefined(enterprise)) {
            if (!_.isString(enterprise)) {
                throw new Error('invalid enterprise argument, expecting string');
            }
            walletParams.enterprise = enterprise;
        }
        // EVM TSS wallets must use wallet version 3, 5 and 6
        if (isTss &&
            this.baseCoin.isEVM() &&
            !(params.walletVersion === 3 || params.walletVersion === 5 || params.walletVersion === 6)) {
            throw new Error('EVM TSS wallets are only supported for wallet version 3, 5 and 6');
        }
        if (isTss) {
            if (!this.baseCoin.supportsTss()) {
                throw new Error(`coin ${this.baseCoin.getFamily()} does not support TSS at this time`);
            }
            if ((params.walletVersion === 5 || params.walletVersion === 6) &&
                !this.baseCoin.getConfig().features.includes(statics_1.CoinFeature.MPCV2)) {
                throw new Error(`coin ${this.baseCoin.getFamily()} does not support TSS MPCv2 at this time`);
            }
            (0, assert_1.default)(enterprise, 'enterprise is required for TSS wallet');
            if (type === 'cold') {
                // validate
                (0, assert_1.default)(params.bitgoKeyId, 'bitgoKeyId is required for SMC TSS wallet');
                (0, assert_1.default)(params.commonKeychain, 'commonKeychain is required for SMC TSS wallet');
                return this.generateSMCMpcWallet({
                    multisigType: 'tss',
                    label,
                    enterprise,
                    walletVersion: params.walletVersion,
                    bitgoKeyId: params.bitgoKeyId,
                    commonKeychain: params.commonKeychain,
                    coldDerivationSeed: params.coldDerivationSeed,
                });
            }
            if (type === 'custodial') {
                return this.generateCustodialMpcWallet({
                    multisigType: 'tss',
                    label,
                    enterprise,
                    walletVersion: params.walletVersion,
                });
            }
            (0, assert_1.default)(passphrase, 'cannot generate TSS keys without passphrase');
            const walletData = await this.generateMpcWallet({
                multisigType: 'tss',
                label,
                passphrase,
                originalPasscodeEncryptionCode: params.passcodeEncryptionCode,
                enterprise,
                walletVersion: params.walletVersion,
            });
            if (params.passcodeEncryptionCode) {
                walletData.encryptedWalletPassphrase = this.bitgo.encrypt({
                    input: passphrase,
                    password: params.passcodeEncryptionCode,
                });
            }
            return walletData;
        }
        // Handle distributed custody
        if (isDistributedCustody) {
            if (!enterprise) {
                throw new Error('must provide enterprise when creating distributed custody wallet');
            }
            if (!type || type !== 'cold') {
                throw new Error('distributed custody wallets must be type: cold');
            }
        }
        const hasBackupXpub = !!params.backupXpub;
        const hasBackupXpubProvider = !!params.backupXpubProvider;
        if (hasBackupXpub && hasBackupXpubProvider) {
            throw new Error('Cannot provide more than one backupXpub or backupXpubProvider flag');
        }
        if (params.gasPrice && params.eip1559) {
            throw new Error('can not use both eip1559 and gasPrice values');
        }
        if (!_.isUndefined(params.disableTransactionNotifications)) {
            if (!_.isBoolean(params.disableTransactionNotifications)) {
                throw new Error('invalid disableTransactionNotifications argument, expecting boolean');
            }
            walletParams.disableTransactionNotifications = params.disableTransactionNotifications;
        }
        if (!_.isUndefined(params.gasPrice)) {
            const gasPriceBN = new bignumber_js_1.BigNumber(params.gasPrice);
            if (gasPriceBN.isNaN()) {
                throw new Error('invalid gas price argument, expecting number or number as string');
            }
            walletParams.gasPrice = gasPriceBN.toString();
        }
        if (!_.isUndefined(params.eip1559) && !_.isEmpty(params.eip1559)) {
            const maxFeePerGasBN = new bignumber_js_1.BigNumber(params.eip1559.maxFeePerGas);
            if (maxFeePerGasBN.isNaN()) {
                throw new Error('invalid max fee argument, expecting number or number as string');
            }
            const maxPriorityFeePerGasBN = new bignumber_js_1.BigNumber(params.eip1559.maxPriorityFeePerGas);
            if (maxPriorityFeePerGasBN.isNaN()) {
                throw new Error('invalid priority fee argument, expecting number or number as string');
            }
            walletParams.eip1559 = {
                maxFeePerGas: maxFeePerGasBN.toString(),
                maxPriorityFeePerGas: maxPriorityFeePerGasBN.toString(),
            };
        }
        if (!_.isUndefined(params.disableKRSEmail)) {
            if (!_.isBoolean(params.disableKRSEmail)) {
                throw new Error('invalid disableKRSEmail argument, expecting boolean');
            }
            walletParams.disableKRSEmail = params.disableKRSEmail;
        }
        if (!_.isUndefined(params.walletVersion)) {
            if (!_.isNumber(params.walletVersion)) {
                throw new Error('invalid walletVersion provided, expecting number');
            }
            walletParams.walletVersion = params.walletVersion;
        }
        // Ensure each krsSpecific param is either a string, boolean, or number
        const { krsSpecific } = params;
        if (!_.isUndefined(krsSpecific)) {
            Object.keys(krsSpecific).forEach((key) => {
                const val = krsSpecific[key];
                if (!_.isBoolean(val) && !_.isString(val) && !_.isNumber(val)) {
                    throw new Error('krsSpecific object contains illegal values. values must be strings, booleans, or numbers');
                }
            });
        }
        let derivationPath = undefined;
        const reqId = new utils_1.RequestTracer();
        if (params.type === 'custodial' && (params.multisigType ?? 'onchain') === 'onchain') {
            // for custodial multisig, when the wallet is created on the platfor side, the keys are not needed
            walletParams.n = undefined;
            walletParams.m = undefined;
            walletParams.keys = undefined;
            walletParams.keySignatures = undefined;
            const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(walletParams).result(); // returns the ids
            const userKeychain = this.baseCoin.keychains().get({ id: newWallet.keys[keychain_1.KeyIndices.USER], reqId });
            const backupKeychain = this.baseCoin.keychains().get({ id: newWallet.keys[keychain_1.KeyIndices.BACKUP], reqId });
            const bitgoKeychain = this.baseCoin.keychains().get({ id: newWallet.keys[keychain_1.KeyIndices.BITGO], reqId });
            const [userKey, bitgoKey, backupKey] = await Promise.all([userKeychain, bitgoKeychain, backupKeychain]);
            const result = {
                wallet: new wallet_1.Wallet(this.bitgo, this.baseCoin, newWallet),
                userKeychain: userKey,
                backupKeychain: bitgoKey,
                bitgoKeychain: backupKey,
                responseType: 'WalletWithKeychains',
            };
            return result;
        }
        else {
            const userKeychainPromise = async () => {
                let userKeychainParams;
                let userKeychain;
                // User provided user key
                if (params.userKey) {
                    userKeychain = { pub: params.userKey };
                    userKeychainParams = userKeychain;
                    if (params.coldDerivationSeed) {
                        // the derivation only makes sense when a key already exists
                        const derivation = this.baseCoin.deriveKeyWithSeed({
                            key: params.userKey,
                            seed: params.coldDerivationSeed,
                        });
                        derivationPath = derivation.derivationPath;
                        userKeychain.pub = derivation.key;
                        userKeychain.derivedFromParentWithSeed = params.coldDerivationSeed;
                    }
                }
                else {
                    if (!canEncrypt) {
                        throw new Error('cannot generate user keypair without passphrase');
                    }
                    // Create the user key.
                    userKeychain = this.baseCoin.keychains().create();
                    userKeychain.encryptedPrv = this.bitgo.encrypt({ password: passphrase, input: userKeychain.prv });
                    userKeychainParams = {
                        pub: userKeychain.pub,
                        encryptedPrv: userKeychain.encryptedPrv,
                        originalPasscodeEncryptionCode: params.passcodeEncryptionCode,
                    };
                }
                userKeychainParams.reqId = reqId;
                const newUserKeychain = await this.baseCoin.keychains().add(userKeychainParams);
                return _.extend({}, newUserKeychain, userKeychain);
            };
            const backupKeychainPromise = async () => {
                if (params.backupXpubProvider) {
                    // If requested, use a KRS or backup key provider
                    return this.baseCoin.keychains().createBackup({
                        provider: params.backupXpubProvider || 'defaultRMGBackupProvider',
                        disableKRSEmail: params.disableKRSEmail,
                        krsSpecific: params.krsSpecific,
                        type: this.baseCoin.getChain(),
                        passphrase: params.passphrase,
                        reqId,
                    });
                }
                // User provided backup xpub
                if (params.backupXpub) {
                    // user provided backup ethereum address
                    return this.baseCoin.keychains().add({
                        pub: params.backupXpub,
                        source: 'backup',
                        reqId,
                    });
                }
                else {
                    if (!canEncrypt) {
                        throw new Error('cannot generate backup keypair without passphrase');
                    }
                    // No provided backup xpub or address, so default to creating one here
                    return this.baseCoin.keychains().createBackup({ reqId, passphrase: params.passphrase });
                }
            };
            const { userKeychain, backupKeychain, bitgoKeychain } = await (0, utils_1.promiseProps)({
                userKeychain: userKeychainPromise(),
                backupKeychain: backupKeychainPromise(),
                bitgoKeychain: this.baseCoin
                    .keychains()
                    .createBitGo({ enterprise: params.enterprise, reqId, isDistributedCustody: params.isDistributedCustody }),
            });
            walletParams.keys = [userKeychain.id, backupKeychain.id, bitgoKeychain.id];
            const { prv } = userKeychain;
            if (_.isString(prv)) {
                (0, assert_1.default)(backupKeychain.pub);
                (0, assert_1.default)(bitgoKeychain.pub);
                walletParams.keySignatures = {
                    backup: (await this.baseCoin.signMessage({ prv }, backupKeychain.pub)).toString('hex'),
                    bitgo: (await this.baseCoin.signMessage({ prv }, bitgoKeychain.pub)).toString('hex'),
                };
            }
            const keychains = {
                userKeychain,
                backupKeychain,
                bitgoKeychain,
            };
            const finalWalletParams = await this.baseCoin.supplementGenerateWallet(walletParams, keychains);
            if (_.includes(['xrp', 'xlm', 'cspr'], this.baseCoin.getFamily()) && !_.isUndefined(params.rootPrivateKey)) {
                walletParams.rootPrivateKey = params.rootPrivateKey;
            }
            this.bitgo.setRequestTracer(reqId);
            const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(finalWalletParams).result();
            const result = {
                wallet: new wallet_1.Wallet(this.bitgo, this.baseCoin, newWallet),
                userKeychain: userKeychain,
                backupKeychain: backupKeychain,
                bitgoKeychain: bitgoKeychain,
                responseType: 'WalletWithKeychains',
            };
            if (!_.isUndefined(backupKeychain.prv)) {
                result.warning = 'Be sure to backup the backup keychain -- it is not stored anywhere else!';
            }
            if (!_.isUndefined(derivationPath)) {
                userKeychain.derivationPath = derivationPath;
            }
            if (canEncrypt && params.passcodeEncryptionCode) {
                result.encryptedWalletPassphrase = this.bitgo.encrypt({
                    input: passphrase,
                    password: params.passcodeEncryptionCode,
                });
            }
            return result;
        }
    }
    /**
     * List the user's wallet shares
     * @param params
     */
    async listShares(params = {}) {
        return await this.bitgo.get(this.baseCoin.url('/walletshare')).result();
    }
    /**
     * List the user's wallet shares v2
     * @returns {Promise<WalletShares>}
     */
    async listSharesV2() {
        return await this.bitgo.get(this.bitgo.url('/walletshares', 2)).result();
    }
    /**
     * Gets a wallet share information, including the encrypted sharing keychain. requires unlock if keychain is present.
     * @param params
     * @param params.walletShareId - the wallet share to get information on
     */
    async getShare(params = {}) {
        common.validateParams(params, ['walletShareId'], []);
        return await this.bitgo.get(this.baseCoin.url('/walletshare/' + params.walletShareId)).result();
    }
    /**
     * Update a wallet share
     * @param params.walletShareId - the wallet share to update
     * @param params.state - the new state of the wallet share
     * @param params
     */
    async updateShare(params = {}) {
        common.validateParams(params, ['walletShareId'], []);
        return await this.bitgo
            .post(this.baseCoin.url('/walletshare/' + params.walletShareId))
            .send(params)
            .result();
    }
    /**
     * Bulk accept wallet shares
     * @param params AcceptShareOptionsRequest[]
     * @returns {Promise<BulkAcceptShareResponse>}
     */
    async bulkAcceptShareRequest(params) {
        return await this.bulkAcceptShareRequestWithRetry(params);
    }
    async bulkAcceptShareRequestWithRetry(params) {
        // Server has a limit of approximately 1MB for payload size
        let MAX_PAYLOAD_SIZE = 950000; // ~950KB to leave some buffer
        // Function to calculate the size of a payload
        const calculatePayloadSize = (items) => {
            return Buffer.byteLength(JSON.stringify({ keysForWalletShares: items }), 'utf8');
        };
        const results = [];
        const remainingParams = [...params];
        while (remainingParams.length > 0) {
            // Build optimal batch by adding items until we reach size limit
            const batch = [];
            // Start with empty batch
            // Add items one by one while monitoring payload size
            while (remainingParams.length > 0) {
                // Test adding the next item
                const testBatch = [...batch, remainingParams[0]];
                const testSize = calculatePayloadSize(testBatch);
                // If adding this item would exceed the size limit, stop adding
                if (testSize > MAX_PAYLOAD_SIZE && batch.length > 0) {
                    break;
                }
                // Otherwise, add the item to the batch
                batch.push(remainingParams.shift());
            }
            // Handle case where even a single item is too large
            if (batch.length === 0 && remainingParams.length > 0) {
                // Send just the first item even if it's oversized
                batch.push(remainingParams.shift());
            }
            const payloadObj = { keysForWalletShares: batch };
            try {
                const result = await this.bitgo.put(this.bitgo.url('/walletshares/accept', 2)).send(payloadObj).result();
                if (result.acceptedWalletShares && Array.isArray(result.acceptedWalletShares)) {
                    results.push(...result.acceptedWalletShares);
                }
            }
            catch (error) {
                if (error.status === 413 && batch.length > 1) {
                    // If we still get 413 with multiple items, put them back and try with half the batch size
                    remainingParams.unshift(...batch);
                    MAX_PAYLOAD_SIZE = Math.floor(MAX_PAYLOAD_SIZE / 2); // Reduce size limit for next attempt
                    continue;
                }
                throw error;
            }
        }
        return {
            acceptedWalletShares: results,
        };
    }
    async bulkUpdateWalletShareRequest(params) {
        return await this.bitgo
            .put(this.bitgo.url('/walletshares/update', 2))
            .send({
            shares: params,
        })
            .result();
    }
    /**
     * Resend a wallet share invitation email
     * @param params
     * @param params.walletShareId - the wallet share whose invitiation should be resent
     */
    async resendShareInvite(params = {}) {
        common.validateParams(params, ['walletShareId'], []);
        const urlParts = params.walletShareId + '/resendemail';
        return this.bitgo.post(this.baseCoin.url('/walletshare/' + urlParts)).result();
    }
    /**
     * Cancel a wallet share
     * @param params
     * @param params.walletShareId - the wallet share to update
     */
    async cancelShare(params = {}) {
        common.validateParams(params, ['walletShareId'], []);
        return await this.bitgo
            .del(this.baseCoin.url('/walletshare/' + params.walletShareId))
            .send()
            .result();
    }
    /**
     * Re-share wallet with existing spenders of the wallet
     * @param walletId
     * @param userPassword
     */
    async reshareWalletWithSpenders(walletId, userPassword) {
        const wallet = await this.get({ id: walletId });
        if (!wallet?._wallet?.enterprise) {
            throw new Error('Enterprise not found for the wallet');
        }
        const enterpriseUsersResponse = await this.bitgo
            .get(this.bitgo.url(`/enterprise/${wallet?._wallet?.enterprise}/user`))
            .result();
        // create a map of users for easy lookup - we need the user email id to share the wallet
        const usersMap = new Map([...enterpriseUsersResponse?.adminUsers, ...enterpriseUsersResponse?.nonAdminUsers].map((obj) => [obj.id, obj]));
        if (wallet._wallet.users) {
            for (const user of wallet._wallet.users) {
                const userObject = usersMap.get(user.user);
                if (user.permissions.includes('spend') && !user.permissions.includes('admin') && userObject) {
                    const shareParams = {
                        walletId: walletId,
                        user: user.user,
                        permissions: user.permissions.join(','),
                        walletPassphrase: userPassword,
                        email: userObject.email.email,
                        reshare: true,
                        skipKeychain: false,
                    };
                    await wallet.shareWallet(shareParams);
                }
            }
        }
    }
    /**
     * Accepts a wallet share, adding the wallet to the user's list
     * Needs a user's password to decrypt the shared key
     *
     * @param params
     * @param params.walletShareId - the wallet share to accept
     * @param params.userPassword - (required if more a keychain was shared) user's password to decrypt the shared wallet
     * @param params.newWalletPassphrase - new wallet passphrase for saving the shared wallet prv.
     *                                     If left blank and a wallet with more than view permissions was shared,
     *                                     then the user's login password is used.
     * @param params.overrideEncryptedPrv - set only if the prv was received out-of-band.
     */
    async acceptShare(params = {}) {
        common.validateParams(params, ['walletShareId'], ['overrideEncryptedPrv', 'userPassword', 'newWalletPassphrase']);
        let encryptedPrv = params.overrideEncryptedPrv;
        const walletShare = await this.getShare({ walletShareId: params.walletShareId });
        if (walletShare.keychainOverrideRequired &&
            walletShare.permissions.indexOf('admin') !== -1 &&
            walletShare.permissions.indexOf('spend') !== -1) {
            if (_.isUndefined(params.userPassword)) {
                throw new Error('userPassword param must be provided to decrypt shared key');
            }
            const walletKeychain = await this.baseCoin.keychains().createUserKeychain(params.userPassword);
            if (_.isUndefined(walletKeychain.encryptedPrv)) {
                throw new Error('encryptedPrv was not found on wallet keychain');
            }
            const payload = {
                tradingAccountId: walletShare.wallet,
                pubkey: walletKeychain.pub,
                timestamp: new Date().toISOString(),
            };
            const payloadString = JSON.stringify(payload);
            const privateKey = this.bitgo.decrypt({
                password: params.userPassword,
                input: walletKeychain.encryptedPrv,
            });
            const signature = await this.baseCoin.signMessage({ prv: privateKey }, payloadString);
            const response = await this.updateShare({
                walletShareId: params.walletShareId,
                state: 'accepted',
                keyId: walletKeychain.id,
                signature: signature.toString('hex'),
                payload: payloadString,
            });
            // If the wallet share was accepted successfully (changed=true), reshare the wallet with the spenders
            if (response.changed && response.state === 'accepted') {
                try {
                    await this.reshareWalletWithSpenders(walletShare.wallet, params.userPassword);
                }
                catch (e) {
                    // TODO: PX-3826
                    // Do nothing
                }
            }
            return response;
        }
        // Return right away if there is no keychain to decrypt, or if explicit encryptedPrv was provided
        if (!walletShare.keychain || !walletShare.keychain.encryptedPrv || encryptedPrv) {
            return this.updateShare({
                walletShareId: params.walletShareId,
                state: 'accepted',
            });
        }
        // More than viewing was requested, so we need to process the wallet keys using the shared ecdh scheme
        if (_.isUndefined(params.userPassword)) {
            throw new Error('userPassword param must be provided to decrypt shared key');
        }
        const sharingKeychain = (await this.bitgo.getECDHKeychain());
        if (_.isUndefined(sharingKeychain.encryptedXprv)) {
            throw new Error('encryptedXprv was not found on sharing keychain');
        }
        // Now we have the sharing keychain, we can work out the secret used for sharing the wallet with us
        sharingKeychain.prv = this.bitgo.decrypt({
            password: params.userPassword,
            input: sharingKeychain.encryptedXprv,
        });
        const secret = (0, ecdh_1.getSharedSecret)(
        // Derive key by path (which is used between these 2 users only)
        utxo_lib_1.bip32.fromBase58(sharingKeychain.prv).derivePath((0, api_1.sanitizeLegacyPath)(walletShare.keychain.path)), Buffer.from(walletShare.keychain.fromPubKey, 'hex')).toString('hex');
        // Yes! We got the secret successfully here, now decrypt the shared wallet prv
        const decryptedSharedWalletPrv = this.bitgo.decrypt({
            password: secret,
            input: walletShare.keychain.encryptedPrv,
        });
        // We will now re-encrypt the wallet with our own password
        const newWalletPassphrase = params.newWalletPassphrase || params.userPassword;
        encryptedPrv = this.bitgo.encrypt({
            password: newWalletPassphrase,
            input: decryptedSharedWalletPrv,
        });
        const updateParams = {
            walletShareId: params.walletShareId,
            state: 'accepted',
        };
        if (encryptedPrv) {
            updateParams.encryptedPrv = encryptedPrv;
        }
        return this.updateShare(updateParams);
    }
    /**
     * Bulk Accept wallet shares, adding the wallets to the user's list
     * Needs a user's password to decrypt the shared key
     *
     * @param params BulkAcceptShareOptions
     * @param params.walletShareId - array of the wallet shares to accept
     * @param params.userPassword - user's password to decrypt the shared wallet key
     * @param params.newWalletPassphrase - new wallet passphrase for saving the shared wallet prv.
     *                                     If left blank then the user's login password is used.
     *
     *@returns {Promise<BulkAcceptShareResponse>}
     */
    async bulkAcceptShare(params) {
        try {
            common.validateParams(params, ['userLoginPassword'], ['newWalletPassphrase']);
        }
        catch (e) {
            if ('newWalletPassphrase' in params) {
                throw new Error('Please provide a valid wallet passphrase');
            }
            throw new Error('Please provide a valid user login password');
        }
        (0, assert_1.default)(params.walletShareIds.length > 0, 'Please provide at least one wallet share to accept');
        const allWalletShares = await this.listSharesV2();
        const walletShareMap = allWalletShares.incoming.reduce((map, share) => ({ ...map, [share.id]: share }), {});
        const walletShares = params.walletShareIds
            .map((walletShareId) => walletShareMap[walletShareId])
            .filter((walletShare) => walletShare && walletShare.keychain);
        if (!walletShares.length) {
            throw new Error('No valid wallet shares found to accept');
        }
        const sharingKeychain = await this.bitgo.getECDHKeychain();
        if (_.isUndefined(sharingKeychain.encryptedXprv)) {
            throw new Error('encryptedXprv was not found on sharing keychain');
        }
        sharingKeychain.prv = this.bitgo.decrypt({
            password: params.userLoginPassword,
            input: sharingKeychain.encryptedXprv,
        });
        const newWalletPassphrase = params.newWalletPassphrase || params.userLoginPassword;
        const keysForWalletShares = walletShares.flatMap((walletShare) => {
            if (!walletShare.keychain) {
                return [];
            }
            const secret = (0, ecdh_1.getSharedSecret)(utxo_lib_1.bip32.fromBase58(sharingKeychain.prv).derivePath((0, api_1.sanitizeLegacyPath)(walletShare.keychain.path)), Buffer.from(walletShare.keychain.fromPubKey, 'hex')).toString('hex');
            const decryptedSharedWalletPrv = this.bitgo.decrypt({
                password: secret,
                input: walletShare.keychain.encryptedPrv,
            });
            const newEncryptedPrv = this.bitgo.encrypt({
                password: newWalletPassphrase,
                input: decryptedSharedWalletPrv,
            });
            return [
                {
                    walletShareId: walletShare.id,
                    encryptedPrv: newEncryptedPrv,
                },
            ];
        });
        return this.bulkAcceptShareRequest(keysForWalletShares);
    }
    /**
     * Updates multiple wallet shares in bulk
     * This method allows users to accept or reject multiple wallet shares in a single operation.
     * It handles different types of wallet shares including those requiring special keychain overrides
     * and those with encrypted private keys that need to be decrypted and re-encrypted.
     * After processing, it also reshares accepted wallets with spenders for special override cases.
     *
     * @param params - Options for bulk updating wallet shares
     * @param params.shares - Array of wallet shares to update with their status (accept/reject)
     * @param params.userLoginPassword - User's login password for decryption operations
     * @param params.newWalletPassphrase - New wallet passphrase for re-encryption
     * @returns Array of responses for each wallet share update
     */
    async bulkUpdateWalletShare(params) {
        if (!params.shares) {
            throw new Error('Missing parameter: shares');
        }
        if (!Array.isArray(params.shares)) {
            throw new Error('Expecting parameter array: shares but found ' + typeof params.shares);
        }
        // Validate each share in the array
        for (const share of params.shares) {
            if (!share.walletShareId) {
                throw new Error('Missing walletShareId in share');
            }
            if (!share.status) {
                throw new Error('Missing status in share');
            }
            if (share.status !== 'accept' && share.status !== 'reject') {
                throw new Error('Invalid status in share: ' + share.status + '. Must be either "accept" or "reject"');
            }
            if (typeof share.walletShareId !== 'string') {
                throw new Error('Expecting walletShareId to be a string but found ' + typeof share.walletShareId);
            }
        }
        // Validate optional parameters if provided
        if (params.userLoginPassword !== undefined && typeof params.userLoginPassword !== 'string') {
            throw new Error('Expecting parameter string: userLoginPassword but found ' + typeof params.userLoginPassword);
        }
        if (params.newWalletPassphrase !== undefined && typeof params.newWalletPassphrase !== 'string') {
            throw new Error('Expecting parameter string: newWalletPassphrase but found ' + typeof params.newWalletPassphrase);
        }
        (0, assert_1.default)(params.shares.length > 0, 'no shares are passed');
        const { shares: inputShares, userLoginPassword, newWalletPassphrase } = params;
        const allWalletShares = await this.listSharesV2();
        // Only include shares that are in the input array for efficiency
        const shareIds = new Set(inputShares.map((share) => share.walletShareId));
        const walletShareMap = new Map();
        allWalletShares.incoming
            .filter((share) => shareIds.has(share.id))
            .forEach((share) => walletShareMap.set(share.id, share));
        allWalletShares.outgoing
            .filter((share) => shareIds.has(share.id))
            .forEach((share) => walletShareMap.set(share.id, share));
        const resolvedShares = inputShares.map((share) => {
            const walletShare = walletShareMap.get(share.walletShareId);
            if (!walletShare) {
                throw new Error(`invalid wallet share provided: ${share.walletShareId}`);
            }
            return { ...share, walletShare };
        });
        // Identify special override cases that need resharing after acceptance
        const specialOverrideCases = new Map();
        resolvedShares.forEach((share) => {
            if (share.status === 'accept' &&
                share.walletShare.keychainOverrideRequired &&
                share.walletShare.permissions.includes('admin') &&
                share.walletShare.permissions.includes('spend')) {
                specialOverrideCases.set(share.walletShareId, share.walletShare.wallet);
            }
        });
        // Decrypt sharing keychain if needed (only once)
        let sharingKeychainPrv;
        // Only decrypt if there are shares to accept that might need it
        const hasSharesRequiringDecryption = specialOverrideCases.size > 0 ||
            resolvedShares.some((share) => share.status === 'accept' && share.walletShare.keychain?.encryptedPrv);
        if (userLoginPassword && hasSharesRequiringDecryption) {
            const sharingKeychain = await this.bitgo.getECDHKeychain();
            if (!sharingKeychain.encryptedXprv) {
                throw new Error('encryptedXprv was not found on sharing keychain');
            }
            sharingKeychainPrv = this.bitgo.decrypt({
                password: userLoginPassword,
                input: sharingKeychain.encryptedXprv,
            });
        }
        const settledUpdates = await Promise.allSettled(resolvedShares.map(async (share) => {
            const { walletShareId, status, walletShare } = share;
            // Handle accept case
            if (status === 'accept') {
                return this.processAcceptShare(walletShareId, walletShare, userLoginPassword, newWalletPassphrase, sharingKeychainPrv);
            }
            // Handle reject case
            return [
                {
                    walletShareId,
                    status: 'reject',
                },
            ];
        }));
        // Extract successful updates
        const successfulUpdates = settledUpdates.flatMap((result) => (result.status === 'fulfilled' ? result.value : []));
        // Extract failed updates - only from rejected promises
        const failedUpdates = settledUpdates.reduce((acc, result, index) => {
            if (result.status === 'rejected') {
                const rejectedResult = result;
                acc.push({
                    walletShareId: resolvedShares[index].walletShareId,
                    reason: rejectedResult.reason?.message || String(rejectedResult.reason),
                });
            }
            return acc;
        }, []);
        // Send successful updates to the server
        const response = await this.bulkUpdateWalletShareRequest(successfulUpdates);
        // Process accepted special override cases - reshare with spenders
        if (response.acceptedWalletShares && response.acceptedWalletShares.length > 0 && userLoginPassword) {
            // For each accepted wallet share that is a special override case, reshare with spenders
            for (const walletShareId of response.acceptedWalletShares) {
                if (specialOverrideCases.has(walletShareId)) {
                    const walletId = specialOverrideCases.get(walletShareId);
                    try {
                        await this.reshareWalletWithSpenders(walletId, userLoginPassword);
                    }
                    catch (e) {
                        // Log error but continue processing other shares
                        console.error(`Error resharing wallet ${walletId} with spenders: ${e?.message}`);
                    }
                }
            }
        }
        // Add information about failed updates to the response
        if (failedUpdates.length > 0) {
            response.walletShareUpdateErrors.push(...failedUpdates);
        }
        return response;
    }
    /**
     * Process a wallet share that is being accepted
     * This method handles the different cases for accepting a wallet share:
     * 1. Special override case requiring user keychain and signing
     * 2. Simple case with no keychain to decrypt
     * 3. Standard case requiring decryption and re-encryption
     *
     * @param walletShareId - ID of the wallet share
     * @param walletShare - Wallet share object
     * @param userLoginPassword - User's login password
     * @param newWalletPassphrase - New wallet passphrase
     * @param sharingKeychainPrv - Decrypted sharing keychain private key
     * @returns Array of wallet share update requests
     */
    async processAcceptShare(walletShareId, walletShare, userLoginPassword, newWalletPassphrase, sharingKeychainPrv) {
        // Special override case: requires user keychain and signing
        if (walletShare.keychainOverrideRequired &&
            walletShare.permissions.includes('admin') &&
            walletShare.permissions.includes('spend')) {
            if (!userLoginPassword) {
                throw new Error('userLoginPassword param must be provided to decrypt shared key');
            }
            const walletKeychain = await this.baseCoin.keychains().createUserKeychain(userLoginPassword);
            if (!walletKeychain.encryptedPrv) {
                throw new Error('encryptedPrv was not found on wallet keychain');
            }
            const payload = JSON.stringify({
                tradingAccountId: walletShare.wallet,
                pubkey: walletKeychain.pub,
                timestamp: new Date().toISOString(),
            });
            const prv = this.bitgo.decrypt({
                password: userLoginPassword,
                input: walletKeychain.encryptedPrv,
            });
            const signature = await this.baseCoin.signMessage({ prv }, payload);
            return [
                {
                    walletShareId,
                    status: 'accept',
                    keyId: walletKeychain.id,
                    signature: signature.toString('hex'),
                    payload,
                },
            ];
        }
        // Return right away if there is no keychain to decrypt
        if (!walletShare.keychain || !walletShare.keychain.encryptedPrv) {
            return [
                {
                    walletShareId,
                    status: 'accept',
                },
            ];
        }
        // More than viewing was requested, so we need to process the wallet keys using the shared ecdh scheme
        if (!userLoginPassword) {
            throw new Error('userLoginPassword param must be provided to decrypt shared key');
        }
        if (!sharingKeychainPrv) {
            throw new Error('failed to retrieve and decrypt sharing keychain');
        }
        const derivedKey = utxo_lib_1.bip32.fromBase58(sharingKeychainPrv).derivePath((0, api_1.sanitizeLegacyPath)(walletShare.keychain.path));
        const sharedSecret = (0, ecdh_1.getSharedSecret)(derivedKey, Buffer.from(walletShare.keychain.fromPubKey, 'hex')).toString('hex');
        const decryptedPrv = this.bitgo.decrypt({
            password: sharedSecret,
            input: walletShare.keychain.encryptedPrv,
        });
        // We will now re-encrypt the wallet with our own password
        const encryptedPrv = this.bitgo.encrypt({
            password: newWalletPassphrase || userLoginPassword,
            input: decryptedPrv,
        });
        return [
            {
                walletShareId,
                status: 'accept',
                encryptedPrv,
            },
        ];
    }
    /**
     * Get a wallet by its ID
     * @param params
     * @param params.id wallet id
     * @returns {*}
     */
    async getWallet(params = {}) {
        common.validateParams(params, ['id'], []);
        const query = {};
        if (params.allTokens) {
            if (!_.isBoolean(params.allTokens)) {
                throw new Error('invalid allTokens argument, expecting boolean');
            }
            query.allTokens = params.allTokens;
        }
        if (params.includeBalance !== undefined) {
            query.includeBalance = params.includeBalance;
        }
        this.bitgo.setRequestTracer(params.reqId || new utils_1.RequestTracer());
        const wallet = await this.bitgo
            .get(this.baseCoin.url('/wallet/' + params.id))
            .query(query)
            .result();
        return new wallet_1.Wallet(this.bitgo, this.baseCoin, wallet);
    }
    /**
     * Get a wallet by its address
     * @param params
     * @param params.address wallet address
     * @returns {*}
     */
    async getWalletByAddress(params = {}) {
        common.validateParams(params, ['address'], []);
        this.bitgo.setRequestTracer(params.reqId || new utils_1.RequestTracer());
        const wallet = await this.bitgo.get(this.baseCoin.url('/wallet/address/' + params.address)).result();
        return new wallet_1.Wallet(this.bitgo, this.baseCoin, wallet);
    }
    /**
     * For any given supported coin, get total balances for all wallets of that
     * coin type on the account.
     * @param params
     * @returns {*}
     */
    async getTotalBalances(params = {}) {
        return await this.bitgo.get(this.baseCoin.url('/wallet/balances')).result();
    }
    /**
     * Generates a TSS or BLS-DKG Wallet.
     * @param params
     * @private
     */
    async generateMpcWallet({ passphrase, label, multisigType, enterprise, walletVersion, originalPasscodeEncryptionCode, }) {
        if (multisigType === 'tss' && this.baseCoin.getMPCAlgorithm() === 'ecdsa') {
            const tssSettings = await this.bitgo
                .get(this.bitgo.microservicesUrl('/api/v2/tss/settings'))
                .result();
            const multisigTypeVersion = tssSettings.coinSettings[this.baseCoin.getFamily()]?.walletCreationSettings?.multiSigTypeVersion;
            walletVersion = this.determineEcdsaMpcWalletVersion(walletVersion, multisigTypeVersion);
        }
        const reqId = new utils_1.RequestTracer();
        this.bitgo.setRequestTracer(reqId);
        // Create MPC Keychains
        const keychains = await this.baseCoin.keychains().createMpc({
            multisigType,
            passphrase,
            enterprise,
            originalPasscodeEncryptionCode,
        });
        // Create Wallet
        const { userKeychain, backupKeychain, bitgoKeychain } = keychains;
        const walletParams = {
            label,
            m: 2,
            n: 3,
            keys: [userKeychain.id, backupKeychain.id, bitgoKeychain.id],
            type: 'hot',
            multisigType,
            enterprise,
            walletVersion,
        };
        const finalWalletParams = await this.baseCoin.supplementGenerateWallet(walletParams, keychains);
        const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(finalWalletParams).result();
        const result = {
            wallet: new wallet_1.Wallet(this.bitgo, this.baseCoin, newWallet),
            userKeychain,
            backupKeychain,
            bitgoKeychain,
            responseType: 'WalletWithKeychains',
        };
        if (!_.isUndefined(backupKeychain.prv)) {
            result.warning = 'Be sure to backup the backup keychain -- it is not stored anywhere else!';
        }
        return result;
    }
    /**
     * Generates a Self-Managed Cold TSS Wallet.
     * @param params
     * @private
     */
    async generateSMCMpcWallet({ label, multisigType, enterprise, walletVersion, bitgoKeyId, commonKeychain, coldDerivationSeed, }) {
        const reqId = new utils_1.RequestTracer();
        this.bitgo.setRequestTracer(reqId);
        let multisigTypeVersion;
        if (multisigType === 'tss' && this.baseCoin.getMPCAlgorithm() === 'ecdsa') {
            const tssSettings = await this.bitgo
                .get(this.bitgo.microservicesUrl('/api/v2/tss/settings'))
                .result();
            multisigTypeVersion =
                tssSettings.coinSettings[this.baseCoin.getFamily()]?.walletCreationSettings?.coldMultiSigTypeVersion;
            walletVersion = this.determineEcdsaMpcWalletVersion(walletVersion, multisigTypeVersion);
        }
        // Create MPC Keychains
        const bitgoKeychain = await this.baseCoin.keychains().get({ id: bitgoKeyId });
        if (!bitgoKeychain || !bitgoKeychain.commonKeychain) {
            throw new Error('BitGo keychain not found');
        }
        if (bitgoKeychain.source !== 'bitgo') {
            throw new Error('The provided bitgoKeyId is not a BitGo keychain');
        }
        if (bitgoKeychain.commonKeychain !== commonKeychain) {
            throw new Error('The provided Common keychain mismatch with the provided Bitgo key');
        }
        if (!coldDerivationSeed) {
            throw new Error('derivedFromParentWithSeed is required');
        }
        const userKeychainParams = {
            source: 'user',
            keyType: 'tss',
            commonKeychain: commonKeychain,
            derivedFromParentWithSeed: coldDerivationSeed,
            isMPCv2: multisigTypeVersion === 'MPCv2' ? true : undefined,
        };
        const userKeychain = await this.baseCoin.keychains().add(userKeychainParams);
        const backupKeyChainParams = {
            source: 'backup',
            keyType: 'tss',
            commonKeychain: commonKeychain,
            derivedFromParentWithSeed: coldDerivationSeed,
            isMPCv2: multisigTypeVersion === 'MPCv2' ? true : undefined,
        };
        const backupKeychain = await this.baseCoin.keychains().add(backupKeyChainParams);
        // Create Wallet
        const keychains = { userKeychain, backupKeychain, bitgoKeychain };
        const walletParams = {
            label,
            m: 2,
            n: 3,
            keys: [userKeychain.id, backupKeychain.id, bitgoKeychain.id],
            type: 'cold',
            multisigType,
            enterprise,
            walletVersion,
        };
        const finalWalletParams = await this.baseCoin.supplementGenerateWallet(walletParams, keychains);
        const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(finalWalletParams).result();
        const result = {
            wallet: new wallet_1.Wallet(this.bitgo, this.baseCoin, newWallet),
            userKeychain,
            backupKeychain,
            bitgoKeychain,
            responseType: 'WalletWithKeychains',
        };
        return result;
    }
    /**
     * Generates a Custodial TSS Wallet.
     * @param params
     * @private
     */
    async generateCustodialMpcWallet({ label, multisigType, enterprise, walletVersion, }) {
        const reqId = new utils_1.RequestTracer();
        this.bitgo.setRequestTracer(reqId);
        if (multisigType === 'tss' && this.baseCoin.getMPCAlgorithm() === 'ecdsa') {
            const tssSettings = await this.bitgo
                .get(this.bitgo.microservicesUrl('/api/v2/tss/settings'))
                .result();
            const multisigTypeVersion = tssSettings.coinSettings[this.baseCoin.getFamily()]?.walletCreationSettings?.custodialMultiSigTypeVersion;
            walletVersion = this.determineEcdsaMpcWalletVersion(walletVersion, multisigTypeVersion);
        }
        const finalWalletParams = {
            label,
            multisigType,
            enterprise,
            walletVersion,
            type: 'custodial',
        };
        // Create Wallet
        const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(finalWalletParams).result();
        const wallet = new wallet_1.Wallet(this.bitgo, this.baseCoin, newWallet);
        const keychains = wallet.keyIds();
        const result = {
            wallet,
            userKeychain: { id: keychains[0], type: multisigType, source: 'user' },
            backupKeychain: { id: keychains[1], type: multisigType, source: 'backup' },
            bitgoKeychain: { id: keychains[2], type: multisigType, source: 'bitgo' },
            responseType: 'WalletWithKeychains',
        };
        return result;
    }
    determineEcdsaMpcWalletVersion(walletVersion, multisigTypeVersion) {
        if (this.baseCoin.isEVM() && multisigTypeVersion === 'MPCv2') {
            if (!walletVersion || (walletVersion !== 5 && walletVersion !== 6)) {
                return 5;
            }
        }
        return walletVersion;
    }
}
exports.Wallets = Wallets;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"wallets.js","sourceRoot":"","sources":["../../../../src/bitgo/wallet/wallets.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport assert from 'assert';\nimport { BigNumber } from 'bignumber.js';\nimport { bip32 } from '@bitgo/utxo-lib';\nimport * as _ from 'lodash';\nimport { CoinFeature } from '@bitgo/statics';\n\nimport { sanitizeLegacyPath } from '../../api';\nimport * as common from '../../common';\nimport { IBaseCoin, KeychainsTriplet, SupplementGenerateWalletOptions } from '../baseCoin';\nimport { BitGoBase } from '../bitgoBase';\nimport { getSharedSecret } from '../ecdh';\nimport { AddKeychainOptions, Keychain, KeyIndices } from '../keychain';\nimport { decodeOrElse, promiseProps, RequestTracer } from '../utils';\nimport {\n  AcceptShareOptions,\n  AcceptShareOptionsRequest,\n  AddWalletOptions,\n  BulkAcceptShareOptions,\n  BulkAcceptShareResponse,\n  BulkUpdateWalletShareOptions,\n  BulkUpdateWalletShareOptionsRequest,\n  BulkUpdateWalletShareResponse,\n  GenerateBaseMpcWalletOptions,\n  GenerateLightningWalletOptions,\n  GenerateLightningWalletOptionsCodec,\n  GenerateMpcWalletOptions,\n  GenerateSMCMpcWalletOptions,\n  GenerateWalletOptions,\n  GetWalletByAddressOptions,\n  GetWalletOptions,\n  IWallets,\n  LightningWalletWithKeychains,\n  ListWalletOptions,\n  UpdateShareOptions,\n  WalletShares,\n  WalletWithKeychains,\n} from './iWallets';\nimport { WalletShare } from './iWallet';\nimport { Wallet } from './wallet';\nimport { TssSettings } from '@bitgo/public-types';\n\n/**\n * Check if a wallet is a WalletWithKeychains\n */\nexport function isWalletWithKeychains(\n  wallet: WalletWithKeychains | LightningWalletWithKeychains\n): wallet is WalletWithKeychains {\n  return wallet.responseType === 'WalletWithKeychains';\n}\n\nexport class Wallets implements IWallets {\n  private readonly bitgo: BitGoBase;\n  private readonly baseCoin: IBaseCoin;\n\n  constructor(bitgo: BitGoBase, baseCoin: IBaseCoin) {\n    this.bitgo = bitgo;\n    this.baseCoin = baseCoin;\n  }\n\n  /**\n   * Get a wallet by ID (proxy for getWallet)\n   * @param params\n   */\n  async get(params: GetWalletOptions = {}): Promise<Wallet> {\n    return this.getWallet(params);\n  }\n\n  /**\n   * List a user's wallets\n   * @param params\n   * @returns {*}\n   */\n  async list(params: ListWalletOptions & { enterprise?: string } = {}): Promise<{ wallets: Wallet[] }> {\n    if (params.skip && params.prevId) {\n      throw new Error('cannot specify both skip and prevId');\n    }\n    const body = (await this.bitgo.get(this.baseCoin.url('/wallet')).query(params).result()) as any;\n    body.wallets = body.wallets.map((w) => new Wallet(this.bitgo, this.baseCoin, w));\n    return body;\n  }\n\n  /**\n   * add\n   * Add a new wallet (advanced mode).\n   * This allows you to manually submit the keys, type, m and n of the wallet\n   * Parameters include:\n   *    \"label\": label of the wallet to be shown in UI\n   *    \"m\": number of keys required to unlock wallet (2)\n   *    \"n\": number of keys available on the wallet (3)\n   *    \"keys\": array of keychain ids\n   */\n  async add(params: AddWalletOptions): Promise<any> {\n    params = params || {};\n\n    common.validateParams(params, [], ['label', 'enterprise', 'type']);\n\n    if (typeof params.label !== 'string') {\n      throw new Error('missing required string parameter label');\n    }\n\n    // no need to pass keys for (single) custodial wallets\n    if (params.type !== 'custodial') {\n      if (Array.isArray(params.keys) === false || !_.isNumber(params.m) || !_.isNumber(params.n)) {\n        throw new Error('invalid argument');\n      }\n\n      // TODO: support more types of multisig\n      if (!this.baseCoin.isValidMofNSetup(params)) {\n        throw new Error('unsupported multi-sig type');\n      }\n    }\n\n    if (params.gasPrice && !_.isNumber(params.gasPrice)) {\n      throw new Error('invalid argument for gasPrice - number expected');\n    }\n\n    if (params.walletVersion) {\n      if (!_.isNumber(params.walletVersion)) {\n        throw new Error('invalid argument for walletVersion - number expected');\n      }\n      if (params.multisigType === 'tss' && this.baseCoin.getMPCAlgorithm() === 'ecdsa' && params.walletVersion === 3) {\n        const tssSettings: TssSettings = await this.bitgo\n          .get(this.bitgo.microservicesUrl('/api/v2/tss/settings'))\n          .result();\n        const multisigTypeVersion =\n          tssSettings.coinSettings[this.baseCoin.getFamily()]?.walletCreationSettings?.multiSigTypeVersion;\n        if (multisigTypeVersion === 'MPCv2') {\n          params.walletVersion = 5;\n        }\n      }\n    }\n\n    if (params.tags && Array.isArray(params.tags) === false) {\n      throw new Error('invalid argument for tags - array expected');\n    }\n\n    if (params.clientFlags && Array.isArray(params.clientFlags) === false) {\n      throw new Error('invalid argument for clientFlags - array expected');\n    }\n\n    if (params.isCold && !_.isBoolean(params.isCold)) {\n      throw new Error('invalid argument for isCold - boolean expected');\n    }\n\n    if (params.isCustodial && !_.isBoolean(params.isCustodial)) {\n      throw new Error('invalid argument for isCustodial - boolean expected');\n    }\n\n    if (params.address && (!_.isString(params.address) || !this.baseCoin.isValidAddress(params.address))) {\n      throw new Error('invalid argument for address - valid address string expected');\n    }\n\n    const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(params).result();\n    return {\n      wallet: new Wallet(this.bitgo, this.baseCoin, newWallet),\n    };\n  }\n\n  private async generateLightningWallet(params: GenerateLightningWalletOptions): Promise<LightningWalletWithKeychains> {\n    const reqId = new RequestTracer();\n    this.bitgo.setRequestTracer(reqId);\n\n    const { label, passphrase, enterprise, passcodeEncryptionCode, subType } = params;\n\n    // TODO BTC-1899: only userAuth key is required for custodial lightning wallet. all 3 keys are required for self custodial lightning.\n    // to avoid changing the platform for custodial flow, let us all 3 keys both wallet types.\n    const keychainPromises = ([undefined, 'userAuth', 'nodeAuth'] as const).map((purpose) => {\n      return async (): Promise<Keychain> => {\n        const keychain = this.baseCoin.keychains().create();\n        const keychainParams: AddKeychainOptions = {\n          pub: keychain.pub,\n          encryptedPrv: this.bitgo.encrypt({ password: passphrase, input: keychain.prv }),\n          originalPasscodeEncryptionCode: purpose === undefined ? passcodeEncryptionCode : undefined,\n          coinSpecific: purpose === undefined ? undefined : { [this.baseCoin.getChain()]: { purpose } },\n          keyType: 'independent',\n          source: 'user',\n        };\n        return await this.baseCoin.keychains().add(keychainParams);\n      };\n    });\n\n    const { userKeychain, userAuthKeychain, nodeAuthKeychain } = await promiseProps({\n      userKeychain: keychainPromises[0](),\n      userAuthKeychain: keychainPromises[1](),\n      nodeAuthKeychain: keychainPromises[2](),\n    });\n\n    const walletParams: SupplementGenerateWalletOptions = {\n      label,\n      m: 1,\n      n: 1,\n      type: 'hot',\n      subType,\n      enterprise,\n      keys: [userKeychain.id],\n      coinSpecific: { [this.baseCoin.getChain()]: { keys: [userAuthKeychain.id, nodeAuthKeychain.id] } },\n    };\n\n    const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(walletParams).result();\n    const wallet = new Wallet(this.bitgo, this.baseCoin, newWallet);\n    return {\n      wallet,\n      userKeychain,\n      userAuthKeychain,\n      nodeAuthKeychain,\n      responseType: 'LightningWalletWithKeychains',\n    };\n  }\n\n  /**\n   * Generate a new wallet\n   * 1. Creates the user keychain locally on the client, and encrypts it with the provided passphrase\n   * 2. If no pub was provided, creates the backup keychain locally on the client, and encrypts it with the provided passphrase\n   * 3. Uploads the encrypted user and backup keychains to BitGo\n   * 4. Creates the BitGo key on the service\n   * 5. Creates the wallet on BitGo with the 3 public keys above\n   * @param params\n   * @param params.label Label for the wallet\n   * @param params.passphrase Passphrase to be used to encrypt the user and backup keychains\n   * @param params.userKey User xpub\n   * @param params.backupXpub Backup xpub\n   * @param params.backupXpubProvider\n   * @param params.enterprise the enterpriseId\n   * @param params.disableTransactionNotifications\n   * @param params.passcodeEncryptionCode optional this is a recovery code that can be used to decrypt the original passphrase in a recovery case.\n   *                                      The user must generate and keep the encrypted original passphrase safe while this code is stored on BitGo\n   * @param params.coldDerivationSeed optional seed for SMC wallets\n   * @param params.gasPrice\n   * @param params.disableKRSEmail\n   * @param params.walletVersion\n   * @param params.multisigType optional multisig type, 'onchain' or 'tss' or 'blsdkg'; if absent, we will defer to the coin's default type\n   * @param params.isDistributedCustody optional parameter for creating bitgo key. This is only necessary if you want to create\n   *                                    a distributed custody wallet. If provided, you must have the enterprise license and pass in\n   *                                    `params.enterprise` into `generateWallet` as well.\n   * @param params.type optional wallet type, 'hot' or 'cold' or 'custodial'; if absent, we will defer to 'hot'\n   * @param params.bitgoKeyId optional bitgo key id for SMC TSS wallets\n   * @param params.commonKeychain optional common keychain for SMC TSS wallets\n   *\n   * @returns {*}\n   */\n  async generateWallet(\n    params: GenerateWalletOptions = {}\n  ): Promise<WalletWithKeychains | LightningWalletWithKeychains> {\n    // Assign the default multiSig type value based on the coin\n    if (!params.multisigType) {\n      params.multisigType = this.baseCoin.getDefaultMultisigType();\n    }\n\n    if (this.baseCoin.getFamily() === 'lnbtc') {\n      const options = decodeOrElse(\n        GenerateLightningWalletOptionsCodec.name,\n        GenerateLightningWalletOptionsCodec,\n        params,\n        (errors) => {\n          throw new Error(`error(s) parsing generate lightning wallet request params: ${errors}`);\n        }\n      );\n\n      const walletData = await this.generateLightningWallet(options);\n      walletData.encryptedWalletPassphrase = this.bitgo.encrypt({\n        input: options.passphrase,\n        password: options.passcodeEncryptionCode,\n      });\n      return walletData;\n    }\n\n    common.validateParams(params, ['label'], ['passphrase', 'userKey', 'backupXpub']);\n    if (typeof params.label !== 'string') {\n      throw new Error('missing required string parameter label');\n    }\n\n    const { type = 'hot', label, passphrase, enterprise, isDistributedCustody } = params;\n    const isTss = params.multisigType === 'tss' && this.baseCoin.supportsTss();\n    const canEncrypt = !!passphrase && typeof passphrase === 'string';\n\n    const walletParams: SupplementGenerateWalletOptions = {\n      label: label,\n      m: 2,\n      n: 3,\n      keys: [],\n      type: !!params.userKey && params.multisigType !== 'onchain' ? 'cold' : type,\n    };\n\n    if (!_.isUndefined(params.passcodeEncryptionCode)) {\n      if (!_.isString(params.passcodeEncryptionCode)) {\n        throw new Error('passcodeEncryptionCode must be a string');\n      }\n    }\n\n    if (!_.isUndefined(enterprise)) {\n      if (!_.isString(enterprise)) {\n        throw new Error('invalid enterprise argument, expecting string');\n      }\n      walletParams.enterprise = enterprise;\n    }\n\n    // EVM TSS wallets must use wallet version 3, 5 and 6\n    if (\n      isTss &&\n      this.baseCoin.isEVM() &&\n      !(params.walletVersion === 3 || params.walletVersion === 5 || params.walletVersion === 6)\n    ) {\n      throw new Error('EVM TSS wallets are only supported for wallet version 3, 5 and 6');\n    }\n\n    if (isTss) {\n      if (!this.baseCoin.supportsTss()) {\n        throw new Error(`coin ${this.baseCoin.getFamily()} does not support TSS at this time`);\n      }\n      if (\n        (params.walletVersion === 5 || params.walletVersion === 6) &&\n        !this.baseCoin.getConfig().features.includes(CoinFeature.MPCV2)\n      ) {\n        throw new Error(`coin ${this.baseCoin.getFamily()} does not support TSS MPCv2 at this time`);\n      }\n      assert(enterprise, 'enterprise is required for TSS wallet');\n\n      if (type === 'cold') {\n        // validate\n        assert(params.bitgoKeyId, 'bitgoKeyId is required for SMC TSS wallet');\n        assert(params.commonKeychain, 'commonKeychain is required for SMC TSS wallet');\n        return this.generateSMCMpcWallet({\n          multisigType: 'tss',\n          label,\n          enterprise,\n          walletVersion: params.walletVersion,\n          bitgoKeyId: params.bitgoKeyId,\n          commonKeychain: params.commonKeychain,\n          coldDerivationSeed: params.coldDerivationSeed,\n        });\n      }\n\n      if (type === 'custodial') {\n        return this.generateCustodialMpcWallet({\n          multisigType: 'tss',\n          label,\n          enterprise,\n          walletVersion: params.walletVersion,\n        });\n      }\n\n      assert(passphrase, 'cannot generate TSS keys without passphrase');\n      const walletData = await this.generateMpcWallet({\n        multisigType: 'tss',\n        label,\n        passphrase,\n        originalPasscodeEncryptionCode: params.passcodeEncryptionCode,\n        enterprise,\n        walletVersion: params.walletVersion,\n      });\n      if (params.passcodeEncryptionCode) {\n        walletData.encryptedWalletPassphrase = this.bitgo.encrypt({\n          input: passphrase,\n          password: params.passcodeEncryptionCode,\n        });\n      }\n      return walletData;\n    }\n\n    // Handle distributed custody\n    if (isDistributedCustody) {\n      if (!enterprise) {\n        throw new Error('must provide enterprise when creating distributed custody wallet');\n      }\n      if (!type || type !== 'cold') {\n        throw new Error('distributed custody wallets must be type: cold');\n      }\n    }\n\n    const hasBackupXpub = !!params.backupXpub;\n    const hasBackupXpubProvider = !!params.backupXpubProvider;\n    if (hasBackupXpub && hasBackupXpubProvider) {\n      throw new Error('Cannot provide more than one backupXpub or backupXpubProvider flag');\n    }\n\n    if (params.gasPrice && params.eip1559) {\n      throw new Error('can not use both eip1559 and gasPrice values');\n    }\n\n    if (!_.isUndefined(params.disableTransactionNotifications)) {\n      if (!_.isBoolean(params.disableTransactionNotifications)) {\n        throw new Error('invalid disableTransactionNotifications argument, expecting boolean');\n      }\n      walletParams.disableTransactionNotifications = params.disableTransactionNotifications;\n    }\n\n    if (!_.isUndefined(params.gasPrice)) {\n      const gasPriceBN = new BigNumber(params.gasPrice);\n      if (gasPriceBN.isNaN()) {\n        throw new Error('invalid gas price argument, expecting number or number as string');\n      }\n      walletParams.gasPrice = gasPriceBN.toString();\n    }\n\n    if (!_.isUndefined(params.eip1559) && !_.isEmpty(params.eip1559)) {\n      const maxFeePerGasBN = new BigNumber(params.eip1559.maxFeePerGas);\n      if (maxFeePerGasBN.isNaN()) {\n        throw new Error('invalid max fee argument, expecting number or number as string');\n      }\n      const maxPriorityFeePerGasBN = new BigNumber(params.eip1559.maxPriorityFeePerGas);\n      if (maxPriorityFeePerGasBN.isNaN()) {\n        throw new Error('invalid priority fee argument, expecting number or number as string');\n      }\n      walletParams.eip1559 = {\n        maxFeePerGas: maxFeePerGasBN.toString(),\n        maxPriorityFeePerGas: maxPriorityFeePerGasBN.toString(),\n      };\n    }\n\n    if (!_.isUndefined(params.disableKRSEmail)) {\n      if (!_.isBoolean(params.disableKRSEmail)) {\n        throw new Error('invalid disableKRSEmail argument, expecting boolean');\n      }\n      walletParams.disableKRSEmail = params.disableKRSEmail;\n    }\n\n    if (!_.isUndefined(params.walletVersion)) {\n      if (!_.isNumber(params.walletVersion)) {\n        throw new Error('invalid walletVersion provided, expecting number');\n      }\n      walletParams.walletVersion = params.walletVersion;\n    }\n\n    // Ensure each krsSpecific param is either a string, boolean, or number\n    const { krsSpecific } = params;\n    if (!_.isUndefined(krsSpecific)) {\n      Object.keys(krsSpecific).forEach((key) => {\n        const val = krsSpecific[key];\n        if (!_.isBoolean(val) && !_.isString(val) && !_.isNumber(val)) {\n          throw new Error('krsSpecific object contains illegal values. values must be strings, booleans, or numbers');\n        }\n      });\n    }\n\n    let derivationPath: string | undefined = undefined;\n\n    const reqId = new RequestTracer();\n\n    if (params.type === 'custodial' && (params.multisigType ?? 'onchain') === 'onchain') {\n      // for custodial multisig, when the wallet is created on the platfor side, the keys are not needed\n      walletParams.n = undefined;\n      walletParams.m = undefined;\n      walletParams.keys = undefined;\n      walletParams.keySignatures = undefined;\n\n      const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(walletParams).result(); // returns the ids\n\n      const userKeychain = this.baseCoin.keychains().get({ id: newWallet.keys[KeyIndices.USER], reqId });\n      const backupKeychain = this.baseCoin.keychains().get({ id: newWallet.keys[KeyIndices.BACKUP], reqId });\n      const bitgoKeychain = this.baseCoin.keychains().get({ id: newWallet.keys[KeyIndices.BITGO], reqId });\n\n      const [userKey, bitgoKey, backupKey] = await Promise.all([userKeychain, bitgoKeychain, backupKeychain]);\n\n      const result: WalletWithKeychains = {\n        wallet: new Wallet(this.bitgo, this.baseCoin, newWallet),\n        userKeychain: userKey,\n        backupKeychain: bitgoKey,\n        bitgoKeychain: backupKey,\n        responseType: 'WalletWithKeychains',\n      };\n\n      return result;\n    } else {\n      const userKeychainPromise = async (): Promise<Keychain> => {\n        let userKeychainParams;\n        let userKeychain;\n        // User provided user key\n        if (params.userKey) {\n          userKeychain = { pub: params.userKey };\n          userKeychainParams = userKeychain;\n          if (params.coldDerivationSeed) {\n            // the derivation only makes sense when a key already exists\n            const derivation = this.baseCoin.deriveKeyWithSeed({\n              key: params.userKey,\n              seed: params.coldDerivationSeed,\n            });\n            derivationPath = derivation.derivationPath;\n            userKeychain.pub = derivation.key;\n            userKeychain.derivedFromParentWithSeed = params.coldDerivationSeed;\n          }\n        } else {\n          if (!canEncrypt) {\n            throw new Error('cannot generate user keypair without passphrase');\n          }\n          // Create the user key.\n          userKeychain = this.baseCoin.keychains().create();\n          userKeychain.encryptedPrv = this.bitgo.encrypt({ password: passphrase, input: userKeychain.prv });\n          userKeychainParams = {\n            pub: userKeychain.pub,\n            encryptedPrv: userKeychain.encryptedPrv,\n            originalPasscodeEncryptionCode: params.passcodeEncryptionCode,\n          };\n        }\n\n        userKeychainParams.reqId = reqId;\n        const newUserKeychain = await this.baseCoin.keychains().add(userKeychainParams);\n        return _.extend({}, newUserKeychain, userKeychain);\n      };\n\n      const backupKeychainPromise = async (): Promise<Keychain> => {\n        if (params.backupXpubProvider) {\n          // If requested, use a KRS or backup key provider\n          return this.baseCoin.keychains().createBackup({\n            provider: params.backupXpubProvider || 'defaultRMGBackupProvider',\n            disableKRSEmail: params.disableKRSEmail,\n            krsSpecific: params.krsSpecific,\n            type: this.baseCoin.getChain(),\n            passphrase: params.passphrase,\n            reqId,\n          });\n        }\n\n        // User provided backup xpub\n        if (params.backupXpub) {\n          // user provided backup ethereum address\n          return this.baseCoin.keychains().add({\n            pub: params.backupXpub,\n            source: 'backup',\n            reqId,\n          });\n        } else {\n          if (!canEncrypt) {\n            throw new Error('cannot generate backup keypair without passphrase');\n          }\n          // No provided backup xpub or address, so default to creating one here\n          return this.baseCoin.keychains().createBackup({ reqId, passphrase: params.passphrase });\n        }\n      };\n      const { userKeychain, backupKeychain, bitgoKeychain }: KeychainsTriplet = await promiseProps({\n        userKeychain: userKeychainPromise(),\n        backupKeychain: backupKeychainPromise(),\n        bitgoKeychain: this.baseCoin\n          .keychains()\n          .createBitGo({ enterprise: params.enterprise, reqId, isDistributedCustody: params.isDistributedCustody }),\n      });\n\n      walletParams.keys = [userKeychain.id, backupKeychain.id, bitgoKeychain.id];\n\n      const { prv } = userKeychain;\n      if (_.isString(prv)) {\n        assert(backupKeychain.pub);\n        assert(bitgoKeychain.pub);\n        walletParams.keySignatures = {\n          backup: (await this.baseCoin.signMessage({ prv }, backupKeychain.pub)).toString('hex'),\n          bitgo: (await this.baseCoin.signMessage({ prv }, bitgoKeychain.pub)).toString('hex'),\n        };\n      }\n\n      const keychains = {\n        userKeychain,\n        backupKeychain,\n        bitgoKeychain,\n      };\n\n      const finalWalletParams = await this.baseCoin.supplementGenerateWallet(walletParams, keychains);\n\n      if (_.includes(['xrp', 'xlm', 'cspr'], this.baseCoin.getFamily()) && !_.isUndefined(params.rootPrivateKey)) {\n        walletParams.rootPrivateKey = params.rootPrivateKey;\n      }\n\n      this.bitgo.setRequestTracer(reqId);\n      const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(finalWalletParams).result();\n\n      const result: WalletWithKeychains = {\n        wallet: new Wallet(this.bitgo, this.baseCoin, newWallet),\n        userKeychain: userKeychain,\n        backupKeychain: backupKeychain,\n        bitgoKeychain: bitgoKeychain,\n        responseType: 'WalletWithKeychains',\n      };\n\n      if (!_.isUndefined(backupKeychain.prv)) {\n        result.warning = 'Be sure to backup the backup keychain -- it is not stored anywhere else!';\n      }\n\n      if (!_.isUndefined(derivationPath)) {\n        userKeychain.derivationPath = derivationPath;\n      }\n\n      if (canEncrypt && params.passcodeEncryptionCode) {\n        result.encryptedWalletPassphrase = this.bitgo.encrypt({\n          input: passphrase,\n          password: params.passcodeEncryptionCode,\n        });\n      }\n\n      return result;\n    }\n  }\n\n  /**\n   * List the user's wallet shares\n   * @param params\n   */\n  async listShares(params: Record<string, unknown> = {}): Promise<any> {\n    return await this.bitgo.get(this.baseCoin.url('/walletshare')).result();\n  }\n\n  /**\n   * List the user's wallet shares v2\n   * @returns {Promise<WalletShares>}\n   */\n  async listSharesV2(): Promise<WalletShares> {\n    return await this.bitgo.get(this.bitgo.url('/walletshares', 2)).result();\n  }\n\n  /**\n   * Gets a wallet share information, including the encrypted sharing keychain. requires unlock if keychain is present.\n   * @param params\n   * @param params.walletShareId - the wallet share to get information on\n   */\n  async getShare(params: { walletShareId?: string } = {}): Promise<any> {\n    common.validateParams(params, ['walletShareId'], []);\n\n    return await this.bitgo.get(this.baseCoin.url('/walletshare/' + params.walletShareId)).result();\n  }\n\n  /**\n   * Update a wallet share\n   * @param params.walletShareId - the wallet share to update\n   * @param params.state - the new state of the wallet share\n   * @param params\n   */\n  async updateShare(params: UpdateShareOptions = {}): Promise<any> {\n    common.validateParams(params, ['walletShareId'], []);\n\n    return await this.bitgo\n      .post(this.baseCoin.url('/walletshare/' + params.walletShareId))\n      .send(params)\n      .result();\n  }\n\n  /**\n   * Bulk accept wallet shares\n   * @param params AcceptShareOptionsRequest[]\n   * @returns {Promise<BulkAcceptShareResponse>}\n   */\n  async bulkAcceptShareRequest(params: AcceptShareOptionsRequest[]): Promise<BulkAcceptShareResponse> {\n    return await this.bulkAcceptShareRequestWithRetry(params);\n  }\n\n  private async bulkAcceptShareRequestWithRetry(params: AcceptShareOptionsRequest[]): Promise<BulkAcceptShareResponse> {\n    // Server has a limit of approximately 1MB for payload size\n    let MAX_PAYLOAD_SIZE = 950000; // ~950KB to leave some buffer\n\n    // Function to calculate the size of a payload\n    const calculatePayloadSize = (items: AcceptShareOptionsRequest[]): number => {\n      return Buffer.byteLength(JSON.stringify({ keysForWalletShares: items }), 'utf8');\n    };\n\n    const results: any[] = [];\n    const remainingParams = [...params];\n\n    while (remainingParams.length > 0) {\n      // Build optimal batch by adding items until we reach size limit\n      const batch: AcceptShareOptionsRequest[] = [];\n      // Start with empty batch\n\n      // Add items one by one while monitoring payload size\n      while (remainingParams.length > 0) {\n        // Test adding the next item\n        const testBatch = [...batch, remainingParams[0]];\n        const testSize = calculatePayloadSize(testBatch);\n\n        // If adding this item would exceed the size limit, stop adding\n        if (testSize > MAX_PAYLOAD_SIZE && batch.length > 0) {\n          break;\n        }\n\n        // Otherwise, add the item to the batch\n        batch.push(remainingParams.shift()!);\n      }\n\n      // Handle case where even a single item is too large\n      if (batch.length === 0 && remainingParams.length > 0) {\n        // Send just the first item even if it's oversized\n        batch.push(remainingParams.shift()!);\n      }\n\n      const payloadObj = { keysForWalletShares: batch };\n\n      try {\n        const result = await this.bitgo.put(this.bitgo.url('/walletshares/accept', 2)).send(payloadObj).result();\n\n        if (result.acceptedWalletShares && Array.isArray(result.acceptedWalletShares)) {\n          results.push(...result.acceptedWalletShares);\n        }\n      } catch (error: any) {\n        if (error.status === 413 && batch.length > 1) {\n          // If we still get 413 with multiple items, put them back and try with half the batch size\n          remainingParams.unshift(...batch);\n          MAX_PAYLOAD_SIZE = Math.floor(MAX_PAYLOAD_SIZE / 2); // Reduce size limit for next attempt\n          continue;\n        }\n        throw error;\n      }\n    }\n\n    return {\n      acceptedWalletShares: results,\n    };\n  }\n\n  async bulkUpdateWalletShareRequest(\n    params: BulkUpdateWalletShareOptionsRequest[]\n  ): Promise<BulkUpdateWalletShareResponse> {\n    return await this.bitgo\n      .put(this.bitgo.url('/walletshares/update', 2))\n      .send({\n        shares: params,\n      })\n      .result();\n  }\n\n  /**\n   * Resend a wallet share invitation email\n   * @param params\n   * @param params.walletShareId - the wallet share whose invitiation should be resent\n   */\n  async resendShareInvite(params: { walletShareId?: string } = {}): Promise<any> {\n    common.validateParams(params, ['walletShareId'], []);\n\n    const urlParts = params.walletShareId + '/resendemail';\n    return this.bitgo.post(this.baseCoin.url('/walletshare/' + urlParts)).result();\n  }\n\n  /**\n   * Cancel a wallet share\n   * @param params\n   * @param params.walletShareId - the wallet share to update\n   */\n  async cancelShare(params: { walletShareId?: string } = {}): Promise<any> {\n    common.validateParams(params, ['walletShareId'], []);\n\n    return await this.bitgo\n      .del(this.baseCoin.url('/walletshare/' + params.walletShareId))\n      .send()\n      .result();\n  }\n\n  /**\n   * Re-share wallet with existing spenders of the wallet\n   * @param walletId\n   * @param userPassword\n   */\n  async reshareWalletWithSpenders(walletId: string, userPassword: string): Promise<void> {\n    const wallet = await this.get({ id: walletId });\n    if (!wallet?._wallet?.enterprise) {\n      throw new Error('Enterprise not found for the wallet');\n    }\n\n    const enterpriseUsersResponse = await this.bitgo\n      .get(this.bitgo.url(`/enterprise/${wallet?._wallet?.enterprise}/user`))\n      .result();\n    // create a map of users for easy lookup - we need the user email id to share the wallet\n    const usersMap = new Map(\n      [...enterpriseUsersResponse?.adminUsers, ...enterpriseUsersResponse?.nonAdminUsers].map((obj) => [obj.id, obj])\n    );\n\n    if (wallet._wallet.users) {\n      for (const user of wallet._wallet.users) {\n        const userObject = usersMap.get(user.user);\n        if (user.permissions.includes('spend') && !user.permissions.includes('admin') && userObject) {\n          const shareParams = {\n            walletId: walletId,\n            user: user.user,\n            permissions: user.permissions.join(','),\n            walletPassphrase: userPassword,\n            email: userObject.email.email,\n            reshare: true,\n            skipKeychain: false,\n          };\n          await wallet.shareWallet(shareParams);\n        }\n      }\n    }\n  }\n\n  /**\n   * Accepts a wallet share, adding the wallet to the user's list\n   * Needs a user's password to decrypt the shared key\n   *\n   * @param params\n   * @param params.walletShareId - the wallet share to accept\n   * @param params.userPassword - (required if more a keychain was shared) user's password to decrypt the shared wallet\n   * @param params.newWalletPassphrase - new wallet passphrase for saving the shared wallet prv.\n   *                                     If left blank and a wallet with more than view permissions was shared,\n   *                                     then the user's login password is used.\n   * @param params.overrideEncryptedPrv - set only if the prv was received out-of-band.\n   */\n  async acceptShare(params: AcceptShareOptions = {}): Promise<any> {\n    common.validateParams(params, ['walletShareId'], ['overrideEncryptedPrv', 'userPassword', 'newWalletPassphrase']);\n\n    let encryptedPrv = params.overrideEncryptedPrv;\n    const walletShare = await this.getShare({ walletShareId: params.walletShareId });\n    if (\n      walletShare.keychainOverrideRequired &&\n      walletShare.permissions.indexOf('admin') !== -1 &&\n      walletShare.permissions.indexOf('spend') !== -1\n    ) {\n      if (_.isUndefined(params.userPassword)) {\n        throw new Error('userPassword param must be provided to decrypt shared key');\n      }\n\n      const walletKeychain = await this.baseCoin.keychains().createUserKeychain(params.userPassword);\n      if (_.isUndefined(walletKeychain.encryptedPrv)) {\n        throw new Error('encryptedPrv was not found on wallet keychain');\n      }\n\n      const payload = {\n        tradingAccountId: walletShare.wallet,\n        pubkey: walletKeychain.pub,\n        timestamp: new Date().toISOString(),\n      };\n      const payloadString = JSON.stringify(payload);\n\n      const privateKey = this.bitgo.decrypt({\n        password: params.userPassword,\n        input: walletKeychain.encryptedPrv,\n      });\n      const signature = await this.baseCoin.signMessage({ prv: privateKey }, payloadString);\n\n      const response = await this.updateShare({\n        walletShareId: params.walletShareId,\n        state: 'accepted',\n        keyId: walletKeychain.id,\n        signature: signature.toString('hex'),\n        payload: payloadString,\n      });\n      // If the wallet share was accepted successfully (changed=true), reshare the wallet with the spenders\n      if (response.changed && response.state === 'accepted') {\n        try {\n          await this.reshareWalletWithSpenders(walletShare.wallet, params.userPassword);\n        } catch (e) {\n          // TODO: PX-3826\n          // Do nothing\n        }\n      }\n      return response;\n    }\n    // Return right away if there is no keychain to decrypt, or if explicit encryptedPrv was provided\n    if (!walletShare.keychain || !walletShare.keychain.encryptedPrv || encryptedPrv) {\n      return this.updateShare({\n        walletShareId: params.walletShareId,\n        state: 'accepted',\n      });\n    }\n\n    // More than viewing was requested, so we need to process the wallet keys using the shared ecdh scheme\n    if (_.isUndefined(params.userPassword)) {\n      throw new Error('userPassword param must be provided to decrypt shared key');\n    }\n\n    const sharingKeychain = (await this.bitgo.getECDHKeychain()) as any;\n    if (_.isUndefined(sharingKeychain.encryptedXprv)) {\n      throw new Error('encryptedXprv was not found on sharing keychain');\n    }\n\n    // Now we have the sharing keychain, we can work out the secret used for sharing the wallet with us\n    sharingKeychain.prv = this.bitgo.decrypt({\n      password: params.userPassword,\n      input: sharingKeychain.encryptedXprv,\n    });\n    const secret = getSharedSecret(\n      // Derive key by path (which is used between these 2 users only)\n      bip32.fromBase58(sharingKeychain.prv).derivePath(sanitizeLegacyPath(walletShare.keychain.path)),\n      Buffer.from(walletShare.keychain.fromPubKey, 'hex')\n    ).toString('hex');\n\n    // Yes! We got the secret successfully here, now decrypt the shared wallet prv\n    const decryptedSharedWalletPrv = this.bitgo.decrypt({\n      password: secret,\n      input: walletShare.keychain.encryptedPrv,\n    });\n\n    // We will now re-encrypt the wallet with our own password\n    const newWalletPassphrase = params.newWalletPassphrase || params.userPassword;\n    encryptedPrv = this.bitgo.encrypt({\n      password: newWalletPassphrase,\n      input: decryptedSharedWalletPrv,\n    });\n    const updateParams: UpdateShareOptions = {\n      walletShareId: params.walletShareId,\n      state: 'accepted',\n    };\n\n    if (encryptedPrv) {\n      updateParams.encryptedPrv = encryptedPrv;\n    }\n    return this.updateShare(updateParams);\n  }\n\n  /**\n   * Bulk Accept wallet shares, adding the wallets to the user's list\n   * Needs a user's password to decrypt the shared key\n   *\n   * @param params BulkAcceptShareOptions\n   * @param params.walletShareId - array of the wallet shares to accept\n   * @param params.userPassword - user's password to decrypt the shared wallet key\n   * @param params.newWalletPassphrase - new wallet passphrase for saving the shared wallet prv.\n   *                                     If left blank then the user's login password is used.\n   *\n   *@returns {Promise<BulkAcceptShareResponse>}\n   */\n  async bulkAcceptShare(params: BulkAcceptShareOptions): Promise<BulkAcceptShareResponse> {\n    try {\n      common.validateParams(params, ['userLoginPassword'], ['newWalletPassphrase']);\n    } catch (e) {\n      if ('newWalletPassphrase' in params) {\n        throw new Error('Please provide a valid wallet passphrase');\n      }\n      throw new Error('Please provide a valid user login password');\n    }\n\n    assert(params.walletShareIds.length > 0, 'Please provide at least one wallet share to accept');\n\n    const allWalletShares = await this.listSharesV2();\n    const walletShareMap = allWalletShares.incoming.reduce(\n      (map: { [key: string]: WalletShare }, share) => ({ ...map, [share.id]: share }),\n      {}\n    );\n\n    const walletShares = params.walletShareIds\n      .map((walletShareId) => walletShareMap[walletShareId])\n      .filter((walletShare) => walletShare && walletShare.keychain);\n\n    if (!walletShares.length) {\n      throw new Error('No valid wallet shares found to accept');\n    }\n\n    const sharingKeychain = await this.bitgo.getECDHKeychain();\n    if (_.isUndefined(sharingKeychain.encryptedXprv)) {\n      throw new Error('encryptedXprv was not found on sharing keychain');\n    }\n\n    sharingKeychain.prv = this.bitgo.decrypt({\n      password: params.userLoginPassword,\n      input: sharingKeychain.encryptedXprv,\n    });\n    const newWalletPassphrase = params.newWalletPassphrase || params.userLoginPassword;\n    const keysForWalletShares = walletShares.flatMap((walletShare) => {\n      if (!walletShare.keychain) {\n        return [];\n      }\n      const secret = getSharedSecret(\n        bip32.fromBase58(sharingKeychain.prv).derivePath(sanitizeLegacyPath(walletShare.keychain.path)),\n        Buffer.from(walletShare.keychain.fromPubKey, 'hex')\n      ).toString('hex');\n\n      const decryptedSharedWalletPrv = this.bitgo.decrypt({\n        password: secret,\n        input: walletShare.keychain.encryptedPrv,\n      });\n      const newEncryptedPrv = this.bitgo.encrypt({\n        password: newWalletPassphrase,\n        input: decryptedSharedWalletPrv,\n      });\n      return [\n        {\n          walletShareId: walletShare.id,\n          encryptedPrv: newEncryptedPrv,\n        },\n      ];\n    });\n\n    return this.bulkAcceptShareRequest(keysForWalletShares);\n  }\n\n  /**\n   * Updates multiple wallet shares in bulk\n   * This method allows users to accept or reject multiple wallet shares in a single operation.\n   * It handles different types of wallet shares including those requiring special keychain overrides\n   * and those with encrypted private keys that need to be decrypted and re-encrypted.\n   * After processing, it also reshares accepted wallets with spenders for special override cases.\n   *\n   * @param params - Options for bulk updating wallet shares\n   * @param params.shares - Array of wallet shares to update with their status (accept/reject)\n   * @param params.userLoginPassword - User's login password for decryption operations\n   * @param params.newWalletPassphrase - New wallet passphrase for re-encryption\n   * @returns Array of responses for each wallet share update\n   */\n  async bulkUpdateWalletShare(params: BulkUpdateWalletShareOptions): Promise<BulkUpdateWalletShareResponse> {\n    if (!params.shares) {\n      throw new Error('Missing parameter: shares');\n    }\n\n    if (!Array.isArray(params.shares)) {\n      throw new Error('Expecting parameter array: shares but found ' + typeof params.shares);\n    }\n\n    // Validate each share in the array\n    for (const share of params.shares) {\n      if (!share.walletShareId) {\n        throw new Error('Missing walletShareId in share');\n      }\n\n      if (!share.status) {\n        throw new Error('Missing status in share');\n      }\n\n      if (share.status !== 'accept' && share.status !== 'reject') {\n        throw new Error('Invalid status in share: ' + share.status + '. Must be either \"accept\" or \"reject\"');\n      }\n\n      if (typeof share.walletShareId !== 'string') {\n        throw new Error('Expecting walletShareId to be a string but found ' + typeof share.walletShareId);\n      }\n    }\n\n    // Validate optional parameters if provided\n    if (params.userLoginPassword !== undefined && typeof params.userLoginPassword !== 'string') {\n      throw new Error('Expecting parameter string: userLoginPassword but found ' + typeof params.userLoginPassword);\n    }\n\n    if (params.newWalletPassphrase !== undefined && typeof params.newWalletPassphrase !== 'string') {\n      throw new Error('Expecting parameter string: newWalletPassphrase but found ' + typeof params.newWalletPassphrase);\n    }\n    assert(params.shares.length > 0, 'no shares are passed');\n\n    const { shares: inputShares, userLoginPassword, newWalletPassphrase } = params;\n\n    const allWalletShares = await this.listSharesV2();\n\n    // Only include shares that are in the input array for efficiency\n    const shareIds = new Set(inputShares.map((share) => share.walletShareId));\n    const walletShareMap = new Map();\n\n    allWalletShares.incoming\n      .filter((share) => shareIds.has(share.id))\n      .forEach((share) => walletShareMap.set(share.id, share));\n\n    allWalletShares.outgoing\n      .filter((share) => shareIds.has(share.id))\n      .forEach((share) => walletShareMap.set(share.id, share));\n\n    const resolvedShares = inputShares.map((share) => {\n      const walletShare = walletShareMap.get(share.walletShareId);\n      if (!walletShare) {\n        throw new Error(`invalid wallet share provided: ${share.walletShareId}`);\n      }\n      return { ...share, walletShare };\n    });\n\n    // Identify special override cases that need resharing after acceptance\n    const specialOverrideCases = new Map();\n    resolvedShares.forEach((share) => {\n      if (\n        share.status === 'accept' &&\n        share.walletShare.keychainOverrideRequired &&\n        share.walletShare.permissions.includes('admin') &&\n        share.walletShare.permissions.includes('spend')\n      ) {\n        specialOverrideCases.set(share.walletShareId, share.walletShare.wallet);\n      }\n    });\n\n    // Decrypt sharing keychain if needed (only once)\n    let sharingKeychainPrv: string | undefined;\n\n    // Only decrypt if there are shares to accept that might need it\n    const hasSharesRequiringDecryption =\n      specialOverrideCases.size > 0 ||\n      resolvedShares.some((share) => share.status === 'accept' && share.walletShare.keychain?.encryptedPrv);\n\n    if (userLoginPassword && hasSharesRequiringDecryption) {\n      const sharingKeychain = await this.bitgo.getECDHKeychain();\n      if (!sharingKeychain.encryptedXprv) {\n        throw new Error('encryptedXprv was not found on sharing keychain');\n      }\n      sharingKeychainPrv = this.bitgo.decrypt({\n        password: userLoginPassword,\n        input: sharingKeychain.encryptedXprv,\n      });\n    }\n\n    const settledUpdates = await Promise.allSettled(\n      resolvedShares.map(async (share) => {\n        const { walletShareId, status, walletShare } = share;\n\n        // Handle accept case\n        if (status === 'accept') {\n          return this.processAcceptShare(\n            walletShareId,\n            walletShare,\n            userLoginPassword,\n            newWalletPassphrase,\n            sharingKeychainPrv\n          );\n        }\n\n        // Handle reject case\n        return [\n          {\n            walletShareId,\n            status: 'reject' as const,\n          },\n        ];\n      })\n    );\n\n    // Extract successful updates\n    const successfulUpdates = settledUpdates.flatMap((result) => (result.status === 'fulfilled' ? result.value : []));\n\n    // Extract failed updates - only from rejected promises\n    const failedUpdates = settledUpdates.reduce<Array<{ walletShareId: string; reason: string }>>(\n      (acc, result, index) => {\n        if (result.status === 'rejected') {\n          const rejectedResult = result;\n          acc.push({\n            walletShareId: resolvedShares[index].walletShareId,\n            reason: rejectedResult.reason?.message || String(rejectedResult.reason),\n          });\n        }\n        return acc;\n      },\n      []\n    );\n\n    // Send successful updates to the server\n    const response = await this.bulkUpdateWalletShareRequest(successfulUpdates);\n\n    // Process accepted special override cases - reshare with spenders\n    if (response.acceptedWalletShares && response.acceptedWalletShares.length > 0 && userLoginPassword) {\n      // For each accepted wallet share that is a special override case, reshare with spenders\n      for (const walletShareId of response.acceptedWalletShares) {\n        if (specialOverrideCases.has(walletShareId)) {\n          const walletId = specialOverrideCases.get(walletShareId);\n          try {\n            await this.reshareWalletWithSpenders(walletId, userLoginPassword);\n          } catch (e) {\n            // Log error but continue processing other shares\n            console.error(`Error resharing wallet ${walletId} with spenders: ${e?.message}`);\n          }\n        }\n      }\n    }\n\n    // Add information about failed updates to the response\n    if (failedUpdates.length > 0) {\n      response.walletShareUpdateErrors.push(...failedUpdates);\n    }\n\n    return response;\n  }\n\n  /**\n   * Process a wallet share that is being accepted\n   * This method handles the different cases for accepting a wallet share:\n   * 1. Special override case requiring user keychain and signing\n   * 2. Simple case with no keychain to decrypt\n   * 3. Standard case requiring decryption and re-encryption\n   *\n   * @param walletShareId - ID of the wallet share\n   * @param walletShare - Wallet share object\n   * @param userLoginPassword - User's login password\n   * @param newWalletPassphrase - New wallet passphrase\n   * @param sharingKeychainPrv - Decrypted sharing keychain private key\n   * @returns Array of wallet share update requests\n   */\n  private async processAcceptShare(\n    walletShareId: string,\n    walletShare: WalletShare,\n    userLoginPassword?: string,\n    newWalletPassphrase?: string,\n    sharingKeychainPrv?: string\n  ): Promise<BulkUpdateWalletShareOptionsRequest[]> {\n    // Special override case: requires user keychain and signing\n    if (\n      walletShare.keychainOverrideRequired &&\n      walletShare.permissions.includes('admin') &&\n      walletShare.permissions.includes('spend')\n    ) {\n      if (!userLoginPassword) {\n        throw new Error('userLoginPassword param must be provided to decrypt shared key');\n      }\n\n      const walletKeychain = await this.baseCoin.keychains().createUserKeychain(userLoginPassword);\n      if (!walletKeychain.encryptedPrv) {\n        throw new Error('encryptedPrv was not found on wallet keychain');\n      }\n\n      const payload = JSON.stringify({\n        tradingAccountId: walletShare.wallet,\n        pubkey: walletKeychain.pub,\n        timestamp: new Date().toISOString(),\n      });\n\n      const prv = this.bitgo.decrypt({\n        password: userLoginPassword,\n        input: walletKeychain.encryptedPrv,\n      });\n\n      const signature = await this.baseCoin.signMessage({ prv }, payload);\n\n      return [\n        {\n          walletShareId,\n          status: 'accept' as const,\n          keyId: walletKeychain.id,\n          signature: signature.toString('hex'),\n          payload,\n        },\n      ];\n    }\n\n    // Return right away if there is no keychain to decrypt\n    if (!walletShare.keychain || !walletShare.keychain.encryptedPrv) {\n      return [\n        {\n          walletShareId,\n          status: 'accept' as const,\n        },\n      ];\n    }\n\n    // More than viewing was requested, so we need to process the wallet keys using the shared ecdh scheme\n    if (!userLoginPassword) {\n      throw new Error('userLoginPassword param must be provided to decrypt shared key');\n    }\n    if (!sharingKeychainPrv) {\n      throw new Error('failed to retrieve and decrypt sharing keychain');\n    }\n\n    const derivedKey = bip32.fromBase58(sharingKeychainPrv).derivePath(sanitizeLegacyPath(walletShare.keychain.path));\n\n    const sharedSecret = getSharedSecret(derivedKey, Buffer.from(walletShare.keychain.fromPubKey, 'hex')).toString(\n      'hex'\n    );\n\n    const decryptedPrv = this.bitgo.decrypt({\n      password: sharedSecret,\n      input: walletShare.keychain.encryptedPrv,\n    });\n\n    // We will now re-encrypt the wallet with our own password\n    const encryptedPrv = this.bitgo.encrypt({\n      password: newWalletPassphrase || userLoginPassword,\n      input: decryptedPrv,\n    });\n\n    return [\n      {\n        walletShareId,\n        status: 'accept' as const,\n        encryptedPrv,\n      },\n    ];\n  }\n\n  /**\n   * Get a wallet by its ID\n   * @param params\n   * @param params.id wallet id\n   * @returns {*}\n   */\n  async getWallet(params: GetWalletOptions = {}): Promise<Wallet> {\n    common.validateParams(params, ['id'], []);\n\n    const query: GetWalletOptions = {};\n    if (params.allTokens) {\n      if (!_.isBoolean(params.allTokens)) {\n        throw new Error('invalid allTokens argument, expecting boolean');\n      }\n      query.allTokens = params.allTokens;\n    }\n\n    if (params.includeBalance !== undefined) {\n      query.includeBalance = params.includeBalance;\n    }\n\n    this.bitgo.setRequestTracer(params.reqId || new RequestTracer());\n\n    const wallet = await this.bitgo\n      .get(this.baseCoin.url('/wallet/' + params.id))\n      .query(query)\n      .result();\n    return new Wallet(this.bitgo, this.baseCoin, wallet);\n  }\n\n  /**\n   * Get a wallet by its address\n   * @param params\n   * @param params.address wallet address\n   * @returns {*}\n   */\n  async getWalletByAddress(params: GetWalletByAddressOptions = {}): Promise<Wallet> {\n    common.validateParams(params, ['address'], []);\n\n    this.bitgo.setRequestTracer(params.reqId || new RequestTracer());\n\n    const wallet = await this.bitgo.get(this.baseCoin.url('/wallet/address/' + params.address)).result();\n    return new Wallet(this.bitgo, this.baseCoin, wallet);\n  }\n\n  /**\n   * For any given supported coin, get total balances for all wallets of that\n   * coin type on the account.\n   * @param params\n   * @returns {*}\n   */\n  async getTotalBalances(params: Record<string, never> = {}): Promise<any> {\n    return await this.bitgo.get(this.baseCoin.url('/wallet/balances')).result();\n  }\n\n  /**\n   * Generates a TSS or BLS-DKG Wallet.\n   * @param params\n   * @private\n   */\n  private async generateMpcWallet({\n    passphrase,\n    label,\n    multisigType,\n    enterprise,\n    walletVersion,\n    originalPasscodeEncryptionCode,\n  }: GenerateMpcWalletOptions): Promise<WalletWithKeychains> {\n    if (multisigType === 'tss' && this.baseCoin.getMPCAlgorithm() === 'ecdsa') {\n      const tssSettings: TssSettings = await this.bitgo\n        .get(this.bitgo.microservicesUrl('/api/v2/tss/settings'))\n        .result();\n      const multisigTypeVersion =\n        tssSettings.coinSettings[this.baseCoin.getFamily()]?.walletCreationSettings?.multiSigTypeVersion;\n      walletVersion = this.determineEcdsaMpcWalletVersion(walletVersion, multisigTypeVersion);\n    }\n\n    const reqId = new RequestTracer();\n    this.bitgo.setRequestTracer(reqId);\n\n    // Create MPC Keychains\n    const keychains = await this.baseCoin.keychains().createMpc({\n      multisigType,\n      passphrase,\n      enterprise,\n      originalPasscodeEncryptionCode,\n    });\n\n    // Create Wallet\n    const { userKeychain, backupKeychain, bitgoKeychain } = keychains;\n    const walletParams: SupplementGenerateWalletOptions = {\n      label,\n      m: 2,\n      n: 3,\n      keys: [userKeychain.id, backupKeychain.id, bitgoKeychain.id],\n      type: 'hot',\n      multisigType,\n      enterprise,\n      walletVersion,\n    };\n    const finalWalletParams = await this.baseCoin.supplementGenerateWallet(walletParams, keychains);\n    const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(finalWalletParams).result();\n\n    const result: WalletWithKeychains = {\n      wallet: new Wallet(this.bitgo, this.baseCoin, newWallet),\n      userKeychain,\n      backupKeychain,\n      bitgoKeychain,\n      responseType: 'WalletWithKeychains',\n    };\n\n    if (!_.isUndefined(backupKeychain.prv)) {\n      result.warning = 'Be sure to backup the backup keychain -- it is not stored anywhere else!';\n    }\n\n    return result;\n  }\n\n  /**\n   * Generates a Self-Managed Cold TSS Wallet.\n   * @param params\n   * @private\n   */\n  private async generateSMCMpcWallet({\n    label,\n    multisigType,\n    enterprise,\n    walletVersion,\n    bitgoKeyId,\n    commonKeychain,\n    coldDerivationSeed,\n  }: GenerateSMCMpcWalletOptions): Promise<WalletWithKeychains> {\n    const reqId = new RequestTracer();\n    this.bitgo.setRequestTracer(reqId);\n\n    let multisigTypeVersion: 'MPCv2' | undefined;\n    if (multisigType === 'tss' && this.baseCoin.getMPCAlgorithm() === 'ecdsa') {\n      const tssSettings: TssSettings = await this.bitgo\n        .get(this.bitgo.microservicesUrl('/api/v2/tss/settings'))\n        .result();\n      multisigTypeVersion =\n        tssSettings.coinSettings[this.baseCoin.getFamily()]?.walletCreationSettings?.coldMultiSigTypeVersion;\n      walletVersion = this.determineEcdsaMpcWalletVersion(walletVersion, multisigTypeVersion);\n    }\n\n    // Create MPC Keychains\n    const bitgoKeychain = await this.baseCoin.keychains().get({ id: bitgoKeyId });\n\n    if (!bitgoKeychain || !bitgoKeychain.commonKeychain) {\n      throw new Error('BitGo keychain not found');\n    }\n\n    if (bitgoKeychain.source !== 'bitgo') {\n      throw new Error('The provided bitgoKeyId is not a BitGo keychain');\n    }\n\n    if (bitgoKeychain.commonKeychain !== commonKeychain) {\n      throw new Error('The provided Common keychain mismatch with the provided Bitgo key');\n    }\n\n    if (!coldDerivationSeed) {\n      throw new Error('derivedFromParentWithSeed is required');\n    }\n\n    const userKeychainParams: AddKeychainOptions = {\n      source: 'user',\n      keyType: 'tss',\n      commonKeychain: commonKeychain,\n      derivedFromParentWithSeed: coldDerivationSeed,\n      isMPCv2: multisigTypeVersion === 'MPCv2' ? true : undefined,\n    };\n    const userKeychain = await this.baseCoin.keychains().add(userKeychainParams);\n\n    const backupKeyChainParams: AddKeychainOptions = {\n      source: 'backup',\n      keyType: 'tss',\n      commonKeychain: commonKeychain,\n      derivedFromParentWithSeed: coldDerivationSeed,\n      isMPCv2: multisigTypeVersion === 'MPCv2' ? true : undefined,\n    };\n\n    const backupKeychain = await this.baseCoin.keychains().add(backupKeyChainParams);\n\n    // Create Wallet\n    const keychains = { userKeychain, backupKeychain, bitgoKeychain };\n    const walletParams: SupplementGenerateWalletOptions = {\n      label,\n      m: 2,\n      n: 3,\n      keys: [userKeychain.id, backupKeychain.id, bitgoKeychain.id],\n      type: 'cold',\n      multisigType,\n      enterprise,\n      walletVersion,\n    };\n\n    const finalWalletParams = await this.baseCoin.supplementGenerateWallet(walletParams, keychains);\n    const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(finalWalletParams).result();\n\n    const result: WalletWithKeychains = {\n      wallet: new Wallet(this.bitgo, this.baseCoin, newWallet),\n      userKeychain,\n      backupKeychain,\n      bitgoKeychain,\n      responseType: 'WalletWithKeychains',\n    };\n\n    return result;\n  }\n\n  /**\n   * Generates a Custodial TSS Wallet.\n   * @param params\n   * @private\n   */\n  private async generateCustodialMpcWallet({\n    label,\n    multisigType,\n    enterprise,\n    walletVersion,\n  }: GenerateBaseMpcWalletOptions): Promise<WalletWithKeychains> {\n    const reqId = new RequestTracer();\n    this.bitgo.setRequestTracer(reqId);\n\n    if (multisigType === 'tss' && this.baseCoin.getMPCAlgorithm() === 'ecdsa') {\n      const tssSettings: TssSettings = await this.bitgo\n        .get(this.bitgo.microservicesUrl('/api/v2/tss/settings'))\n        .result();\n      const multisigTypeVersion =\n        tssSettings.coinSettings[this.baseCoin.getFamily()]?.walletCreationSettings?.custodialMultiSigTypeVersion;\n      walletVersion = this.determineEcdsaMpcWalletVersion(walletVersion, multisigTypeVersion);\n    }\n\n    const finalWalletParams = {\n      label,\n      multisigType,\n      enterprise,\n      walletVersion,\n      type: 'custodial',\n    };\n\n    // Create Wallet\n    const newWallet = await this.bitgo.post(this.baseCoin.url('/wallet/add')).send(finalWalletParams).result();\n    const wallet = new Wallet(this.bitgo, this.baseCoin, newWallet);\n    const keychains = wallet.keyIds();\n    const result: WalletWithKeychains = {\n      wallet,\n      userKeychain: { id: keychains[0], type: multisigType, source: 'user' },\n      backupKeychain: { id: keychains[1], type: multisigType, source: 'backup' },\n      bitgoKeychain: { id: keychains[2], type: multisigType, source: 'bitgo' },\n      responseType: 'WalletWithKeychains',\n    };\n\n    return result;\n  }\n\n  private determineEcdsaMpcWalletVersion(walletVersion?: number, multisigTypeVersion?: string): number | undefined {\n    if (this.baseCoin.isEVM() && multisigTypeVersion === 'MPCv2') {\n      if (!walletVersion || (walletVersion !== 5 && walletVersion !== 6)) {\n        return 5;\n      }\n    }\n    return walletVersion;\n  }\n}\n"]}

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


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