PHP WebShell

Текущая директория: /opt/BitGoJS/modules/bitgo/dist/test/v2/unit

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

"use strict";
//
// Tests for Wallet
//
Object.defineProperty(exports, "__esModule", { value: true });
const should = require("should");
const sinon = require("sinon");
require("../lib/asserts");
const nock = require("nock");
const _ = require("lodash");
const sdk_core_1 = require("@bitgo/sdk-core");
const sdk_test_1 = require("@bitgo/sdk-test");
const src_1 = require("../../../src");
const utxoLib = require("@bitgo/utxo-lib");
const crypto_1 = require("crypto");
const util_1 = require("./coins/utxo/util");
const sdk_coin_sol_1 = require("@bitgo/sdk-coin-sol");
const sdk_coin_eth_1 = require("@bitgo/sdk-coin-eth");
const nftResponses_1 = require("../fixtures/nfts/nftResponses");
require('should-sinon');
nock.disableNetConnect();
describe('V2 Wallet:', function () {
    const reqId = new sdk_core_1.RequestTracer();
    const bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'test' });
    bitgo.initializeTestVars();
    const basecoin = bitgo.coin('tbtc');
    const walletData = {
        id: '5b34252f1bf349930e34020a00000000',
        coin: 'tbtc',
        keys: ['5b3424f91bf349930e34017500000000', '5b3424f91bf349930e34017600000000', '5b3424f91bf349930e34017700000000'],
        coinSpecific: {},
        multisigType: 'onchain',
        type: 'hot',
    };
    const coldWalletData = {
        id: '65774419fb4d9690847fbe4b00000000',
        coin: 'tbtc',
        keys: ['65774412e54b7516393c9df800000000', '6577442428664ffe791af7ea00000000', '6577442b7317a945756c2fd900000000'],
        coinSpecific: {},
        multisigType: 'onchain',
        type: 'cold',
    };
    const wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
    const coldWallet = new sdk_core_1.Wallet(bitgo, basecoin, coldWalletData);
    const bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
    const address1 = '0x174cfd823af8ce27ed0afee3fcf3c3ba259116be';
    const address2 = '0x7e85bdc27c050e3905ebf4b8e634d9ad6edd0de6';
    const tbtcHotWalletDefaultParams = {
        txFormat: 'psbt',
        changeAddressType: ['p2trMusig2', 'p2wsh', 'p2shP2wsh', 'p2sh', 'p2tr'],
    };
    afterEach(function () {
        sinon.restore();
        sinon.reset();
    });
    describe('Wallet transfers', function () {
        it('should search in wallet for a transfer', async function () {
            const params = { limit: 1, searchLabel: 'test' };
            const scope = nock(bgUrl)
                .get(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/transfer`)
                .query(params)
                .reply(200, {
                coin: 'tbch',
                transfers: [
                    {
                        wallet: wallet.id(),
                        comment: 'tests',
                    },
                ],
            });
            try {
                await wallet.transfers(params);
            }
            catch (e) {
                // test is successful if nock is consumed, HMAC errors expected
            }
            scope.isDone().should.be.True();
        });
        it('should forward all valid parameters', async function () {
            const params = {
                limit: 1,
                address: ['address1', 'address2'],
                dateGte: 'dateString0',
                dateLt: 'dateString1',
                valueGte: 0,
                valueLt: 300000000,
                allTokens: true,
                searchLabel: 'abc',
                includeHex: true,
                type: 'transfer_type',
                state: 'transfer_state',
            };
            // The actual api request will only send strings, but the SDK function expects numbers for some values
            const apiParams = _.mapValues(params, (param) => (Array.isArray(param) ? param : String(param)));
            const scope = nock(bgUrl)
                .get(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/transfer`)
                .query(_.matches(apiParams))
                .reply(200);
            await wallet.transfers(params);
            scope.isDone().should.be.True();
        });
        it('should accept a string argument for address', async function () {
            const params = {
                limit: 1,
                address: 'stringAddress',
            };
            const apiParams = {
                limit: '1',
                address: 'stringAddress',
            };
            const scope = nock(bgUrl)
                .get(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/transfer`)
                .query(_.matches(apiParams))
                .reply(200);
            try {
                await wallet.transfers(params);
            }
            catch (e) {
                // test is successful if nock is consumed, HMAC errors expected
            }
            scope.isDone().should.be.True();
        });
        it('should throw errors for invalid expected parameters', async function () {
            await wallet
                // @ts-expect-error checking type mismatch
                .transfers({ address: 13375 })
                .should.be.rejectedWith('invalid address argument, expecting string or array');
            await wallet
                // @ts-expect-error checking type mismatch
                .transfers({ address: [null] })
                .should.be.rejectedWith('invalid address argument, expecting array of address strings');
            await wallet
                // @ts-expect-error checking type mismatch
                .transfers({ dateGte: 20101904 })
                .should.be.rejectedWith('invalid dateGte argument, expecting string');
            // @ts-expect-error checking type mismatch
            await wallet.transfers({ dateLt: 20101904 }).should.be.rejectedWith('invalid dateLt argument, expecting string');
            await wallet
                // @ts-expect-error checking type mismatch
                .transfers({ valueGte: '10230005' })
                .should.be.rejectedWith('invalid valueGte argument, expecting number');
            // @ts-expect-error checking type mismatch
            await wallet.transfers({ valueLt: '-5e8' }).should.be.rejectedWith('invalid valueLt argument, expecting number');
            await wallet
                // @ts-expect-error checking type mismatch
                .transfers({ includeHex: '123' })
                .should.be.rejectedWith('invalid includeHex argument, expecting boolean');
            await wallet
                // @ts-expect-error checking type mismatch
                .transfers({ state: 123 })
                .should.be.rejectedWith('invalid state argument, expecting string or array');
            await wallet
                // @ts-expect-error checking type mismatch
                .transfers({ state: [123, 456] })
                .should.be.rejectedWith('invalid state argument, expecting array of state strings');
            // @ts-expect-error checking type mismatch
            await wallet.transfers({ type: 123 }).should.be.rejectedWith('invalid type argument, expecting string');
        });
    });
    describe('Wallet addresses', function () {
        it('should search in wallet addresses', async function () {
            const params = { limit: 1, labelContains: 'test' };
            const scope = nock(bgUrl)
                .get(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/addresses`)
                .query(params)
                .reply(200, {
                coin: 'tbch',
                transfers: [
                    {
                        wallet: wallet.id(),
                        comment: 'tests',
                    },
                ],
            });
            try {
                await wallet.addresses(params);
            }
            catch (e) {
                // test is successful if nock is consumed, HMAC errors expected
            }
            scope.isDone().should.be.True();
        });
    });
    it('should verify bch cashaddr format as valid', async function () {
        const coin = bitgo.coin('tbch');
        const valid = coin.isValidAddress('bchtest:pzfkxv532t0q5zchck2mhmmf2y02cdejyssq5qrz7a');
        valid.should.be.True();
    });
    it('should verify bch legacy format as valid', async function () {
        const coin = bitgo.coin('tbch');
        const valid = coin.isValidAddress('2N6gY9r9iuXQQzZiSyngWJeoUuL5mC1x4Ac');
        valid.should.be.True();
    });
    describe('TETH Wallet Addresses', function () {
        let ethWallet;
        before(async function () {
            const walletData = {
                id: '598f606cd8fc24710d2ebadb1d9459bb',
                coin: 'teth',
                keys: [
                    '598f606cd8fc24710d2ebad89dce86c2',
                    '598f606cc8e43aef09fcb785221d9dd2',
                    '5935d59cf660764331bafcade1855fd7',
                ],
            };
            ethWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('teth'), walletData);
        });
        it('search list addresses should return success', async function () {
            const params = {
                includeBalances: true,
                includeTokens: true,
                returnBalancesForToken: 'gterc6dp',
                pendingDeployment: false,
                includeTotalAddressCount: true,
            };
            const scope = nock(bgUrl)
                .get(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/addresses`)
                .query(params)
                .reply(200);
            try {
                await wallet.addresses(params);
                throw '';
            }
            catch (error) {
                // test is successful if nock is consumed, HMAC errors expected
            }
            scope.isDone().should.be.True();
        });
        it('should throw errors for invalid expected parameters', async function () {
            await ethWallet
                .addresses({ includeBalances: true, returnBalancesForToken: 1 })
                .should.be.rejectedWith('invalid returnBalancesForToken argument, expecting string');
            await ethWallet
                .addresses({ pendingDeployment: 1 })
                .should.be.rejectedWith('invalid pendingDeployment argument, expecting boolean');
            await ethWallet
                .addresses({ includeBalances: 1 })
                .should.be.rejectedWith('invalid includeBalances argument, expecting boolean');
            await ethWallet
                .addresses({ includeTokens: 1 })
                .should.be.rejectedWith('invalid includeTokens argument, expecting boolean');
            await ethWallet
                .addresses({ includeTotalAddressCount: 1 })
                .should.be.rejectedWith('invalid includeTotalAddressCount argument, expecting boolean');
        });
        it('get forwarder balance', async function () {
            const forwarders = [
                {
                    address: '0xbfbcc0fe2b865de877134246af09378e9bc3c91d',
                    balance: '200000',
                },
                {
                    address: '0xe59524ed8b47165f4cb0850c9428069a6002e5eb',
                    balance: '10000000000000000',
                },
            ];
            nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/forwarders/balances`).reply(200, {
                forwarders,
            });
            const forwarderBalance = await ethWallet.getForwarderBalance();
            forwarderBalance.forwarders[0].address.should.eql(forwarders[0].address);
            forwarderBalance.forwarders[0].balance.should.eql(forwarders[0].balance);
            forwarderBalance.forwarders[1].address.should.eql(forwarders[1].address);
            forwarderBalance.forwarders[1].balance.should.eql(forwarders[1].balance);
        });
    });
    describe('Get User Prv', () => {
        const prv = 'xprv9s21ZrQH143K3hekyNj7TciR4XNYe1kMj68W2ipjJGNHETWP7o42AjDnSPgKhdZ4x8NBAvaL72RrXjuXNdmkMqLERZza73oYugGtbLFXG8g';
        const derivedPrv = 'xprv9yoG67Td11uwjXwbV8zEmrySVXERu5FZAsLD9suBeEJbgJqANs8Yng5dEJoii7hag5JermK6PbfxgDmSzW7ewWeLmeJEkmPfmZUSLdETtHx';
        it('should use the cold derivation seed to derive the proper user private key', async () => {
            const userPrvOptions = {
                prv,
                coldDerivationSeed: '123',
            };
            wallet.getUserPrv(userPrvOptions).should.eql(derivedPrv);
        });
        it('should use the user keychain derivedFromParentWithSeed as the cold derivation seed if none is provided', async () => {
            const userPrvOptions = {
                prv,
                keychain: {
                    derivedFromParentWithSeed: '123',
                    id: '456',
                    pub: '789',
                    type: 'independent',
                },
            };
            wallet.getUserPrv(userPrvOptions).should.eql(derivedPrv);
        });
        it('should prefer the explicit cold derivation seed to the user keychain derivedFromParentWithSeed', async () => {
            const userPrvOptions = {
                prv,
                coldDerivationSeed: '123',
                keychain: {
                    derivedFromParentWithSeed: '456',
                    id: '789',
                    pub: '012',
                    type: 'independent',
                },
            };
            wallet.getUserPrv(userPrvOptions).should.eql(derivedPrv);
        });
        it('should return the prv provided for TSS SMC', async () => {
            const tssWalletData = {
                id: '5b34252f1bf349930e34020a00000000',
                coin: 'tsol',
                keys: [
                    '5b3424f91bf349930e34017500000000',
                    '5b3424f91bf349930e34017600000000',
                    '5b3424f91bf349930e34017700000000',
                ],
                coinSpecific: {},
                multisigType: 'tss',
            };
            const tsolcoin = bitgo.coin('tsol');
            const wallet = new sdk_core_1.Wallet(bitgo, tsolcoin, tssWalletData);
            const prv = 'longstringifiedjson';
            const keychain = {
                derivedFromParentWithSeed: 'random seed',
                id: '123',
                commonKeychain: 'longstring',
                type: 'tss',
            };
            const userPrvOptions = {
                prv,
                keychain,
            };
            wallet.getUserPrv(userPrvOptions).should.eql(prv);
        });
    });
    describe('UTXO Custom Signer Function', function () {
        const recipients = [
            { address: 'abc', amount: 123 },
            { address: 'def', amount: 456 },
        ];
        const rootWalletKey = (0, util_1.getDefaultWalletKeys)();
        let customSigningFunction;
        let stubs;
        beforeEach(function () {
            customSigningFunction = sinon.stub().returns({
                txHex: 'this-is-a-tx',
            });
            stubs = [
                sinon.stub(wallet.baseCoin, 'postProcessPrebuild').returnsArg(0),
                sinon.stub(wallet.baseCoin, 'verifyTransaction').resolves(true),
                sinon.stub(wallet.baseCoin, 'signTransaction').resolves({ txHex: 'this-is-a-tx' }),
            ];
        });
        function nocks(txPrebuild) {
            return nock(bgUrl)
                .post(wallet.url('/tx/build').replace(bgUrl, ''))
                .reply(200, txPrebuild)
                .get(wallet.baseCoin.url('/public/block/latest').replace(bgUrl, ''))
                .reply(200)
                .get(wallet.baseCoin.url(`/key/${wallet.keyIds()[0]}`).replace(bgUrl, ''))
                .reply(200, { pub: 'pub' })
                .get(wallet.baseCoin.url(`/key/${wallet.keyIds()[1]}`).replace(bgUrl, ''))
                .reply(200, { pub: 'pub' })
                .get(wallet.baseCoin.url(`/key/${wallet.keyIds()[2]}`).replace(bgUrl, ''))
                .reply(200, { pub: 'pub' })
                .post(wallet.url('/tx/send').replace(bgUrl, ''))
                .reply(200, { ok: true });
        }
        it('should use a custom signing function if provided for PSBT with taprootKeyPathSpend input', async function () {
            const psbt = utxoLib.testutil.constructPsbt([{ scriptType: 'taprootKeyPathSpend', value: BigInt(1000) }], [{ scriptType: 'p2sh', value: BigInt(900) }], basecoin.network, rootWalletKey, 'unsigned');
            const scope = nocks({ txHex: psbt.toHex() });
            const result = await wallet.sendMany({ recipients, customSigningFunction });
            result.should.have.property('ok', true);
            customSigningFunction.should.have.been.calledTwice();
            scope.done();
            stubs.forEach((s) => s.restore());
        });
        it('should use a custom signing function if provided for PSBT without taprootKeyPathSpend input', async function () {
            const psbt = utxoLib.testutil.constructPsbt([{ scriptType: 'p2wsh', value: BigInt(1000) }], [{ scriptType: 'p2sh', value: BigInt(900) }], basecoin.network, rootWalletKey, 'unsigned');
            const scope = nocks({ txHex: psbt.toHex() });
            const result = await wallet.sendMany({ recipients, customSigningFunction });
            result.should.have.property('ok', true);
            customSigningFunction.should.have.been.calledOnce();
            scope.done();
            stubs.forEach((s) => s.restore());
        });
        it('should use a custom signing function if provided for Tx without taprootKeyPathSpend input', async function () {
            const tx = utxoLib.testutil.constructTxnBuilder([{ scriptType: 'p2wsh', value: BigInt(1000) }], [{ scriptType: 'p2sh', value: BigInt(900) }], basecoin.network, rootWalletKey, 'unsigned');
            const scope = nocks({ txHex: tx.buildIncomplete().toHex() });
            const result = await wallet.sendMany({ recipients, customSigningFunction });
            result.should.have.property('ok', true);
            customSigningFunction.should.have.been.calledOnce();
            scope.done();
            stubs.forEach((s) => s.restore());
        });
    });
    describe('TETH Wallet Transactions', function () {
        let ethWallet;
        before(async function () {
            const walletData = {
                id: '598f606cd8fc24710d2ebadb1d9459bb',
                coin: 'teth',
                keys: [
                    '598f606cd8fc24710d2ebad89dce86c2',
                    '598f606cc8e43aef09fcb785221d9dd2',
                    '5935d59cf660764331bafcade1855fd7',
                ],
                multisigType: 'onchain',
            };
            ethWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('teth'), walletData);
        });
        afterEach(async function () {
            nock.cleanAll();
        });
        it('should error eip1559 and gasPrice are passed', async function () {
            const params = {
                gasPrice: 100,
                eip1559: {
                    maxPriorityFeePerGas: 10,
                    maxFeePerGas: 10,
                },
                amount: 10,
                address: sdk_test_1.TestBitGo.V2.TEST_WALLET1_ADDRESS,
                walletPassphrase: sdk_test_1.TestBitGo.V2.TEST_WALLET1_PASSCODE,
            };
            await ethWallet.send(params).should.be.rejected();
        });
        it('should search for pending transaction correctly', async function () {
            const params = { walletId: wallet.id() };
            const scope = nock(bgUrl).get(`/api/v2/${wallet.coin()}/tx/pending/first`).query(params).reply(200);
            try {
                await wallet.getFirstPendingTransaction();
                throw '';
            }
            catch (error) {
                // test is successful if nock is consumed, HMAC errors expected
            }
            scope.isDone().should.be.True();
        });
        it('should try to change the fee correctly', async function () {
            const params = { txid: '0xffffffff', fee: '10000000' };
            const scope = nock(bgUrl).post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/changeFee`, params).reply(200);
            try {
                await wallet.changeFee({ txid: '0xffffffff', fee: '10000000' });
                throw '';
            }
            catch (error) {
                // test is successful if nock is consumed, HMAC errors expected
            }
            scope.isDone().should.be.True();
        });
        it('should try to change the fee correctly using eip1559', async function () {
            const params = {
                txid: '0xffffffff',
                eip1559: {
                    maxPriorityFeePerGas: '1000000000',
                    maxFeePerGas: '25000000000',
                },
            };
            const scope = nock(bgUrl).post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/changeFee`, params).reply(200);
            try {
                await wallet.changeFee(params);
                throw '';
            }
            catch (error) {
                // test is successful if nock is consumed, HMAC errors expected
            }
            scope.isDone().should.be.True();
        });
        it('should pass data parameter and amount: 0 when using sendTransaction', async function () {
            const path = `/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/tx/build`;
            const recipientAddress = '0x7db562c4dd465cc895761c56f83b6af0e32689ba';
            const recipients = [
                {
                    address: recipientAddress,
                    amount: 0,
                    data: '0x00110011',
                },
            ];
            const response = nock(bgUrl)
                .post(path, _.matches({ recipients })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            const nockKeyChain = nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[0]}`).reply(200, {});
            try {
                await ethWallet.send({
                    address: recipients[0].address,
                    data: recipients[0].data,
                    amount: recipients[0].amount,
                });
            }
            catch (e) {
                // test is successful if nock is consumed, HMAC errors expected
            }
            response.isDone().should.be.true();
            nockKeyChain.isDone().should.be.true();
        });
        it('should pass data parameter and amount: 0 when using sendMany', async function () {
            const path = `/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/tx/build`;
            const recipientAddress = '0x7db562c4dd465cc895761c56f83b6af0e32689ba';
            const recipients = [
                {
                    address: recipientAddress,
                    amount: 0,
                    data: '0x00110011',
                },
            ];
            const response = nock(bgUrl)
                .post(path, _.matches({ recipients })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            const nockKeyChain = nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[0]}`).reply(200, {});
            try {
                await ethWallet.sendMany({ recipients });
            }
            catch (e) {
                // test is successful if nock is consumed, HMAC errors expected
            }
            response.isDone().should.be.true();
            nockKeyChain.isDone().should.be.true();
        });
        it('should not pass recipients in sendMany when transaction type is fillNonce', async function () {
            const recipientAddress = '0x7db562c4dd465cc895761c56f83b6af0e32689ba';
            const recipients = [
                {
                    address: recipientAddress,
                    amount: 0,
                },
            ];
            const sendManyParams = { recipients, type: 'fillNonce', isTss: true, nonce: '13' };
            try {
                await ethWallet.sendMany(sendManyParams);
            }
            catch (e) {
                e.message.should.equal('cannot provide recipients for transaction type fillNonce');
                // test is successful if nock is consumed, HMAC errors expected
            }
        });
        it('should not pass receiveAddress in sendMany when TSS transaction type is transfer or transferToken', async function () {
            const recipientAddress = '0x7db562c4dd465cc895761c56f83b6af0e32689ba';
            const recipients = [
                {
                    address: recipientAddress,
                    amount: 0,
                },
            ];
            const errorMessage = 'cannot use receive address for TSS transactions of type transfer';
            const sendManyParamsReceiveAddressError = {
                receiveAddress: 'throw',
                recipients,
                type: 'transfer',
                isTss: true,
                nonce: '13',
            };
            const sendManyParams = { recipients, type: 'transfer', isTss: true, nonce: '13' };
            try {
                await ethWallet.sendMany(sendManyParamsReceiveAddressError);
            }
            catch (e) {
                e.message.should.equal(errorMessage);
            }
            try {
                await ethWallet.sendMany(sendManyParams);
            }
            catch (e) {
                e.message.should.not.equal(errorMessage);
            }
        });
        it('should throw error early if password is wrong', async function () {
            const recipientAddress = '0x7db562c4dd465cc895761c56f83b6af0e32689ba';
            const recipients = [
                {
                    address: recipientAddress,
                    amount: 0,
                },
            ];
            const errorMessage = `unable to decrypt keychain with the given wallet passphrase`;
            const sendManyParamsCorrectPassPhrase = {
                recipients,
                type: 'transfer',
                isTss: true,
                nonce: '13',
                walletPassphrase: sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_PASSPHRASE,
            };
            const nockKeychain = nock(bgUrl)
                .get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[0]}`)
                .times(2)
                .reply(200, {
                id: '598f606cd8fc24710d2ebad89dce86c2',
                pub: 'xpub661MyMwAqRbcFXDcWD2vxuebcT1ZpTF4Vke6qmMW8yzddwNYpAPjvYEEL5jLfyYXW2fuxtAxY8TgjPUJLcf1C8qz9N6VgZxArKX4EwB8rH5',
                ethAddress: '0x26a163ba9739529720c0914c583865dec0d37278',
                source: 'user',
                encryptedPrv: '{"iv":"15FsbDVI1zG9OggD8YX+Hg==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"hHbNH3Sz/aU=","ct":"WoNVKz7afiRxXI2w/YkzMdMyoQg/B15u1Q8aQgi96jJZ9wk6TIaSEc6bXFH3AHzD9MdJCWJQUpRhoQc/rgytcn69scPTjKeeyVMElGCxZdFVS/psQcNE+lue3//2Zlxj+6t1NkvYO+8yAezSMRBK5OdftXEjNQI="}',
                coinSpecific: {},
            });
            await ethWallet
                .sendMany({ ...sendManyParamsCorrectPassPhrase, walletPassphrase: 'wrongPassphrase' })
                .should.be.rejectedWith(errorMessage);
            try {
                const customSigningFunction = () => {
                    return 'mock';
                };
                // Should not validate passphrase if custom signing function is provided
                await ethWallet.sendMany({
                    ...sendManyParamsCorrectPassPhrase,
                    walletPassphrase: 'wrongPassphrase',
                    customSigningFunction,
                });
            }
            catch (e) {
                e.message.should.not.equal(errorMessage);
            }
            try {
                await ethWallet.sendMany({ ...sendManyParamsCorrectPassPhrase });
            }
            catch (e) {
                e.message.should.not.equal(errorMessage);
            }
            nockKeychain.isDone().should.be.true();
        });
    });
    describe('OFC Create Address', () => {
        let ofcWallet;
        let nocks;
        before(async function () {
            const walletDataOfc = {
                id: '5b34252f1bf349930e3400b00000000',
                coin: 'ofc',
                keys: [
                    '5b3424f91bf349930e34017800000000',
                    '5b3424f91bf349930e34017900000000',
                    '5b3424f91bf349930e34018000000000',
                ],
                coinSpecific: {},
                multisigType: 'onchain',
            };
            ofcWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('ofc'), walletDataOfc);
        });
        beforeEach(async function () {
            nocks = [
                nock(bgUrl).get(`/api/v2/ofc/key/${ofcWallet.keyIds()[0]}`).reply(200, {
                    id: ofcWallet.keyIds()[0],
                    pub: 'xpub661MyMwAqRbcFXDcWD2vxuebcT1ZpTF4Vke6qmMW8yzddwNYpAPjvYEEL5jLfyYXW2fuxtAxY8TgjPUJLcf1C8qz9N6VgZxArKX4EwB8rH5',
                    source: 'user',
                    encryptedPrv: '{"iv":"15FsbDVI1zG9OggD8YX+Hg==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"hHbNH3Sz/aU=","ct":"WoNVKz7afiRxXI2w/YkzMdMyoQg/B15u1Q8aQgi96jJZ9wk6TIaSEc6bXFH3AHzD9MdJCWJQUpRhoQc/rgytcn69scPTjKeeyVMElGCxZdFVS/psQcNE+lue3//2Zlxj+6t1NkvYO+8yAezSMRBK5OdftXEjNQI="}',
                    coinSpecific: {},
                }),
                nock(bgUrl).get(`/api/v2/ofc/key/${ofcWallet.keyIds()[1]}`).reply(200, {
                    id: ofcWallet.keyIds()[1],
                    pub: 'xpub661MyMwAqRbcGhSaXikpuTC9KU88Xx9LrjKSw1JKsvXNgabpTdgjy7LSovh9ZHhcqhAHQu7uthu7FguNGdcC4aXTKK5gqTcPe4WvLYRbCSG',
                    source: 'backup',
                    coinSpecific: {},
                }),
                nock(bgUrl).get(`/api/v2/ofc/key/${ofcWallet.keyIds()[2]}`).reply(200, {
                    id: ofcWallet.keyIds()[2],
                    pub: 'xpub661MyMwAqRbcFsXShW8R3hJsHNTYTUwzcejnLkY7KCtaJbDqcGkcBF99BrEJSjNZHeHveiYUrsAdwnjUMGwpgmEbiKcZWRuVA9HxnRaA3r3',
                    source: 'bitgo',
                    coinSpecific: {},
                }),
            ];
        });
        afterEach(async function () {
            nock.cleanAll();
            nocks.forEach((scope) => scope.isDone().should.be.true());
        });
        it('should correctly validate arguments to create address on OFC wallet', async function () {
            await ofcWallet.createAddress().should.be.rejectedWith('onToken is a mandatory parameter for OFC wallets');
            // @ts-expect-error test passing invalid number argument
            await ofcWallet.createAddress({ onToken: 42 }).should.be.rejectedWith('onToken has to be a string');
        });
        it('address creation with valid onToken argument succeeds', async function () {
            const scope = nock(bgUrl)
                .post(`/api/v2/ofc/wallet/${ofcWallet.id()}/address`, { onToken: 'ofctbtc' })
                .reply(200, {
                id: '638a48c6c3dba40007a3497fa49a080c',
                address: 'generated address',
                chain: 0,
                index: 1,
                coin: 'tbtc',
                wallet: ofcWallet.id,
            });
            const address = await ofcWallet.createAddress({ onToken: 'ofctbtc' });
            address.address.should.equal('generated address');
            scope.isDone().should.be.true();
        });
    });
    describe('TETH Create Address', () => {
        let ethWallet, nocks;
        const walletData = {
            id: '598f606cd8fc24710d2ebadb1d9459bb',
            coinSpecific: {
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
            },
            coin: 'teth',
            keys: [
                '598f606cd8fc24710d2ebad89dce86c2',
                '598f606cc8e43aef09fcb785221d9dd2',
                '5935d59cf660764331bafcade1855fd7',
            ],
        };
        beforeEach(async function () {
            ethWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('teth'), walletData);
            nocks = [
                nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[0]}`).reply(200, {
                    id: '598f606cd8fc24710d2ebad89dce86c2',
                    pub: 'xpub661MyMwAqRbcFXDcWD2vxuebcT1ZpTF4Vke6qmMW8yzddwNYpAPjvYEEL5jLfyYXW2fuxtAxY8TgjPUJLcf1C8qz9N6VgZxArKX4EwB8rH5',
                    ethAddress: '0x26a163ba9739529720c0914c583865dec0d37278',
                    source: 'user',
                    encryptedPrv: '{"iv":"15FsbDVI1zG9OggD8YX+Hg==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"hHbNH3Sz/aU=","ct":"WoNVKz7afiRxXI2w/YkzMdMyoQg/B15u1Q8aQgi96jJZ9wk6TIaSEc6bXFH3AHzD9MdJCWJQUpRhoQc/rgytcn69scPTjKeeyVMElGCxZdFVS/psQcNE+lue3//2Zlxj+6t1NkvYO+8yAezSMRBK5OdftXEjNQI="}',
                    coinSpecific: {},
                }),
                nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[1]}`).reply(200, {
                    id: '598f606cc8e43aef09fcb785221d9dd2',
                    pub: 'xpub661MyMwAqRbcGhSaXikpuTC9KU88Xx9LrjKSw1JKsvXNgabpTdgjy7LSovh9ZHhcqhAHQu7uthu7FguNGdcC4aXTKK5gqTcPe4WvLYRbCSG',
                    ethAddress: '0xa1a88a502274073b1bc4fe06ea0f5fe77e151b91',
                    source: 'backup',
                    coinSpecific: {},
                }),
                nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[2]}`).reply(200, {
                    id: '5935d59cf660764331bafcade1855fd7',
                    pub: 'xpub661MyMwAqRbcFsXShW8R3hJsHNTYTUwzcejnLkY7KCtaJbDqcGkcBF99BrEJSjNZHeHveiYUrsAdwnjUMGwpgmEbiKcZWRuVA9HxnRaA3r3',
                    ethAddress: '0x032821b7ea40ea5d446f47c29a0f777ee035aa10',
                    source: 'bitgo',
                    coinSpecific: {},
                }),
            ];
        });
        afterEach(async function () {
            nock.cleanAll();
            nocks.forEach((scope) => scope.isDone().should.be.true());
        });
        it('should correctly validate arguments to create address', async function () {
            let message = 'gasPrice has to be an integer or numeric string';
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ gasPrice: {} }).should.be.rejectedWith(message);
            await wallet.createAddress({ gasPrice: 'abc' }).should.be.rejectedWith(message);
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ gasPrice: null }).should.be.rejectedWith(message);
            message = 'chain has to be an integer';
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ chain: {} }).should.be.rejectedWith(message);
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ chain: 'abc' }).should.be.rejectedWith(message);
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ chain: null }).should.be.rejectedWith(message);
            message = 'count has to be a number between 1 and 250';
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ count: {} }).should.be.rejectedWith(message);
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ count: 'abc' }).should.be.rejectedWith(message);
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ count: null }).should.be.rejectedWith(message);
            await wallet.createAddress({ count: -1 }).should.be.rejectedWith(message);
            await wallet.createAddress({ count: 0 }).should.be.rejectedWith(message);
            await wallet.createAddress({ count: 251 }).should.be.rejectedWith(message);
            message = 'baseAddress has to be a string';
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ baseAddress: {} }).should.be.rejectedWith(message);
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ baseAddress: 123 }).should.be.rejectedWith(message);
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ baseAddress: null }).should.be.rejectedWith(message);
            message = 'allowSkipVerifyAddress has to be a boolean';
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ allowSkipVerifyAddress: {} }).should.be.rejectedWith(message);
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ allowSkipVerifyAddress: 123 }).should.be.rejectedWith(message);
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ allowSkipVerifyAddress: 'abc' }).should.be.rejectedWith(message);
            // @ts-expect-error checking type mismatch
            await wallet.createAddress({ allowSkipVerifyAddress: null }).should.be.rejectedWith(message);
            message = 'forwarderVersion has to be an integer 0, 1, 2, 3 or 4';
            await wallet.createAddress({ forwarderVersion: 5 }).should.be.rejectedWith(message);
            await wallet.createAddress({ forwarderVersion: -1 }).should.be.rejectedWith(message);
        });
        it('address creation with forwarder version 3 succeeds', async function () {
            const scope = nock(bgUrl)
                .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 3 })
                .reply(200, {
                id: '638a48c6c3dba40007a3497fa49a080c',
                address: '0x5e61b64f38f1b5f85078fb84b27394830b4c8e80',
                chain: 0,
                index: 1,
                coin: 'tpolygon',
                lastNonce: 0,
                wallet: '63785f95af7c760007cfae068c2f31ae',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2022-12-02T18:49:42.348Z',
                    txCount: 0,
                    pendingChainInitialization: false,
                    creationFailure: [],
                    salt: '0x1',
                    pendingDeployment: true,
                    forwarderVersion: 3,
                    isTss: true,
                },
            });
            const address = await ethWallet.createAddress({ chain: 0, forwarderVersion: 3 });
            address.coinSpecific.forwarderVersion.should.equal(3);
            scope.isDone().should.be.true();
        });
        it('address creation with forwarder version 3 fails due invalid address', async function () {
            const address = '0x5e61b6'; // invalid address
            nock(bgUrl)
                .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 3 })
                .reply(200, {
                id: '638a48c6c3dba40007a3497fa49a080c',
                address: address,
                chain: 0,
                index: 1,
                coin: 'tpolygon',
                lastNonce: 0,
                wallet: '63785f95af7c760007cfae068c2f31ae',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2022-12-02T18:49:42.348Z',
                    txCount: 0,
                    pendingChainInitialization: false,
                    creationFailure: [],
                    salt: '0x1',
                    pendingDeployment: true,
                    forwarderVersion: 3,
                    isTss: true,
                },
            });
            await ethWallet
                .createAddress({ chain: 0, forwarderVersion: 3 })
                .should.be.rejectedWith(`invalid address: ${address}`);
        });
        it('address creation with forwarder version 2 succeeds', async function () {
            const scope = nock(bgUrl)
                .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 2 })
                .reply(200, {
                id: '638a48c6c3dba40007a3497fa49a080c',
                address: '0x5e61b64f38f1b5f85078fb84b27394830b4c8e80',
                chain: 0,
                index: 1,
                coin: 'tpolygon',
                lastNonce: 0,
                wallet: '63785f95af7c760007cfae068c2f31ae',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2022-12-02T18:49:42.348Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0x1',
                    pendingDeployment: true,
                    forwarderVersion: 2,
                    isTss: true,
                },
            });
            const address = await ethWallet.createAddress({ chain: 0, forwarderVersion: 2 });
            address.coinSpecific.forwarderVersion.should.equal(2);
            scope.isDone().should.be.true();
        });
        it('verify address when pendingChainInitialization is true in case of eth v1 forwarder', async function () {
            const scope = nock(bgUrl)
                .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 1 })
                .reply(200, {
                id: '615c643a98a2a100068e023c639c0f74',
                address: '0x8c13cd0bb198858f628d5631ba4b2293fc08df49',
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                chain: 0,
                index: 3179,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-10-05T14:42:02.399Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0xc6b',
                    pendingDeployment: true,
                    forwarderVersion: 1,
                },
            });
            await ethWallet
                .createAddress({ chain: 0, forwarderVersion: 1 })
                .should.be.rejectedWith('address validation failure: expected 0x32a226cda14e352a47bf4b1658648d8037736f80 but got 0x8c13cd0bb198858f628d5631ba4b2293fc08df49');
            scope.isDone().should.be.true();
        });
        it('verify address when invalid baseAddress is passed', async function () {
            const scope = nock(bgUrl)
                .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 1 })
                .reply(200, {
                id: '615c643a98a2a100068e023c639c0f74',
                address: '0x32a226cda14e352a47bf4b1658648d8037736f80',
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                chain: 0,
                index: 3179,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-10-05T14:42:02.399Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0xc6b',
                    pendingDeployment: true,
                    forwarderVersion: 1,
                },
            });
            await ethWallet
                .createAddress({ chain: 0, forwarderVersion: 1, baseAddress: 'asgf' })
                .should.be.rejectedWith('invalid base address');
            scope.isDone().should.be.true();
        });
        it('verify address when incorrect baseAddress is passed', async function () {
            const scope = nock(bgUrl)
                .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 1 })
                .reply(200, {
                id: '615c643a98a2a100068e023c639c0f74',
                address: '0x32a226cda14e352a47bf4b1658648d8037736f80',
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                chain: 0,
                index: 3179,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-10-05T14:42:02.399Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0xc6b',
                    pendingDeployment: true,
                    forwarderVersion: 1,
                },
            });
            // incorrect address is generated while validating due to incorrect baseAddress
            await ethWallet
                .createAddress({ chain: 0, forwarderVersion: 1, baseAddress: '0x8c13cd0bb198858f628d5631ba4b2293fc08df49' })
                .should.be.rejectedWith('address validation failure: expected 0x36748926007790e7ee416c6485b32e00cfb177a3 but got 0x32a226cda14e352a47bf4b1658648d8037736f80');
            scope.isDone().should.be.true();
        });
        it('verify address when pendingChainInitialization is true  and allowSkipVerifyAddress is false in case of eth v0 forwarder', async function () {
            const scope = nock(bgUrl)
                .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 0 })
                .reply(200, {
                id: '615c643a98a2a100068e023c639c0f74',
                address: '0x32a26cda14e352a47bf4b1658648d8037736f80',
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                chain: 0,
                index: 3179,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-10-05T14:42:02.399Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0xc6b',
                    pendingDeployment: true,
                    forwarderVersion: 1,
                },
            });
            await ethWallet
                .createAddress({ chain: 0, forwarderVersion: 0, allowSkipVerifyAddress: false })
                .should.be.rejectedWith('address verification skipped for count = 1');
            scope.isDone().should.be.true();
        });
        it('verify address with allowSkipVerifyAddress set to false and eth v1 forwarder', async function () {
            const scope = nock(bgUrl)
                .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 1 })
                .reply(200, {
                id: '615c643a98a2a100068e023c639c0f74',
                address: '0x32a226cda14e352a47bf4b1658648d8037736f80',
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                chain: 0,
                index: 3179,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-10-05T14:42:02.399Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0xc6b',
                    pendingDeployment: true,
                    forwarderVersion: 0,
                },
            });
            const newAddress = await ethWallet.createAddress({
                chain: 0,
                forwarderVersion: 1,
                allowSkipVerifyAddress: false,
            });
            newAddress.index.should.equal(3179);
            scope.isDone().should.be.true();
        });
    });
    describe('Algorand tests', () => {
        let algoWallet;
        before(async () => {
            // This is not a real TALGO wallet
            const walletData = {
                id: '650204cf43d8b40007cd9e11a872ce65',
                coin: 'talgo',
                keys: [
                    '650204b78a75c90007790bce979ae34d',
                    '650204b766c56a00072956c08fb9cdf1',
                    '650204b8ccf1370007b32bb8155dfbec',
                ],
                coinSpecific: {
                    rootAddress: '2ULRGE64U7LTMT5M6REB7ORHX5GLJYWHTIV5EAXVLWQTTATVJDGM5KJMII',
                },
            };
            algoWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('talgo'), walletData);
        });
        it('Should build token enablement transactions', async () => {
            const params = {
                enableTokens: [
                    {
                        name: 'talgo:USDt-180447',
                    },
                ],
            };
            const txRequestNock = nock(bgUrl)
                .post(`/api/v2/${algoWallet.coin()}/wallet/${algoWallet.id()}/tx/build`)
                .reply((uri, body) => {
                const params = body;
                params.recipients.length.should.equal(1);
                params.recipients[0].tokenName.should.equal('talgo:USDt-180447');
                params.type.should.equal('enabletoken');
                should.not.exist(params.enableTokens);
                return [200, params];
            });
            await algoWallet.buildTokenEnablements(params);
            txRequestNock.isDone().should.equal(true);
        });
        afterEach(() => {
            nock.cleanAll();
        });
    });
    describe('Hedera tests', () => {
        let hbarWallet;
        before(async () => {
            // This is not a real THBAR wallet
            const walletData = {
                id: '598f606cd8fc24710d2ebadb1d9459bb',
                coin: 'thbar',
                keys: [
                    '598f606cd8fc24710d2ebad89dce86c2',
                    '598f606cc8e43aef09fcb785221d9dd2',
                    '5935d59cf660764331bafcade1855fd7',
                ],
                coinSpecific: {
                    baseAddress: '0.0.47841511',
                },
            };
            hbarWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('thbar'), walletData);
        });
        it('Should build token enablement transactions', async () => {
            const params = {
                enableTokens: [
                    {
                        name: 'thbar:usdc',
                    },
                ],
            };
            const txRequestNock = nock(bgUrl)
                .post(`/api/v2/${hbarWallet.coin()}/wallet/${hbarWallet.id()}/tx/build`)
                .reply((uri, body) => {
                const params = body;
                params.recipients.length.should.equal(1);
                params.recipients[0].tokenName.should.equal('thbar:usdc');
                params.type.should.equal('enabletoken');
                should.not.exist(params.enableTokens);
                return [200, params];
            });
            await hbarWallet.buildTokenEnablements(params);
            txRequestNock.isDone().should.equal(true);
        });
        afterEach(() => {
            nock.cleanAll();
        });
    });
    describe('Solana tests: ', () => {
        let solWallet;
        const passphrase = '#Bondiola1234';
        const solBitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
        solBitgo.initializeTestVars();
        const walletData = {
            id: '598f606cd8fc24710d2ebadb1d9459bb',
            coinSpecific: {
                baseAddress: '5f8WmC2uW9SAk7LMX2r4G1Bx8MMwx8sdgpotyHGodiZo',
                pendingChainInitialization: false,
                minimumFunding: 2447136,
                lastChainIndex: { 0: 0 },
            },
            coin: 'tsol',
            keys: [
                '598f606cd8fc24710d2ebad89dce86c2',
                '598f606cc8e43aef09fcb785221d9dd2',
                '5935d59cf660764331bafcade1855fd7',
            ],
            multisigType: 'tss',
        };
        before(async function () {
            solWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('tsol'), walletData);
            nock(bgUrl).get(`/api/v2/${solWallet.coin()}/key/${solWallet.keyIds()[0]}`).times(3).reply(200, {
                id: '598f606cd8fc24710d2ebad89dce86c2',
                pub: '5f8WmC2uW9SAk7LMX2r4G1Bx8MMwx8sdgpotyHGodiZo',
                source: 'user',
                encryptedPrv: '{"iv":"hNK3rg82P1T94MaueXFAbA==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"cV4wU4EzPjs=","ct":"9VZX99Ztsb6p75Cxl2lrcXBplmssIAQ9k7ZA81vdDYG4N5dZ36BQNWVfDoelj9O31XyJ+Xri0XKIWUzl0KKLfUERplmtNoOCn5ifJcZwCrOxpHZQe3AJ700o8Wmsrk5H"}',
                coinSpecific: {},
            });
            nock(bgUrl).get(`/api/v2/${solWallet.coin()}/key/${solWallet.keyIds()[1]}`).times(2).reply(200, {
                id: '598f606cc8e43aef09fcb785221d9dd2',
                pub: 'G1s43JTzNZzqhUn4aNpwgcc6wb9FUsZQD5JjffG6isyd',
                encryptedPrv: '{"iv":"UFrt/QlIUR1XeQafPBaAlw==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"7VPBYaJXPm8=","ct":"ajFKv2y8yaIBXQ39sAbBWcnbiEEzbjS4AoQtp5cXYqjeDRxt3aCxemPm22pnkJaCijFjJrMHbkmsNhNYzHg5aHFukN+nEAVssyNwHbzlhSnm8/BVN50yAdAAtWreh8cp"}',
                source: 'backup',
                coinSpecific: {},
            });
            nock(bgUrl).get(`/api/v2/${solWallet.coin()}/key/${solWallet.keyIds()[2]}`).times(2).reply(200, {
                id: '5935d59cf660764331bafcade1855fd7',
                pub: 'GH1LV1e9FdqGe8U2c8PMEcma3fDeh1ktcGVBrD3AuFqx',
                encryptedPrv: '{"iv":"iIuWOHIOErEDdiJn6g46mg==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"Rzh7RRJksj0=","ct":"rcNICUfp9FakT53l+adB6XKzS1vNTc0Qq9jAtqnxA+ScssiS4Q0l3sgG/0gDy5DaZKtXryKBDUvGsi7b/fYaFCUpAoZn/VZTOhOUN/mo7ZHb4OhOXL29YPPkiryAq9Cr"}',
                source: 'bitgo',
                coinSpecific: {},
            });
        });
        after(async function () {
            nock.cleanAll();
        });
        describe('prebuildAndSignTransaction: ', function () {
            // TODO (STLX-15018): fix test
            xit('should successfully sign a consolidation transfer', async function () {
                const txParams = {
                    prebuildTx: {
                        walletId: walletData.id,
                        txHex: 'AQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABAAIE9MWWV2ct01mg5Gm4EqcJ9SAn2XuD+FuAHcHFTkc1Tgut3DgTsiSgTQ0dmzj5JJg6qYTpn8FxOYPFCFTMoZi46gAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABUpTWpkpIQZNJOhxYNo4fHw1td28kruB5B+oQEEFRI0Qc+q0Zg6OOpV8eCDVLfYziox7YBA7+QPLX4IRhDCSKwICAgABDAIAAACghgEAAAAAAAMAFVRlc3QgaW50ZWdyYXRpb24gbWVtbw==',
                        txInfo: {
                            feePayer: 'HUVE5NfJyGfU1djZsVLA6fxSTS1E2iRqcTRVNC9K2z7c',
                            lamportsPerSignature: 5000,
                            nonce: '27E3MXFvXMUNYeMJeX1pAbERGsJfUbkaZTfgMgpmNN5g',
                            numSignatures: 0,
                            instructionsData: [
                                {
                                    type: 'Transfer',
                                    params: {
                                        fromAddress: 'HUVE5NfJyGfU1djZsVLA6fxSTS1E2iRqcTRVNC9K2z7c',
                                        toAddress: 'ChgJ5tgDwBUsk9RNMm2iLiwP8RodwgZ6uqrC5paJsXVT',
                                        amount: '100000',
                                    },
                                },
                                {
                                    type: 'Memo',
                                    params: {
                                        memo: 'Test integration memo',
                                    },
                                },
                            ],
                        },
                        buildParams: {
                            memo: {
                                type: 'Memo',
                                value: 'Test integration memo',
                            },
                            recipients: [
                                {
                                    address: 'ChgJ5tgDwBUsk9RNMm2iLiwP8RodwgZ6uqrC5paJsXVT',
                                    amount: '100000',
                                },
                            ],
                            type: 'transfer',
                        },
                        consolidateId: '1234',
                        consolidationDetails: {
                            senderAddressIndex: 1,
                        },
                    },
                    walletPassphrase: passphrase,
                };
                // Build and sign the transaction
                const preBuiltSignedTx = await solWallet.prebuildAndSignTransaction(txParams);
                preBuiltSignedTx.should.have.property('txHex');
            });
        });
        it('Should build token enablement transactions correctly', async function () {
            const params = {
                enableTokens: [{ name: 'tsol:usdc' }, { name: 'tsol:srm' }, { name: 'tsol:gmt' }],
            };
            const txRequestNock = nock(bgUrl)
                .post(`/api/v2/wallet/${solWallet.id()}/txrequests`)
                .reply((url, body) => {
                const bodyParams = body;
                bodyParams.intent.intentType.should.equal('enableToken');
                bodyParams.intent.recipients.length.should.equal(0);
                bodyParams.intent.enableTokens.should.deepEqual(params.enableTokens);
                return [
                    200,
                    {
                        apiVersion: 'full',
                        transactions: [
                            {
                                unsignedTx: {
                                    serializedTxHex: 'fake transaction',
                                    feeInfo: 'fake fee info',
                                },
                            },
                        ],
                    },
                ];
            });
            await solWallet.buildTokenEnablements(params);
            txRequestNock.isDone().should.equal(true);
        });
    });
    describe('Accelerate Transaction', function () {
        it('fails if acceleration ids are not passed', async function () {
            await wallet.accelerateTransaction({}).should.be.rejectedWith({ code: 'cpfptxids_or_rbftxids_required' });
        });
        it('fails if cpfpTxIds is not an array', async function () {
            // @ts-expect-error checking type mismatch
            await wallet.accelerateTransaction({ cpfpTxIds: {} }).should.be.rejectedWith({ code: 'cpfptxids_not_array' });
        });
        it('fails if cpfpTxIds is not of length 1', async function () {
            await wallet.accelerateTransaction({ cpfpTxIds: [] }).should.be.rejectedWith({ code: 'cpfptxids_not_array' });
            await wallet
                .accelerateTransaction({ cpfpTxIds: ['id1', 'id2'] })
                .should.be.rejectedWith({ code: 'cpfptxids_not_array' });
        });
        it('fails if cpfpFeeRate is not passed and neither is noCpfpFeeRate', async function () {
            await wallet.accelerateTransaction({ cpfpTxIds: ['id'] }).should.be.rejectedWith({ code: 'cpfpfeerate_not_set' });
        });
        it('fails if cpfpFeeRate is not an integer', async function () {
            await wallet
                // @ts-expect-error checking type mismatch
                .accelerateTransaction({ cpfpTxIds: ['id'], cpfpFeeRate: 'one' })
                .should.be.rejectedWith({ code: 'cpfpfeerate_not_nonnegative_integer' });
        });
        it('fails if cpfpFeeRate is negative', async function () {
            await wallet
                .accelerateTransaction({ cpfpTxIds: ['id'], cpfpFeeRate: -1 })
                .should.be.rejectedWith({ code: 'cpfpfeerate_not_nonnegative_integer' });
        });
        it('fails if maxFee is not passed and neither is noMaxFee', async function () {
            await wallet
                .accelerateTransaction({ cpfpTxIds: ['id'], noCpfpFeeRate: true })
                .should.be.rejectedWith({ code: 'maxfee_not_set' });
        });
        it('fails if maxFee is not an integer', async function () {
            await wallet
                // @ts-expect-error checking type mismatch
                .accelerateTransaction({ cpfpTxIds: ['id'], noCpfpFeeRate: true, maxFee: 'one' })
                .should.be.rejectedWith({ code: 'maxfee_not_nonnegative_integer' });
        });
        it('fails if maxFee is negative', async function () {
            await wallet
                .accelerateTransaction({ cpfpTxIds: ['id'], noCpfpFeeRate: true, maxFee: -1 })
                .should.be.rejectedWith({ code: 'maxfee_not_nonnegative_integer' });
        });
        it('fails if both rbfTxids and cpfpTxids is set', async function () {
            await wallet
                .accelerateTransaction({ cpfpTxIds: ['id1'], rbfTxIds: ['id2'] })
                .should.be.rejectedWith({ code: 'cannot_specify_both_cpfp_and_rbf_txids' });
        });
        it('fails if rbfTxIds is set but feeMultiplier is missing', async function () {
            await wallet
                .accelerateTransaction({ rbfTxIds: ['id'] })
                .should.be.rejectedWith({ code: 'feemultiplier_not_set' });
        });
        it('fails if fee multiplier is less than or equal to 1', async function () {
            await wallet
                .accelerateTransaction({ rbfTxIds: ['id'], feeMultiplier: 1 })
                .should.be.rejectedWith({ code: 'feemultiplier_greater_than_one' });
            await wallet
                .accelerateTransaction({ rbfTxIds: ['id2'], feeMultiplier: 0.5 })
                .should.be.rejectedWith({ code: 'feemultiplier_greater_than_one' });
        });
        it('submits a transaction with all cpfp specific parameters', async function () {
            const params = {
                cpfpTxIds: ['id'],
                cpfpFeeRate: 1,
                maxFee: 1,
            };
            const prebuildReturn = Object.assign({ txHex: '123' }, params);
            const prebuildStub = sinon.stub(wallet, 'prebuildAndSignTransaction').resolves(prebuildReturn);
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/send`;
            nock(bgUrl).post(path, _.matches(prebuildReturn)).reply(200);
            await wallet.accelerateTransaction(params);
            prebuildStub.should.have.been.calledOnceWith(params);
            sinon.restore();
        });
    });
    describe('fanout input maxNumInputsToUse and unspents verification', function () {
        const address = '5b34252f1bf349930e34020a';
        const maxNumInputsToUse = 2;
        const unspents = [
            'cc30565750e2aeb818625aaedaf89db5c614e5977b9645cee1d7289f616fb1d8:0',
            '8c45164787a954ab07864af9b05b34fbde3a8e430a8c65b0e60e4e543d8e1b6c:2',
        ];
        let basecoin;
        let wallet;
        before(async function () {
            basecoin = bitgo.coin('tbtc');
            const walletData = {
                id: '5b34252f1bf349930e34020a',
                coin: 'tbtc',
                keys: ['5b3424f91bf349930e340175'],
            };
            wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
        });
        it('should pass maxNumInputsToUse parameter when calling fanout unspents', async function () {
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/fanoutUnspents`;
            const response = nock(bgUrl)
                .post(path, _.matches({ maxNumInputsToUse })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            try {
                await wallet.fanoutUnspents({ address, maxNumInputsToUse });
            }
            catch (e) {
                // the fanoutUnspents method will probably throw an exception for not having all of the correct nocks
                // we only care about /fanoutUnspents and whether maxNumInputsToUse is an allowed parameter
            }
            response.isDone().should.be.true();
        });
        it('should pass unspents parameter when calling fanout unspents', async function () {
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/fanoutUnspents`;
            const response = nock(bgUrl)
                .post(path, _.matches({ unspents })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            try {
                await wallet.fanoutUnspents({ address, unspents });
            }
            catch (e) {
                // the fanoutUnspents method will probably throw an exception for not having all of the correct nocks
                // we only care about /fanoutUnspents and whether unspents is an allowed parameter
            }
            response.isDone().should.be.true();
        });
        it('should only build tx (not sign/send) while fanning out unspents', async function () {
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/fanoutUnspents`;
            const response = nock(bgUrl).post(path, _.matches({ unspents })).reply(200);
            const unusedNocks = nock(bgUrl);
            unusedNocks.get(`/api/v2/${wallet.coin()}/key/${wallet.keyIds()[0]}`).reply(200);
            unusedNocks.post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/send`).reply(200);
            try {
                await wallet.fanoutUnspents({ address, unspents }, sdk_core_1.ManageUnspentsOptions.BUILD_ONLY);
            }
            catch (e) {
                // the fanoutUnspents method will probably throw an exception for not having all of the correct nocks
                // we only care about /fanoutUnspents and whether unspents is an allowed parameter
            }
            response.isDone().should.be.true();
            unusedNocks.pendingMocks().length.should.eql(2);
            nock.cleanAll();
        });
    });
    describe('manage unspents', function () {
        let rootWalletKey;
        let walletPassphrase;
        let basecoin;
        let wallet;
        let keysObj;
        before(async function () {
            rootWalletKey = (0, util_1.getDefaultWalletKeys)();
            walletPassphrase = 'fixthemoneyfixtheworld';
            keysObj = (0, util_1.toKeychainObjects)(rootWalletKey, walletPassphrase);
            basecoin = bitgo.coin('tbtc');
            const walletData = {
                id: '5b34252f1bf349930e34020a',
                coin: 'tbtc',
                keys: keysObj.map((k) => k.id),
            };
            wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
        });
        it('should pass for bulk consolidating unspents', async function () {
            const psbts = ['p2wsh', 'p2shP2wsh'].map((scriptType) => utxoLib.testutil.constructPsbt([{ scriptType, value: BigInt(1000) }], [{ scriptType, value: BigInt(900) }], basecoin.network, rootWalletKey, 'unsigned'));
            const txHexes = psbts.map((psbt) => ({ txHex: psbt.toHex() }));
            const nocks = [];
            nocks.push(nock(bgUrl).post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/consolidateUnspents`).reply(200, txHexes));
            nocks.push(...keysObj.map((k, i) => nock(bgUrl).get(`/api/v2/${wallet.coin()}/key/${wallet.keyIds()[i]}`).reply(200, k)));
            nocks.push(...psbts.map((psbt) => nock(bgUrl)
                .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/send`, _.matches({ txHex: psbt.signAllInputsHD(rootWalletKey.user).toHex() }))
                .reply(200)));
            await wallet.consolidateUnspents({ bulk: true, walletPassphrase });
            nocks.forEach((n) => {
                console.log(n);
                n.isDone().should.be.true();
            });
        });
        it('should pass for single consolidating unspents', async function () {
            const psbt = utxoLib.testutil.constructPsbt([{ scriptType: 'p2wsh', value: BigInt(1000) }], [{ scriptType: 'p2shP2wsh', value: BigInt(900) }], basecoin.network, rootWalletKey, 'unsigned');
            const nocks = [];
            nocks.push(nock(bgUrl)
                .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/consolidateUnspents`)
                .reply(200, { txHex: psbt.toHex() }));
            nocks.push(...keysObj.map((k, i) => nock(bgUrl).get(`/api/v2/${wallet.coin()}/key/${wallet.keyIds()[i]}`).reply(200, k)));
            nocks.push(nock(bgUrl)
                .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/send`, _.matches({ txHex: psbt.signAllInputsHD(rootWalletKey.user).toHex() }))
                .reply(200));
            await wallet.consolidateUnspents({ walletPassphrase });
            nocks.forEach((n) => {
                n.isDone().should.be.true();
            });
        });
    });
    describe('max recipient', function () {
        const address = '5b34252f1bf349930e34020a';
        const recipients = [
            {
                address,
                amount: 'max',
            },
        ];
        let basecoin;
        let wallet;
        before(async function () {
            basecoin = bitgo.coin('tbtc');
            const walletData = {
                id: '5b34252f1bf349930e34020a',
                coin: 'tbtc',
                keys: ['5b3424f91bf349930e340175'],
            };
            wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
        });
        it('should pass maxFeeRate parameter when building transactions', async function () {
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`;
            const response = nock(bgUrl)
                .post(path, _.matches({
                recipients,
            })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            try {
                await wallet.prebuildTransaction({ recipients });
            }
            catch (e) {
                // the prebuildTransaction method will probably throw an exception for not having all of the correct nocks
                // we only care about /tx/build and whether maxFeeRate is an allowed parameter
            }
            response.isDone().should.be.true();
        });
    });
    describe('maxFeeRate verification', function () {
        const address = '5b34252f1bf349930e34020a';
        const recipients = [
            {
                address,
                amount: 0,
            },
        ];
        const maxFeeRate = 10000;
        let basecoin;
        let wallet;
        before(async function () {
            basecoin = bitgo.coin('tbtc');
            const walletData = {
                id: '5b34252f1bf349930e34020a',
                coin: 'tbtc',
                keys: ['5b3424f91bf349930e340175'],
            };
            wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
        });
        it('should pass maxFeeRate parameter when building transactions', async function () {
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`;
            const response = nock(bgUrl)
                .post(path, _.matches({ recipients, maxFeeRate })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            try {
                await wallet.prebuildTransaction({ recipients, maxFeeRate });
            }
            catch (e) {
                // the prebuildTransaction method will probably throw an exception for not having all of the correct nocks
                // we only care about /tx/build and whether maxFeeRate is an allowed parameter
            }
            response.isDone().should.be.true();
        });
        it('should pass maxFeeRate parameter when consolidating unspents', async function () {
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/consolidateUnspents`;
            const response = nock(bgUrl)
                .post(path, _.matches({ maxFeeRate })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            nock(bgUrl).get(`/api/v2/${wallet.coin()}/key/${wallet.keyIds()[0]}`).reply(200);
            try {
                await wallet.consolidateUnspents({ recipients, maxFeeRate });
            }
            catch (e) {
                // the consolidateUnspents method will probably throw an exception for not having all of the correct nocks
                // we only care about /consolidateUnspents and whether maxFeeRate is an allowed parameter
            }
            response.isDone().should.be.true();
        });
        it('should only build tx (not sign/send) while consolidating unspents', async function () {
            const toBeUsedNock = nock(bgUrl);
            toBeUsedNock.post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/consolidateUnspents`).reply(200);
            const unusedNocks = nock(bgUrl);
            unusedNocks.get(`/api/v2/${wallet.coin()}/key/${wallet.keyIds()[0]}`).reply(200);
            unusedNocks.post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/send`).reply(200);
            await wallet.consolidateUnspents({ recipients }, sdk_core_1.ManageUnspentsOptions.BUILD_ONLY);
            toBeUsedNock.isDone().should.be.true();
            unusedNocks.pendingMocks().length.should.eql(2);
            nock.cleanAll();
        });
        it('should pass maxFeeRate parameter when calling sweep wallets', async function () {
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/sweepWallet`;
            const response = nock(bgUrl)
                .post(path, _.matches({ address, maxFeeRate })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            try {
                await wallet.sweep({ address, maxFeeRate });
            }
            catch (e) {
                // the sweep method will probably throw an exception for not having all of the correct nocks
                // we only care about /sweepWallet and whether maxFeeRate is an allowed parameter
            }
            response.isDone().should.be.true();
        });
        it('should pass maxFeeRate parameter when calling fanout unspents', async function () {
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/fanoutUnspents`;
            const response = nock(bgUrl)
                .post(path, _.matches({ maxFeeRate })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            try {
                await wallet.fanoutUnspents({ address, maxFeeRate });
            }
            catch (e) {
                // the fanoutUnspents method will probably throw an exception for not having all of the correct nocks
                // we only care about /fanoutUnspents and whether maxFeeRate is an allowed parameter
            }
            response.isDone().should.be.true();
        });
    });
    describe('allowPartialSweep verification', function () {
        const address = '5b34252f1bf349930e34020a';
        const allowPartialSweep = true;
        let basecoin;
        let wallet;
        before(async function () {
            basecoin = bitgo.coin('tbtc');
            const walletData = {
                id: '5b34252f1bf349930e34020a',
                coin: 'tbtc',
                keys: ['5b3424f91bf349930e340175'],
            };
            wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
        });
        it('should pass allowPartialSweep parameter when calling sweep wallets', async function () {
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/sweepWallet`;
            const response = nock(bgUrl)
                .post(path, _.matches({ address, allowPartialSweep })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            try {
                await wallet.sweep({ address, allowPartialSweep });
            }
            catch (e) {
                // the sweep method will probably throw an exception for not having all of the correct nocks
                // we only care about /sweepWallet and whether allowPartialSweep is an allowed parameter
            }
            response.isDone().should.be.true();
        });
    });
    describe('sweep wallet', function () {
        let basecoin;
        let wallet;
        before(async function () {
            basecoin = bitgo.coin('ttrx');
            const walletData = {
                id: '5b34252f1bf349930e34020a',
                coin: 'ttrx',
                keys: ['5b3424f91bf349930e340175'],
            };
            wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
        });
        it('should use maximum spendable balance of wallet to sweep funds ', async function () {
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/maximumSpendable`;
            const response = nock(bgUrl).get(path).reply(200, {
                coin: 'ttrx',
                maximumSpendable: 65000,
            });
            const body = {
                coin: 'ttrx',
                address: '2MwvR24yqym2CgHMp7zwvdeqBa4F8KTqunS',
            };
            try {
                await wallet.sweep(body);
            }
            catch (e) {
                // the sweep method will probably throw an exception for not having all of the correct nocks
                // we only care about maximum spendable balance being used to sweep funds
            }
            response.isDone().should.be.true();
        });
    });
    describe('Transaction prebuilds', function () {
        let ethWallet;
        before(async function () {
            const walletData = {
                id: '598f606cd8fc24710d2ebadb1d9459bb',
                coin: 'teth',
                keys: [
                    '598f606cd8fc24710d2ebad89dce86c2',
                    '598f606cc8e43aef09fcb785221d9dd2',
                    '5935d59cf660764331bafcade1855fd7',
                ],
            };
            ethWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('teth'), walletData);
        });
        it('should return reqId if it was passed in the params', async function () {
            const params = { offlineVerification: true };
            const scope = nock(bgUrl)
                .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, tbtcHotWalletDefaultParams)
                .query(params)
                .reply(200, {});
            const blockHeight = 100;
            sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);
            sinon.stub(basecoin, 'postProcessPrebuild').resolves({});
            const txRequest = await wallet.prebuildTransaction({ ...params, reqId: reqId });
            txRequest.reqId?.should.containEql(reqId);
            scope.done();
        });
        it('should pass offlineVerification=true query param if passed truthy value', async function () {
            const params = { offlineVerification: true };
            const scope = nock(bgUrl)
                .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, tbtcHotWalletDefaultParams)
                .query(params)
                .reply(200, {});
            const blockHeight = 100;
            const blockHeightStub = sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);
            const postProcessStub = sinon.stub(basecoin, 'postProcessPrebuild').resolves({});
            await wallet.prebuildTransaction(params);
            blockHeightStub.should.have.been.calledOnce();
            postProcessStub.should.have.been.calledOnceWith({
                blockHeight: 100,
                wallet: wallet,
                buildParams: tbtcHotWalletDefaultParams,
            });
            scope.done();
            blockHeightStub.restore();
            postProcessStub.restore();
        });
        it('should not pass the offlineVerification query param if passed a falsey value', async function () {
            const params = { offlineVerification: false };
            nock(bgUrl)
                .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, tbtcHotWalletDefaultParams)
                .query({})
                .reply(200, {});
            const blockHeight = 100;
            const blockHeightStub = sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);
            const postProcessStub = sinon.stub(basecoin, 'postProcessPrebuild').resolves({});
            await wallet.prebuildTransaction(params);
            blockHeightStub.should.have.been.calledOnce();
            postProcessStub.should.have.been.calledOnceWith({
                blockHeight: 100,
                wallet: wallet,
                buildParams: tbtcHotWalletDefaultParams,
            });
            blockHeightStub.restore();
            postProcessStub.restore();
        });
        it('should pass script outputs with the proper structure to wallet platform', async function () {
            const script = '6a11223344556677889900';
            nock(bgUrl)
                .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, {
                ...tbtcHotWalletDefaultParams,
                recipients: [{ script, amount: 1e6 }],
            })
                .query({})
                .reply(200, {});
            const blockHeight = 100;
            const blockHeightStub = sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);
            const postProcessStub = sinon.stub(basecoin, 'postProcessPrebuild').resolves({});
            await wallet.prebuildTransaction({ recipients: [{ address: `scriptPubKey:${script}`, amount: 1e6 }] });
            blockHeightStub.should.have.been.calledOnce();
            postProcessStub.should.have.been.calledOnceWith({
                blockHeight: 100,
                wallet: wallet,
                buildParams: { ...tbtcHotWalletDefaultParams, recipients: [{ script, amount: 1e6 }] },
            });
            blockHeightStub.restore();
            postProcessStub.restore();
        });
        it('prebuild should call build and getLatestBlockHeight for utxo coins', async function () {
            const params = {};
            nock(bgUrl)
                .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, tbtcHotWalletDefaultParams)
                .query(params)
                .reply(200, {});
            const blockHeight = 100;
            const blockHeightStub = sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);
            const postProcessStub = sinon.stub(basecoin, 'postProcessPrebuild').resolves({});
            await wallet.prebuildTransaction(params);
            blockHeightStub.should.have.been.calledOnce();
            postProcessStub.should.have.been.calledOnceWith({
                blockHeight: 100,
                wallet: wallet,
                buildParams: tbtcHotWalletDefaultParams,
            });
            blockHeightStub.restore();
            postProcessStub.restore();
        });
        it('prebuild should not have changeAddressType array in post body when changeAddressType is defined', async function () {
            const expectedBuildPostBodyParams = {
                changeAddressType: 'p2trMusig2',
                txFormat: 'psbt',
            };
            nock(bgUrl)
                .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, expectedBuildPostBodyParams)
                .query({})
                .reply(200, {});
            const blockHeight = 100;
            const blockHeightStub = sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);
            const postProcessStub = sinon.stub(basecoin, 'postProcessPrebuild').resolves({});
            await wallet.prebuildTransaction({ changeAddressType: 'p2trMusig2' });
            blockHeightStub.should.have.been.calledOnce();
            postProcessStub.should.have.been.calledOnceWith({
                blockHeight: 100,
                wallet: wallet,
                buildParams: expectedBuildPostBodyParams,
            });
            blockHeightStub.restore();
            postProcessStub.restore();
        });
        it('prebuild should call build but not getLatestBlockHeight for account coins', async function () {
            ['txrp', 'txlm', 'teth'].forEach(async function (coin) {
                const accountcoin = bitgo.coin(coin);
                const walletData = {
                    id: '5b34252f1bf349930e34021a',
                    coin,
                    keys: ['5b3424f91bf349930e340175'],
                };
                const accountWallet = new sdk_core_1.Wallet(bitgo, accountcoin, walletData);
                const params = {};
                nock(bgUrl)
                    .post(`/api/v2/${accountWallet.coin()}/wallet/${accountWallet.id()}/tx/build`)
                    .query(params)
                    .reply(200, {});
                const postProcessStub = sinon.stub(accountcoin, 'postProcessPrebuild').resolves({});
                await accountWallet.prebuildTransaction(params);
                postProcessStub.should.have.been.calledOnceWith({
                    wallet: accountWallet,
                    buildParams: {},
                });
                postProcessStub.restore();
            });
        });
        it('should have isBatch = true in the txPrebuild if txParams has more than one recipient', async function () {
            const txParams = {
                recipients: [
                    { amount: '1000000000000000', address: address1 },
                    { amount: '1000000000000000', address: address2 },
                ],
                walletContractAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                walletPassphrase: 'moon',
            };
            const totalAmount = '2000000000000000';
            nock(bgUrl)
                .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/tx/build`, _.matches({ recipients: txParams.recipients }))
                .reply(200, {
                recipients: [
                    {
                        address: '0xc0aaf2649e7b0f3950164681eca2b1a8f654a478',
                        amount: '2000000000000000',
                        data: '0xc00c4e9e000000000000000000000000000000000000000000000000000000000000004000000000000000000000000000000000000000000000000000000000000000a00000000000000000000000000000000000000000000000000000000000000002000000000000000000000000174cfd823af8ce27ed0afee3fcf3c3ba259116be0000000000000000000000007e85bdc27c050e3905ebf4b8e634d9ad6edd0de6000000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000038d7ea4c6800000000000000000000000000000000000000000000000000000038d7ea4c68000',
                    },
                ],
                nextContractSequenceId: 10896,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: true,
                coin: 'teth',
            });
            const txPrebuild = await ethWallet.prebuildTransaction(txParams);
            txPrebuild.isBatch.should.equal(true);
            txPrebuild.recipients[0].address.should.equal(bitgo.coin('teth').staticsCoin.network.batcherContractAddress);
            txPrebuild.recipients[0].amount.should.equal(totalAmount);
        });
        it('should have isBatch = false and hopTransaction field should not be there in the txPrebuild  for normal eth tx', async function () {
            const txParams = {
                recipients: [{ amount: '1000000000000000', address: address1 }],
                walletContractAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                walletPassphrase: 'moon',
            };
            nock(bgUrl)
                .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/tx/build`, _.matches({ recipients: txParams.recipients }))
                .reply(200, {
                recipients: [
                    {
                        amount: '1000000000000000',
                        address: '0x174cfd823af8ce27ed0afee3fcf3c3ba259116be',
                    },
                ],
                nextContractSequenceId: 10897,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: false,
                coin: 'teth',
            });
            const txPrebuild = await ethWallet.prebuildTransaction(txParams);
            txPrebuild.isBatch.should.equal(false);
            txPrebuild.should.not.have.property('hopTransaction');
            txPrebuild.recipients[0].address.should.equal(address1);
            txPrebuild.recipients[0].amount.should.equal('1000000000000000');
        });
        it('should pass unspent reservation parameter through when building transactions', async function () {
            const reservation = {
                expireTime: '2029-08-12',
            };
            const recipients = [
                {
                    address: 'aaa',
                    amount: '1000',
                },
            ];
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`;
            const response = nock(bgUrl)
                .post(path, _.matches({ recipients, reservation })) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            try {
                await wallet.prebuildTransaction({ recipients, reservation });
            }
            catch (e) {
                // the prebuildTransaction method will probably throw an exception for not having all of the correct nocks
                // we only care about /tx/build and whether reservation is an allowed parameter
            }
            response.isDone().should.be.true();
        });
        it('should pass gas limit parameter through when building transaction for sui', async function () {
            const params = { gasLimit: 100 };
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`;
            const response = nock(bgUrl)
                .post(path, _.matches(params)) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200);
            try {
                await wallet.prebuildTransaction(params);
            }
            catch (e) {
                // the prebuildTransaction method will probably throw an exception for not having all of the correct nocks
                // we only care about /tx/build and whether reservation is an allowed parameter
            }
            response.isDone().should.be.true();
        });
    });
    describe('Maximum Spendable', function maximumSpendable() {
        let bgUrl;
        before(async function () {
            nock.pendingMocks().should.be.empty();
            bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
        });
        it('arguments', async function () {
            const optionalParams = {
                limit: 25,
                minValue: '0',
                maxValue: '9999999999999',
                minHeight: 0,
                minConfirms: 2,
                enforceMinConfirmsForChange: false,
                feeRate: 10000,
                maxFeeRate: 100000,
                recipientAddress: '2NCUFDLiUz9CVnmdVqQe9acVonoM89e76df',
            };
            // The actual api request will only send strings, but the SDK function expects numbers for some values
            const apiParams = _.mapValues(optionalParams, (param) => String(param));
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/maximumSpendable`;
            const response = nock(bgUrl)
                .get(path)
                .query(_.matches(apiParams)) // use _.matches to do a partial match on request body object instead of strict matching
                .reply(200, {
                coin: 'tbch',
                maximumSpendable: 65000,
            });
            try {
                await wallet.maximumSpendable(optionalParams);
            }
            catch (e) {
                // test is successful if nock is consumed
            }
            response.isDone().should.be.true();
        });
    });
    describe('Wallet Sharing', function () {
        it('should share to cold wallet without passing skipKeychain', async function () {
            const userId = '123';
            const email = 'shareto@sdktest.com';
            const permissions = 'view,spend';
            const getSharingKeyNock = nock(bgUrl).post('/api/v1/user/sharingkey', { email }).reply(200, { userId });
            const getKeyNock = nock(bgUrl)
                .get(`/api/v2/tbtc/key/${coldWallet.keyIds()[0]}`)
                .reply(200, {})
                .get(`/api/v2/tbtc/key/${coldWallet.keyIds()[1]}`)
                .reply(200, {})
                .get(`/api/v2/tbtc/key/${coldWallet.keyIds()[2]}`)
                .reply(200, {});
            const createShareNock = nock(bgUrl)
                .post(`/api/v2/tbtc/wallet/${coldWallet.id()}/share`, {
                user: userId,
                permissions,
                skipKeychain: true,
            })
                .reply(200, {});
            await coldWallet.shareWallet({ email, permissions });
            getSharingKeyNock.isDone().should.be.True();
            getKeyNock.isDone().should.be.True();
            createShareNock.isDone().should.be.True();
        });
        describe('Hot Wallet Sharing', function () {
            const userId = '123';
            const email = 'shareto@sdktest.com';
            const permissions = 'view,spend';
            const toKeychain = utxoLib.bip32.fromSeed(Buffer.from('deadbeef02deadbeef02deadbeef02deadbeef02', 'hex'));
            const path = 'm/999999/1/1';
            const pubkey = toKeychain.derivePath(path).publicKey.toString('hex');
            const walletPassphrase = 'bitgo1234';
            const pub = 'Zo1ggzTUKMY5bYnDvT5mtVeZxzf2FaLTbKkmvGUhUQk';
            const lightningCoin = bitgo.coin('tlnbtc');
            const lightningWalletData = {
                id: '5b34252f1bf349930e34020a00000001',
                coin: 'tlnbtc',
                keys: ['5b3424f91bf349930e34017500000001'],
                coinSpecific: { keys: ['5b3424f91bf349930e34017600000000', '5b3424f91bf349930e34017700000000'] },
                type: 'hot',
            };
            const lightningWallet = new sdk_core_1.Wallet(bitgo, lightningCoin, lightningWalletData);
            for (const hotWallet of [wallet, lightningWallet]) {
                it(`should use keychain pub to share ${hotWallet.coin()} hot wallet`, async function () {
                    const getSharingKeyNock = nock(bgUrl)
                        .post('/api/v1/user/sharingkey', { email })
                        .reply(200, { userId, pubkey, path });
                    const getKeyNocks = [];
                    if (hotWallet.baseCoin.getFamily() === 'lnbtc') {
                        for (let i = 0; i < 2; i++) {
                            const keyId = lightningWalletData.coinSpecific.keys[i];
                            const getKeyNock = nock(bgUrl)
                                .get(`/api/v2/tlnbtc/key/${keyId}`)
                                .reply(200, {
                                id: keyId,
                                pub: i === 0 ? pub : 'Zo1ggzTUKMY5bYnDvT5mtVeZxzf2FaLTbKkmvGUhUQm',
                                source: 'user',
                                encryptedPrv: bitgo.encrypt({ input: 'xprv' + i, password: walletPassphrase }),
                                coinSpecific: i === 0
                                    ? { [hotWallet.baseCoin.getChain()]: { purpose: 'userAuth' } }
                                    : { [hotWallet.baseCoin.getChain()]: { purpose: 'nodeAuth' } },
                            });
                            getKeyNocks.push(getKeyNock);
                        }
                    }
                    else {
                        const getKeyNock = nock(bgUrl)
                            .get(`/api/v2/tbtc/key/${wallet.keyIds()[0]}`)
                            .reply(200, {
                            id: wallet.keyIds()[0],
                            pub,
                            source: 'user',
                            encryptedPrv: bitgo.encrypt({ input: 'xprv1', password: walletPassphrase }),
                            coinSpecific: {},
                        });
                        getKeyNocks.push(getKeyNock);
                    }
                    const stub = sinon.stub(hotWallet, 'createShare').callsFake(async (options) => {
                        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
                        options.keychain.pub.should.not.be.undefined();
                        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
                        options.keychain.pub.should.equal(pub);
                        return undefined;
                    });
                    await hotWallet.shareWallet({ email, permissions, walletPassphrase });
                    stub.calledOnce.should.be.true();
                    getSharingKeyNock.isDone().should.be.True();
                    getKeyNocks.every((v) => v.isDone().should.be.True());
                });
            }
        });
        it('should provide skipKeychain to wallet share api for hot wallet', async function () {
            const userId = '123';
            const email = 'shareto@sdktest.com';
            const permissions = 'view,spend';
            const toKeychain = utxoLib.bip32.fromSeed(Buffer.from('deadbeef02deadbeef02deadbeef02deadbeef02', 'hex'));
            const path = 'm/999999/1/1';
            const pubkey = toKeychain.derivePath(path).publicKey.toString('hex');
            const getSharingKeyNock = nock(bgUrl)
                .post('/api/v1/user/sharingkey', { email })
                .reply(200, { userId, pubkey, path });
            const createShareNock = nock(bgUrl)
                .post(`/api/v2/tbtc/wallet/${wallet.id()}/share`, {
                user: userId,
                permissions,
                skipKeychain: true,
            })
                .reply(200, {});
            await wallet.shareWallet({ email, permissions, skipKeychain: true });
            createShareNock.isDone().should.be.True();
            getSharingKeyNock.isDone().should.be.True();
        });
        it('should decrypt webauthn encryptedPrv for wallet share (spend)', async function () {
            const userId = '123';
            const email = 'shareto@sdktest.com';
            const permissions = 'view,spend';
            const toKeychain = utxoLib.bip32.fromSeed(Buffer.from('deadbeef02deadbeef02deadbeef02deadbeef02', 'hex'));
            const path = 'm/999999/1/1';
            const pubkey = toKeychain.derivePath(path).publicKey.toString('hex');
            const privateKey = 'xprv1';
            const walletPassphrase1 = 'bitgo1234';
            const walletPassphrase2 = 'bitgo5678';
            const getSharingKeyNock = nock(bgUrl)
                .post('/api/v1/user/sharingkey', { email })
                .reply(200, { userId, pubkey, path });
            const pub = 'Zo1ggzTUKMY5bYnDvT5mtVeZxzf2FaLTbKkmvGUhUQk';
            const getKeyNock = nock(bgUrl)
                .get(`/api/v2/tbtc/key/${wallet.keyIds()[0]}`)
                .reply(200, {
                id: wallet.keyIds()[0],
                pub,
                source: 'user',
                encryptedPrv: bitgo.encrypt({ input: privateKey, password: walletPassphrase1 }),
                webauthnDevices: [
                    {
                        otpDeviceId: '123',
                        authenticatorInfo: {
                            credID: 'credID',
                            fmt: 'packed',
                            publicKey: 'some value',
                        },
                        prfSalt: '456',
                        encryptedPrv: bitgo.encrypt({ input: privateKey, password: walletPassphrase2 }),
                    },
                ],
                coinSpecific: {},
            });
            const stub = sinon.stub(wallet, 'createShare').callsFake(async (options) => {
                options.keychain.encryptedPrv.should.not.be.undefined();
                return undefined;
            });
            await wallet.shareWallet({ email, permissions, walletPassphrase: walletPassphrase2 });
            stub.calledOnce.should.be.true();
            getSharingKeyNock.isDone().should.be.True();
            getKeyNock.isDone().should.be.True();
        });
    });
    describe('Wallet Freezing', function () {
        it('should freeze wallet for specified duration in seconds', async function () {
            const params = { duration: 60 };
            const scope = nock(bgUrl).post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/freeze`, params).reply(200, {});
            await wallet.freeze(params);
            scope.isDone().should.be.True();
        });
    });
    describe('TSS Wallets', function () {
        const sandbox = sinon.createSandbox();
        const tsol = bitgo.coin('tsol');
        const walletData = {
            id: '5b34252f1bf349930e34020a00000000',
            coin: 'tsol',
            keys: [
                '598f606cd8fc24710d2ebad89dce86c2',
                '598f606cc8e43aef09fcb785221d9dd2',
                '5935d59cf660764331bafcade1855fd7',
            ],
            coinSpecific: {},
            multisigType: 'tss',
        };
        const ethWalletData = {
            id: '598f606cd8fc24710d2ebadb1d9459bb',
            coin: 'teth',
            keys: [
                '598f606cd8fc24710d2ebad89dce86c2',
                '598f606cc8e43aef09fcb785221d9dd2',
                '5935d59cf660764331bafcade1855fd7',
            ],
            multisigType: 'tss',
            coinSpecific: { addressVersion: 1 },
            type: 'hot',
        };
        const polygonWalletData = {
            id: '632826520ee1e5000729017354acaeab',
            coin: 'tpolygon',
            keys: [
                '598f606cd8fc24710d2ebad89dce86c2',
                '598f606cc8e43aef09fcb785221d9dd2',
                '5935d59cf660764331bafcade1855fd7',
            ],
            multisigType: 'tss',
        };
        const tssSolWallet = new sdk_core_1.Wallet(bitgo, tsol, walletData);
        let tssEthWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('teth'), ethWalletData);
        const tssPolygonWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('tpolygon'), polygonWalletData);
        const custodialTssSolWallet = new sdk_core_1.Wallet(bitgo, tsol, {
            ...walletData,
            type: 'custodial',
        });
        const txRequest = {
            txRequestId: 'id',
            transactions: [],
            intent: {
                intentType: 'payment',
            },
            date: new Date().toISOString(),
            latest: true,
            state: 'pendingUserSignature',
            userId: 'userId',
            walletType: 'hot',
            policiesChecked: false,
            version: 1,
            walletId: 'walletId',
            unsignedTxs: [
                {
                    serializedTxHex: 'ababcdcd',
                    signableHex: 'deadbeef',
                    feeInfo: {
                        fee: 5000,
                        feeString: '5000',
                    },
                    derivationPath: 'm/0',
                },
            ],
        };
        const txRequestFull = {
            txRequestId: 'id',
            intent: {
                intentType: 'payment',
            },
            date: new Date().toISOString(),
            latest: true,
            state: 'pendingUserSignature',
            userId: 'userId',
            walletId: 'walletId',
            signatureShares: [],
            version: 1,
            policiesChecked: false,
            walletType: 'hot',
            transactions: [
                {
                    state: 'pendingSignature',
                    unsignedTx: {
                        serializedTxHex: 'ababcdcd',
                        signableHex: 'deadbeef',
                        feeInfo: {
                            fee: 5000,
                            feeString: '5000',
                        },
                        derivationPath: 'm/0',
                    },
                    signatureShares: [],
                    commitmentShares: [],
                },
            ],
            unsignedTxs: [],
            apiVersion: 'full',
        };
        afterEach(function () {
            sandbox.verifyAndRestore();
        });
        describe('preBuildAndSignTransaction', async function () {
            const params = {
                walletPassphrase: 'passphrase12345',
                prebuildTx: { walletId: tssEthWallet.id(), txRequestId: 'randomId' },
                type: 'transfer',
            };
            ['eddsa', 'ecdsa'].forEach((keyCurve) => {
                describe(keyCurve, () => {
                    const wallet = keyCurve === 'eddsa' ? tssSolWallet : tssEthWallet;
                    beforeEach(function () {
                        sandbox
                            .stub(sdk_core_1.Keychains.prototype, 'getKeysForSigning')
                            .resolves([{ commonKeychain: 'test', id: '', pub: '', type: 'independent' }]);
                        if (keyCurve === 'eddsa') {
                            sandbox.stub(sdk_coin_sol_1.Tsol.prototype, 'verifyTransaction').resolves(true);
                        }
                        else {
                            sandbox.stub(sdk_coin_eth_1.Teth.prototype, 'verifyTransaction').resolves(true);
                        }
                    });
                    afterEach(function () {
                        sandbox.verifyAndRestore();
                    });
                    it('it should succeed but not sign if the txRequest is pending approval', async function () {
                        const getTxRequestStub = sandbox.stub(sdk_core_1.BaseTssUtils.default.prototype, 'getTxRequest').resolves({
                            ...txRequestFull,
                            state: 'pendingApproval',
                        });
                        const signTransactionSpy = sandbox.spy(sdk_core_1.Wallet.prototype, 'signTransaction');
                        const result = (await wallet.prebuildAndSignTransaction(params));
                        result.should.have.property('state');
                        result.state.should.equal('pendingApproval');
                        getTxRequestStub.calledOnce.should.be.true();
                        signTransactionSpy.notCalled.should.be.true();
                    });
                    it('it should succeed and sign if the txRequest is not pending approval', async function () {
                        const getTxRequestStub = sandbox.stub(sdk_core_1.BaseTssUtils.default.prototype, 'getTxRequest');
                        getTxRequestStub.resolves(txRequestFull);
                        const signTransactionStub = sandbox.stub(sdk_core_1.Wallet.prototype, 'signTransaction');
                        signTransactionStub.resolves({ ...txRequestFull, state: 'signed' });
                        const result = (await wallet.prebuildAndSignTransaction(params));
                        result.should.have.property('state');
                        result.state.should.equal('signed');
                        getTxRequestStub.calledOnce.should.be.true();
                        signTransactionStub.calledOnce.should.be.true();
                    });
                });
            });
        });
        describe('Transaction prebuilds', function () {
            it('should build a single recipient transfer transaction', async function () {
                const recipients = [
                    {
                        address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',
                        amount: '1000',
                    },
                ];
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.TssUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequest);
                // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke prebuildTransaction
                prebuildTxWithIntent.calledOnceWithExactly({
                    reqId,
                    recipients,
                    intentType: 'payment',
                });
                const txPrebuild = await tssSolWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'transfer',
                });
                txPrebuild.should.deepEqual({
                    walletId: tssSolWallet.id(),
                    wallet: tssSolWallet,
                    txRequestId: 'id',
                    txHex: 'ababcdcd',
                    buildParams: {
                        recipients,
                        type: 'transfer',
                    },
                    feeInfo: {
                        fee: 5000,
                        feeString: '5000',
                    },
                });
            });
            it('should build a single recipient transfer with pending approval id if transaction is having one', async function () {
                const recipients = [
                    {
                        address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',
                        amount: '1000',
                    },
                ];
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.TssUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves({ ...txRequest, state: 'pendingApproval', pendingApprovalId: 'some-id' });
                prebuildTxWithIntent.calledOnceWithExactly({
                    reqId,
                    recipients,
                    intentType: 'payment',
                });
                const txPrebuild = await custodialTssSolWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'transfer',
                });
                txPrebuild.should.deepEqual({
                    walletId: custodialTssSolWallet.id(),
                    wallet: custodialTssSolWallet,
                    txRequestId: 'id',
                    txHex: 'ababcdcd',
                    pendingApprovalId: 'some-id',
                    buildParams: {
                        recipients,
                        type: 'transfer',
                    },
                    feeInfo: {
                        fee: 5000,
                        feeString: '5000',
                    },
                });
            });
            it('should build a multiple recipient transfer transaction with memo', async function () {
                const recipients = [
                    {
                        address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',
                        amount: '1000',
                    },
                    {
                        address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',
                        amount: '2000',
                    },
                ];
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.TssUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequest);
                // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke prebuildTransaction
                prebuildTxWithIntent.calledOnceWithExactly({
                    reqId,
                    recipients,
                    intentType: 'payment',
                    memo: {
                        type: 'type',
                        value: 'test memo',
                    },
                });
                const txPrebuild = await tssSolWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'transfer',
                    memo: {
                        type: 'type',
                        value: 'test memo',
                    },
                });
                txPrebuild.should.deepEqual({
                    walletId: tssSolWallet.id(),
                    wallet: tssSolWallet,
                    txRequestId: 'id',
                    txHex: 'ababcdcd',
                    buildParams: {
                        recipients,
                        memo: {
                            type: 'type',
                            value: 'test memo',
                        },
                        type: 'transfer',
                    },
                    feeInfo: {
                        fee: 5000,
                        feeString: '5000',
                    },
                });
            });
            it('should build an enable token transaction', async function () {
                const recipients = [];
                const tokenName = 'tcoin:tokenName';
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.TssUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequest);
                // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke prebuildTransaction
                prebuildTxWithIntent.calledOnceWithExactly({
                    reqId,
                    recipients,
                    intentType: 'createAccount',
                    memo: {
                        type: 'type',
                        value: 'test memo',
                    },
                    tokenName,
                });
                const txPrebuild = await tssSolWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'enabletoken',
                    memo: {
                        type: 'type',
                        value: 'test memo',
                    },
                    tokenName,
                });
                txPrebuild.should.deepEqual({
                    walletId: tssSolWallet.id(),
                    wallet: tssSolWallet,
                    txRequestId: 'id',
                    txHex: 'ababcdcd',
                    buildParams: {
                        recipients,
                        memo: {
                            type: 'type',
                            value: 'test memo',
                        },
                        type: 'enabletoken',
                        tokenName,
                    },
                    feeInfo: {
                        fee: 5000,
                        feeString: '5000',
                    },
                });
            });
            it('should build an enable token transaction for cold wallets', async function () {
                const recipients = [];
                const tokenName = 'tcoin:tokenName';
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.TssUtils.prototype, 'prebuildTxWithIntent');
                txRequest.walletType = 'cold';
                prebuildTxWithIntent.resolves(txRequest);
                prebuildTxWithIntent.calledOnceWithExactly({
                    reqId,
                    recipients,
                    intentType: 'createAccount',
                    memo: {
                        type: 'type',
                        value: 'test memo',
                    },
                    tokenName,
                });
                const txPrebuild = await tssSolWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'enabletoken',
                    memo: {
                        type: 'type',
                        value: 'test memo',
                    },
                    tokenName,
                });
                txPrebuild.should.deepEqual({
                    walletId: tssSolWallet.id(),
                    wallet: tssSolWallet,
                    txRequestId: 'id',
                    txHex: 'ababcdcd',
                    buildParams: {
                        recipients,
                        memo: {
                            type: 'type',
                            value: 'test memo',
                        },
                        type: 'enabletoken',
                        tokenName,
                    },
                    feeInfo: {
                        fee: 5000,
                        feeString: '5000',
                    },
                });
            });
            it('should fail for non-transfer transaction types', async function () {
                await tssSolWallet
                    .prebuildTransaction({
                    reqId,
                    recipients: [
                        {
                            address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',
                            amount: '1000',
                        },
                    ],
                    type: 'stake',
                })
                    .should.be.rejectedWith('transaction type not supported: stake');
            });
            it('should fail for full api version compatibility', async function () {
                await custodialTssSolWallet
                    .prebuildTransaction({
                    reqId,
                    apiVersion: 'lite',
                    recipients: [
                        {
                            address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',
                            amount: '1000',
                        },
                    ],
                    type: 'transfer',
                })
                    .should.be.rejectedWith('For non self-custodial (hot) tss wallets, parameter `apiVersion` must be `full`.');
            });
            it('should build a single recipient transfer transaction for full', async function () {
                const recipients = [
                    {
                        address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',
                        amount: '1000',
                    },
                ];
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.TssUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequestFull);
                // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke prebuildTransaction
                prebuildTxWithIntent.calledOnceWithExactly({
                    reqId,
                    recipients,
                    intentType: 'payment',
                }, 'full');
                const txPrebuild = await custodialTssSolWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'transfer',
                });
                txPrebuild.should.deepEqual({
                    walletId: tssSolWallet.id(),
                    wallet: custodialTssSolWallet,
                    txRequestId: 'id',
                    txHex: 'ababcdcd',
                    buildParams: {
                        recipients,
                        type: 'transfer',
                    },
                    feeInfo: {
                        fee: 5000,
                        feeString: '5000',
                    },
                });
            });
            it('should call prebuildTxWithIntent with the correct params for eth transfers', async function () {
                const recipients = [
                    {
                        address: '0xAB100912e133AA06cEB921459aaDdBd62381F5A3',
                        amount: '1000',
                    },
                ];
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequestFull);
                await tssEthWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'transfer',
                    feeOptions,
                });
                sinon.assert.calledOnce(prebuildTxWithIntent);
                const args = prebuildTxWithIntent.args[0];
                args[0].recipients.should.deepEqual(recipients);
                args[0].feeOptions.should.deepEqual(feeOptions);
                args[0].intentType.should.equal('payment');
                args[1].should.equal('full');
            });
            it('should call prebuildTxWithIntent with the correct params for eth transfertokens', async function () {
                const recipients = [
                    {
                        address: '0xAB100912e133AA06cEB921459aaDdBd62381F5A3',
                        amount: '1000',
                        tokenName: 'gterc18dp',
                    },
                ];
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequestFull);
                await tssEthWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'transfertoken',
                    isTss: true,
                    feeOptions,
                });
                sinon.assert.calledOnce(prebuildTxWithIntent);
                const args = prebuildTxWithIntent.args[0];
                args[0].recipients.should.deepEqual(recipients);
                args[0].feeOptions.should.deepEqual(feeOptions);
                args[0].isTss.should.equal(true);
                args[0].intentType.should.equal('transferToken');
                args[1].should.equal('full');
            });
            it('should call prebuildTxWithIntent with the correct params for eth accelerations', async function () {
                const recipients = [
                    {
                        address: '0xAB100912e133AA06cEB921459aaDdBd62381F5A3',
                        amount: '1000',
                        tokenName: 'gterc18dp',
                    },
                ];
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const lowFeeTxid = '0x6ea07f9420f4676be6478ab1660eb92444a7c663e0e24bece929f715e882e0cf';
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequestFull);
                await tssEthWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'acceleration',
                    feeOptions,
                    lowFeeTxid,
                });
                sinon.assert.calledOnce(prebuildTxWithIntent);
                const args = prebuildTxWithIntent.args[0];
                args[0].should.not.have.property('recipients');
                args[0].feeOptions.should.deepEqual(feeOptions);
                args[0].lowFeeTxid.should.equal(lowFeeTxid);
                args[0].intentType.should.equal('acceleration');
                args[1].should.equal('full');
            });
            it('should call prebuildTxWithIntent with the correct params for eth accelerations for receive address', async function () {
                const recipients = [
                    {
                        address: '0xAB100912e133AA06cEB921459aaDdBd62381F5A3',
                        amount: '1000',
                        tokenName: 'gterc18dp',
                    },
                ];
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const lowFeeTxid = '0x6ea07f9420f4676be6478ab1660eb92444a7c663e0e24bece929f715e882e0cf';
                const receiveAddress = '0x062176bc9345da3e8ee90361b0cf6ff883ba7206';
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequestFull);
                await tssEthWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'acceleration',
                    feeOptions,
                    lowFeeTxid,
                    receiveAddress,
                });
                sinon.assert.calledOnce(prebuildTxWithIntent);
                const args = prebuildTxWithIntent.args[0];
                args[0].should.not.have.property('recipients');
                args[0].feeOptions.should.deepEqual(feeOptions);
                args[0].lowFeeTxid.should.equal(lowFeeTxid);
                args[0].receiveAddress.should.equal(receiveAddress);
                args[0].intentType.should.equal('acceleration');
                args[1].should.equal('full');
            });
            it('should call prebuildTxWithIntent with the correct params for eth fillNonce', async function () {
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequestFull);
                const nonce = '1';
                const comment = 'fillNonce comment';
                await tssEthWallet.prebuildTransaction({
                    reqId,
                    type: 'fillNonce',
                    feeOptions,
                    nonce,
                    comment,
                });
                sinon.assert.calledOnce(prebuildTxWithIntent);
                const args = prebuildTxWithIntent.args[0];
                args[0].should.not.have.property('recipients');
                args[0].feeOptions.should.deepEqual(feeOptions);
                args[0].nonce.should.equal(nonce);
                args[0].intentType.should.equal('fillNonce');
                args[0].comment.should.equal(comment);
                args[1].should.equal('full');
            });
            it('should call prebuildTxWithIntent with the correct params for eth fillNonce for receive address nonce filling tx', async function () {
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequestFull);
                const nonce = '1';
                const comment = 'fillNonce comment';
                const receiveAddress = '0x062176bc9345da3e8ee90361b0cf6ff883ba7206';
                await tssEthWallet.prebuildTransaction({
                    reqId,
                    type: 'fillNonce',
                    feeOptions,
                    nonce,
                    receiveAddress,
                    comment,
                });
                sinon.assert.calledOnce(prebuildTxWithIntent);
                const args = prebuildTxWithIntent.args[0];
                args[0].should.not.have.property('recipients');
                args[0].feeOptions.should.deepEqual(feeOptions);
                args[0].nonce.should.equal(nonce);
                args[0].intentType.should.equal('fillNonce');
                args[0].comment.should.equal(comment);
                args[0].receiveAddress.should.equal(receiveAddress);
                args[1].should.equal('full');
            });
            it('should call prebuildTxWithIntent with the correct feeOptions when passing using the legacy format', async function () {
                const recipients = [
                    {
                        address: '0xAB100912e133AA06cEB921459aaDdBd62381F5A3',
                        amount: '1000',
                    },
                ];
                const expectedFeeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                    gasLimit: undefined,
                };
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequestFull);
                await tssEthWallet.prebuildTransaction({
                    reqId,
                    recipients,
                    type: 'transfer',
                    eip1559: {
                        maxFeePerGas: expectedFeeOptions.maxFeePerGas.toString(),
                        maxPriorityFeePerGas: expectedFeeOptions.maxPriorityFeePerGas.toString(),
                    },
                });
                sinon.assert.calledOnce(prebuildTxWithIntent);
                const args = prebuildTxWithIntent.args[0];
                args[0].feeOptions.should.deepEqual(expectedFeeOptions);
            });
            it('populate intent should return valid eth acceleration intent', async function () {
                const mpcUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const lowFeeTxid = '0x6ea07f9420f4676be6478ab1660eb92444a7c663e0e24bece929f715e882e0cf';
                const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {
                    reqId,
                    intentType: 'acceleration',
                    lowFeeTxid,
                    feeOptions,
                });
                intent.should.have.property('recipients', undefined);
                intent.feeOptions.should.deepEqual(feeOptions);
                intent.txid.should.equal(lowFeeTxid);
                intent.intentType.should.equal('acceleration');
            });
            it('populate intent should return valid eth acceleration intent for receive address', async function () {
                const mpcUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const lowFeeTxid = '0x6ea07f9420f4676be6478ab1660eb92444a7c663e0e24bece929f715e882e0cf';
                const receiveAddress = '0x062176bc9345da3e8ee90361b0cf6ff883ba7206';
                const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {
                    reqId,
                    intentType: 'acceleration',
                    lowFeeTxid,
                    receiveAddress,
                    feeOptions,
                });
                intent.should.have.property('recipients', undefined);
                intent.feeOptions.should.deepEqual(feeOptions);
                intent.txid.should.equal(lowFeeTxid);
                intent.receiveAddress.should.equal(receiveAddress);
                intent.intentType.should.equal('acceleration');
            });
            it('populate intent should return valid eth fillNonce intent', async function () {
                const mpcUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const nonce = '1';
                const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {
                    reqId,
                    intentType: 'fillNonce',
                    nonce,
                    feeOptions,
                });
                intent.should.have.property('recipients', undefined);
                intent.feeOptions.should.deepEqual(feeOptions);
                intent.nonce.should.equal(nonce);
                intent.intentType.should.equal('fillNonce');
            });
            it('populate intent should return valid eth fillNonce intent for receive address nonce filling tx', async function () {
                const mpcUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const nonce = '1';
                const receiveAddress = '0x062176bc9345da3e8ee90361b0cf6ff883ba7206';
                const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {
                    reqId,
                    intentType: 'fillNonce',
                    nonce,
                    receiveAddress,
                    feeOptions,
                });
                intent.should.have.property('recipients', undefined);
                intent.feeOptions.should.deepEqual(feeOptions);
                intent.nonce.should.equal(nonce);
                intent.receiveAddress.should.equal(receiveAddress);
                intent.intentType.should.equal('fillNonce');
            });
            it('should populate intent with custodianTransactionId', async function () {
                const mpcUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));
                const feeOptions = {
                    maxFeePerGas: 3000000000,
                    maxPriorityFeePerGas: 2000000000,
                };
                const nonce = '1';
                const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {
                    custodianTransactionId: 'unittest',
                    reqId,
                    intentType: 'fillNonce',
                    nonce,
                    feeOptions,
                    isTss: true,
                });
                intent.custodianTransactionId.should.equal('unittest');
                intent.should.have.property('recipients', undefined);
                intent.feeOptions.should.deepEqual(feeOptions);
                intent.nonce.should.equal(nonce);
                intent.isTss.should.equal(true);
                intent.intentType.should.equal('fillNonce');
            });
            it('should build a single recipient transfer transaction providing apiVersion parameter as "full" ', async function () {
                const recipients = [
                    {
                        address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',
                        amount: '1000',
                    },
                ];
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.TssUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequestFull);
                prebuildTxWithIntent.calledOnceWithExactly({
                    reqId,
                    recipients,
                    intentType: 'payment',
                }, 'full');
                const txPrebuild = await custodialTssSolWallet.prebuildTransaction({
                    reqId,
                    apiVersion: 'full',
                    recipients,
                    type: 'transfer',
                });
                txPrebuild.should.deepEqual({
                    walletId: tssSolWallet.id(),
                    wallet: custodialTssSolWallet,
                    txRequestId: 'id',
                    txHex: 'ababcdcd',
                    buildParams: {
                        apiVersion: 'full',
                        recipients,
                        type: 'transfer',
                    },
                    feeInfo: {
                        fee: 5000,
                        feeString: '5000',
                    },
                });
            });
        });
        describe('Transaction signing', function () {
            it('should sign transaction', async function () {
                const signTxRequest = sandbox.stub(sdk_core_1.TssUtils.prototype, 'signTxRequest');
                signTxRequest.resolves(txRequest);
                // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke signTransaction
                signTxRequest.calledOnceWithExactly({ txRequest, prv: 'secretKey', reqId });
                const txPrebuild = {
                    walletId: tssSolWallet.id(),
                    wallet: tssSolWallet,
                    txRequestId: 'id',
                    txHex: 'ababcdcd',
                };
                const signedTransaction = await tssSolWallet.signTransaction({
                    reqId,
                    txPrebuild,
                    prv: 'sercretKey',
                });
                signedTransaction.should.deepEqual(txRequest);
            });
            it('should fail to sign transaction without txRequestId', async function () {
                const txPrebuild = {
                    walletId: tssSolWallet.id(),
                    wallet: tssSolWallet,
                    txHex: 'ababcdcd',
                };
                await tssSolWallet
                    .signTransaction({
                    reqId,
                    txPrebuild,
                    prv: 'sercretKey',
                })
                    .should.be.rejectedWith('txRequestId required to sign transactions with TSS');
            });
        });
        describe('getUserKeyAndSignTssTransaction', function () {
            ['eddsa', 'ecdsa'].forEach((keyCurve) => {
                describe(keyCurve, () => {
                    const wallet = keyCurve === 'eddsa' ? tssSolWallet : tssEthWallet;
                    let getKeysStub;
                    let signTransactionStub;
                    beforeEach(function () {
                        getKeysStub = sandbox.stub(sdk_core_1.Keychains.prototype, 'getKeysForSigning');
                        signTransactionStub = sandbox
                            .stub(sdk_core_1.Wallet.prototype, 'signTransaction')
                            .resolves({ ...txRequestFull, state: 'signed' });
                    });
                    afterEach(function () {
                        sandbox.verifyAndRestore();
                    });
                    it('should sign transaction', async function () {
                        getKeysStub.resolves([
                            {
                                commonKeychain: 'test',
                                id: '',
                                pub: '',
                                type: 'tss',
                                encryptedPrv: '{"iv":"15FsbDVI1zG9OggD8YX+Hg==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"hHbNH3Sz/aU=","ct":"WoNVKz7afiRxXI2w/YkzMdMyoQg/B15u1Q8aQgi96jJZ9wk6TIaSEc6bXFH3AHzD9MdJCWJQUpRhoQc/rgytcn69scPTjKeeyVMElGCxZdFVS/psQcNE+lue3//2Zlxj+6t1NkvYO+8yAezSMRBK5OdftXEjNQI="}',
                            },
                        ]);
                        const params = {
                            walletPassphrase: sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_PASSPHRASE,
                            txRequestId: 'id',
                        };
                        const response = await wallet.getUserKeyAndSignTssTransaction(params);
                        response.should.deepEqual({ ...txRequestFull, state: 'signed' });
                        getKeysStub.calledOnce.should.be.true();
                        signTransactionStub.calledOnce.should.be.true();
                    });
                    it('should throw if the keychain doesnt have the encryptedKey', async function () {
                        getKeysStub.resolves([{ commonKeychain: 'test', id: '', pub: '', type: 'tss' }]);
                        const params = {
                            walletPassphrase: sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_PASSPHRASE,
                            txRequestId: 'id',
                        };
                        await wallet
                            .getUserKeyAndSignTssTransaction(params)
                            .should.be.rejectedWith('the user keychain does not have property encryptedPrv');
                        getKeysStub.calledOnce.should.be.true();
                        signTransactionStub.notCalled.should.be.true();
                    });
                    it('should throw if password is invalid', async function () {
                        getKeysStub.resolves([
                            {
                                commonKeychain: 'test',
                                id: '',
                                pub: '',
                                type: 'tss',
                                encryptedPrv: '{"iv":"15FsbDVI1zG9OggD8YX+Hg==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"hHbNH3Sz/aU=","ct":"WoNVKz7afiRxXI2w/YkzMdMyoQg/B15u1Q8aQgi96jJZ9wk6TIaSEc6bXFH3AHzD9MdJCWJQUpRhoQc/rgytcn69scPTjKeeyVMElGCxZdFVS/psQcNE+lue3//2Zlxj+6t1NkvYO+8yAezSMRBK5OdftXEjNQI="}',
                            },
                        ]);
                        const params = {
                            walletPassphrase: 'randompass',
                            txRequestId: 'id',
                        };
                        await wallet
                            .getUserKeyAndSignTssTransaction(params)
                            .should.be.rejectedWith(`unable to decrypt keychain with the given wallet passphrase`);
                        getKeysStub.calledOnce.should.be.true();
                        signTransactionStub.notCalled.should.be.true();
                    });
                });
            });
        });
        describe('signAndSendTxRequest', function () {
            const exampleSignedTx = {
                txHex: '0x123',
                txid: '0x456',
                status: 'signed',
            };
            afterEach(async function () {
                sandbox.restore();
            });
            it('should sign lite transaction', async function () {
                const getUserKeyAndSignTssTxSpy = sandbox.stub(tssSolWallet, 'getUserKeyAndSignTssTransaction');
                getUserKeyAndSignTssTxSpy.resolves(exampleSignedTx);
                const submitTxSpy = sandbox.stub(tssSolWallet, 'submitTransaction');
                submitTxSpy.resolves(exampleSignedTx);
                const signedTx = await tssSolWallet.signAndSendTxRequest({
                    walletPassphrase: 'passphrase',
                    txRequestId: 'id',
                    isTxRequestFull: false,
                });
                sandbox.assert.calledOnce(getUserKeyAndSignTssTxSpy);
                sandbox.assert.calledOnce(submitTxSpy);
                signedTx.should.deepEqual(exampleSignedTx);
            });
            it('should sign full transaction', async function () {
                const deleteSignatureSharesSpy = sandbox.stub(sdk_core_1.TssUtils.prototype, 'deleteSignatureShares');
                const getUserKeyAndSignTssTxSpy = sandbox.stub(tssSolWallet, 'getUserKeyAndSignTssTransaction');
                getUserKeyAndSignTssTxSpy.resolves(exampleSignedTx);
                const signedTx = await tssSolWallet.signAndSendTxRequest({
                    walletPassphrase: 'passphrase',
                    txRequestId: 'id',
                    isTxRequestFull: true,
                });
                sandbox.assert.calledOnce(deleteSignatureSharesSpy);
                sandbox.assert.calledOnce(getUserKeyAndSignTssTxSpy);
                signedTx.should.deepEqual(exampleSignedTx);
            });
        });
        describe('Message Signing', function () {
            const txHash = '0xrrrsss1b';
            const txRequestForMessageSigning = {
                txRequestId: reqId.toString(),
                transactions: [],
                intent: {
                    intentType: 'signMessage',
                },
                date: new Date().toISOString(),
                latest: true,
                state: 'pendingUserSignature',
                userId: 'userId',
                walletType: 'hot',
                policiesChecked: false,
                version: 1,
                walletId: 'walletId',
                unsignedTxs: [],
                unsignedMessages: [],
                messages: [
                    {
                        state: 'signed',
                        messageRaw: 'hello world',
                        derivationPath: 'm/0',
                        signatureShares: [{ from: sdk_core_1.SignatureShareType.USER, to: sdk_core_1.SignatureShareType.USER, share: '' }],
                        combineSigShare: '0:rrr:sss:3',
                        txHash,
                    },
                ],
            };
            let signTxRequestForMessage;
            const messageSigningCoins = ['teth', 'tpolygon'];
            const messageRaw = 'test';
            const expected = {
                txRequestId: reqId.toString(),
                txHash,
                signature: txHash,
                messageRaw,
                coin: 'teth',
                messageEncoded: Buffer.from('\u0019Ethereum Signed Message:\n4test').toString('hex'),
            };
            beforeEach(async function () {
                signTxRequestForMessage = sandbox.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'signTxRequestForMessage');
                signTxRequestForMessage.resolves(txRequestForMessageSigning);
                sandbox
                    .stub(sdk_core_1.Keychains.prototype, 'getKeysForSigning')
                    .resolves([{ commonKeychain: 'test', id: '', pub: '', type: 'independent' }]);
                sinon.stub(sdk_core_1.Ecdsa.prototype, 'verify').resolves(true);
            });
            afterEach(async function () {
                sinon.restore();
                nock.cleanAll();
            });
            it('should throw error for unsupported coins', async function () {
                await tssSolWallet
                    .signMessage({
                    reqId,
                    message: { messageRaw },
                    prv: 'secretKey',
                })
                    .should.be.rejectedWith('Message signing not supported for Testnet Solana');
            });
            messageSigningCoins.map((coinName) => {
                const expectedWithCoinField = { ...expected, coin: 'teth' };
                tssEthWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin(coinName), ethWalletData);
                const txRequestId = txRequestForMessageSigning.txRequestId;
                it('should sign message', async function () {
                    const signMessageTssSpy = sinon.spy(tssEthWallet, 'signMessageTss');
                    nock(bgUrl)
                        .get(`/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${txRequestForMessageSigning.txRequestId}&latest=true`)
                        .reply(200, { txRequests: [txRequestForMessageSigning] });
                    const signMessage = await tssEthWallet.signMessage({
                        reqId,
                        message: { messageRaw, txRequestId },
                        prv: 'secretKey',
                    });
                    signMessage.should.deepEqual(expectedWithCoinField);
                    const actualArg = signMessageTssSpy.getCalls()[0].args[0];
                    actualArg.message?.messageEncoded?.should.equal(Buffer.from(`\u0019Ethereum Signed Message:\n${messageRaw.length}${messageRaw}`).toString('hex'));
                });
                it('should sign message when custodianMessageId is provided', async function () {
                    const signMessageTssSpy = sinon.spy(tssEthWallet, 'signMessageTss');
                    nock(bgUrl).post(`/api/v2/wallet/${tssEthWallet.id()}/txrequests`).reply(200, txRequestForMessageSigning);
                    const signMessage = await tssEthWallet.signMessage({
                        custodianMessageId: 'unittest',
                        reqId,
                        message: { messageRaw },
                        prv: 'secretKey',
                    });
                    signMessage.should.deepEqual(expectedWithCoinField);
                    const actualArg = signMessageTssSpy.getCalls()[0].args[0];
                    actualArg.message?.messageEncoded?.should.equal(Buffer.from(`\u0019Ethereum Signed Message:\n${messageRaw.length}${messageRaw}`).toString('hex'));
                });
                it('should sign message when txRequestId not provided', async function () {
                    const signMessageTssSpy = sinon.spy(tssEthWallet, 'signMessageTss');
                    nock(bgUrl).post(`/api/v2/wallet/${tssEthWallet.id()}/txrequests`).reply(200, txRequestForMessageSigning);
                    const signMessage = await tssEthWallet.signMessage({
                        reqId,
                        message: { messageRaw },
                        prv: 'secretKey',
                    });
                    signMessage.should.deepEqual(expectedWithCoinField);
                    const actualArg = signMessageTssSpy.getCalls()[0].args[0];
                    actualArg.message?.messageEncoded?.should.equal(Buffer.from(`\u0019Ethereum Signed Message:\n${messageRaw.length}${messageRaw}`).toString('hex'));
                });
                it('should fail to sign message with empty prv', async function () {
                    await tssEthWallet
                        .signMessage({
                        reqId,
                        message: { messageRaw, txRequestId },
                        prv: '',
                    })
                        .should.be.rejectedWith('keychain does not have property encryptedPrv');
                });
            });
        });
        describe('Typed Data Signing', function () {
            const txHash = '1901493fbf2ae1c27c3ced26a89070c6ab5d3fbf37ed778de9378e7703b7d1f116b3883077a61826129b98b622e54fc68c5008d1b1c16552e1eda6916f870d719220';
            const txRequestForTypedDataSigning = {
                txRequestId: reqId.toString(),
                transactions: [],
                intent: {
                    intentType: 'signMessage',
                },
                date: new Date().toISOString(),
                latest: true,
                state: 'pendingUserSignature',
                userId: 'userId',
                walletType: 'hot',
                policiesChecked: false,
                version: 1,
                walletId: 'walletId',
                unsignedTxs: [],
                unsignedMessages: [],
                messages: [
                    {
                        state: 'signed',
                        messageRaw: 'hello world',
                        derivationPath: 'm/0',
                        signatureShares: [{ from: sdk_core_1.SignatureShareType.USER, to: sdk_core_1.SignatureShareType.USER, share: '' }],
                        combineSigShare: '0:rrr:sss:3',
                        txHash,
                    },
                ],
            };
            let signTxRequestForMessage;
            const messageSigningCoins = ['teth', 'tpolygon'];
            const types = {
                EIP712Domain: [
                    {
                        name: 'name',
                        type: 'string',
                    },
                    {
                        name: 'version',
                        type: 'string',
                    },
                    {
                        name: 'chainId',
                        type: 'uint256',
                    },
                    {
                        name: 'verifyingContract',
                        type: 'address',
                    },
                ],
                Message: [{ name: 'data', type: 'string' }],
            };
            const typedMessage = {
                domain: {
                    name: 'bitgo',
                    version: '1',
                    chainId: 1,
                    verifyingContract: '0x0000000000000000000000000000000000000000',
                },
                primaryType: 'Message',
                types,
                message: { data: 'bitgo says hello!' },
            };
            const typedDataBase = {
                typedDataRaw: JSON.stringify(typedMessage),
                version: sdk_core_1.SignTypedDataVersion.V3,
            };
            beforeEach(async function () {
                signTxRequestForMessage = sandbox.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'signTxRequestForMessage');
                signTxRequestForMessage.resolves(txRequestForTypedDataSigning);
                sandbox
                    .stub(sdk_core_1.Keychains.prototype, 'getKeysForSigning')
                    .resolves([{ commonKeychain: 'test', id: '', pub: '', type: 'independent' }]);
                sinon.stub(sdk_core_1.Ecdsa.prototype, 'verify').resolves(true);
            });
            afterEach(async function () {
                sinon.restore();
                nock.cleanAll();
            });
            it('should throw error for unsupported coins', async function () {
                await tssSolWallet
                    .signTypedData({
                    reqId,
                    typedData: typedDataBase,
                    prv: 'secretKey',
                })
                    .should.be.rejectedWith('Sign typed data not supported for Testnet Solana');
            });
            it('should throw error for sign typed data V1', async function () {
                const typedData = { ...typedDataBase };
                typedData.version = sdk_core_1.SignTypedDataVersion.V1;
                nock(bgUrl)
                    .get(`/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${txRequestForTypedDataSigning.txRequestId}&latest=true`)
                    .reply(200, { txRequests: [txRequestForTypedDataSigning] });
                await tssEthWallet
                    .signTypedData({
                    reqId,
                    typedData,
                    prv: 'secretKey',
                })
                    .should.be.rejectedWith('SignTypedData v1 is not supported due to security concerns');
            });
            messageSigningCoins.map((coinName) => {
                tssEthWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin(coinName), ethWalletData);
                const txRequestId = txRequestForTypedDataSigning.txRequestId;
                typedDataBase.txRequestId = txRequestId;
                const expected = {
                    txRequestId,
                    messageRaw: JSON.stringify(typedMessage),
                    signature: txHash,
                    txHash,
                    coin: 'teth',
                    messageEncoded: txHash,
                };
                describe(`sign typed data V3 for ${coinName}`, async function () {
                    const typedData = { ...typedDataBase };
                    typedData.version = sdk_core_1.SignTypedDataVersion.V3;
                    it('should sign typed data V3', async function () {
                        const signTypedDataTssSpy = sinon.spy(tssEthWallet, 'signTypedDataTss');
                        nock(bgUrl)
                            .get(`/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${txRequestForTypedDataSigning.txRequestId}&latest=true`)
                            .reply(200, { txRequests: [txRequestForTypedDataSigning] });
                        const signedTypedData = await tssEthWallet.signTypedData({
                            reqId,
                            typedData,
                            prv: 'secretKey',
                        });
                        signedTypedData.should.deepEqual(expected);
                        const actualArg = signTypedDataTssSpy.getCalls()[0].args[0];
                        actualArg.typedData?.typedDataEncoded?.toString('hex').should.equal(txHash);
                    });
                    it('should sign typed data V3 when custodianMessageID is provided', async function () {
                        typedData.txRequestId = txRequestId;
                        const signTypedDataTssSpy = sinon.spy(tssEthWallet, 'signTypedDataTss');
                        nock(bgUrl)
                            .get(`/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${txRequestForTypedDataSigning.txRequestId}&latest=true`)
                            .reply(200, { txRequests: [txRequestForTypedDataSigning] });
                        const signedTypedData = await tssEthWallet.signTypedData({
                            custodianMessageId: 'unittest',
                            reqId,
                            typedData,
                            prv: 'secretKey',
                        });
                        signedTypedData.should.deepEqual(expected);
                        const actualArg = signTypedDataTssSpy.getCalls()[0].args[0];
                        actualArg.typedData?.typedDataEncoded?.toString('hex').should.equal(txHash);
                    });
                    it('should fail to sign typed data V3 with empty prv', async function () {
                        await tssEthWallet
                            .signTypedData({
                            reqId,
                            typedData: typedDataBase,
                            prv: '',
                        })
                            .should.be.rejectedWith('keychain does not have property encryptedPrv');
                    });
                    it('should sign typed data V3 when txRequestId not provided ', async function () {
                        delete typedData.txRequestId;
                        const signedTypedDataTssSpy = sinon.spy(tssEthWallet, 'signTypedDataTss');
                        nock(bgUrl).post(`/api/v2/wallet/${tssEthWallet.id()}/txrequests`).reply(200, txRequestForTypedDataSigning);
                        const signedTypedData = await tssEthWallet.signTypedData({
                            reqId,
                            typedData,
                            prv: 'secretKey',
                        });
                        signedTypedData.should.deepEqual(expected);
                        const actualArg = signedTypedDataTssSpy.getCalls()[0].args[0];
                        actualArg.typedData?.typedDataEncoded?.toString('hex').should.equal(txHash);
                    });
                });
                describe(`sign typed data V4 for ${coinName}`, async function () {
                    const typedData = { ...typedDataBase };
                    typedData.version = sdk_core_1.SignTypedDataVersion.V4;
                    it('should sign typed data V4', async function () {
                        typedData.txRequestId = txRequestId;
                        nock(bgUrl)
                            .get(`/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${txRequestForTypedDataSigning.txRequestId}&latest=true`)
                            .reply(200, { txRequests: [txRequestForTypedDataSigning] });
                        const signedTypedData = await tssEthWallet.signTypedData({
                            reqId,
                            typedData,
                            prv: 'secretKey',
                        });
                        signedTypedData.should.deepEqual(expected);
                    });
                    it('should sign typed data V4 when custodianMessageID is provided', async function () {
                        typedData.txRequestId = txRequestId;
                        nock(bgUrl)
                            .get(`/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${txRequestForTypedDataSigning.txRequestId}&latest=true`)
                            .reply(200, { txRequests: [txRequestForTypedDataSigning] });
                        const signedTypedData = await tssEthWallet.signTypedData({
                            custodianMessageId: 'unittest',
                            reqId,
                            typedData,
                            prv: 'secretKey',
                        });
                        signedTypedData.should.deepEqual(expected);
                    });
                    it('should fail to sign typed data V4 with empty prv', async function () {
                        await tssEthWallet
                            .signTypedData({
                            reqId,
                            typedData: typedDataBase,
                            prv: '',
                        })
                            .should.be.rejectedWith('keychain does not have property encryptedPrv');
                    });
                    it('should sign typed data V4 when txRequestId not provided ', async function () {
                        delete typedData.txRequestId;
                        const signedTypedDataTssSpy = sinon.spy(tssEthWallet, 'signTypedDataTss');
                        nock(bgUrl).post(`/api/v2/wallet/${tssEthWallet.id()}/txrequests`).reply(200, txRequestForTypedDataSigning);
                        const signedTypedData = await tssEthWallet.signTypedData({
                            reqId,
                            typedData,
                            prv: 'secretKey',
                        });
                        signedTypedData.should.deepEqual(expected);
                        const actualArg = signedTypedDataTssSpy.getCalls()[0].args[0];
                        actualArg.typedData?.typedDataEncoded?.toString('hex').should.equal(txHash);
                    });
                });
            });
        });
        describe('Send Many', function () {
            const sendManyInput = {
                type: 'transfer',
                recipients: [
                    {
                        address: 'address',
                        amount: '1000',
                    },
                ],
                reqId: new sdk_core_1.RequestTracer(),
            };
            afterEach(function () {
                nock.cleanAll();
            });
            it('should send many', async function () {
                const signedTransaction = {
                    txRequestId: 'txRequestId',
                };
                const prebuildAndSignTransaction = sandbox.stub(tssSolWallet, 'prebuildAndSignTransaction');
                prebuildAndSignTransaction.resolves(signedTransaction);
                // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke sendMany
                prebuildAndSignTransaction.calledOnceWithExactly(sendManyInput);
                const sendTxRequest = sandbox.stub(sdk_core_1.TssUtils.prototype, 'sendTxRequest');
                sendTxRequest.resolves('sendTxResponse');
                // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke sendMany
                sendTxRequest.calledOnceWithExactly(signedTransaction.txRequestId);
                const sendMany = await tssSolWallet.sendMany(sendManyInput);
                sendMany.should.deepEqual('sendTxResponse');
            });
            it('should send many and call setRequestTracer', async function () {
                const signedTransaction = {
                    txRequestId: 'txRequestId',
                };
                const prebuildAndSignTransaction = sandbox.stub(tssSolWallet, 'prebuildAndSignTransaction');
                prebuildAndSignTransaction.resolves(signedTransaction);
                prebuildAndSignTransaction.calledOnceWithExactly(sendManyInput);
                const sendTxRequest = sandbox.stub(sdk_core_1.TssUtils.prototype, 'sendTxRequest');
                sendTxRequest.resolves('sendTxResponse');
                sendTxRequest.calledOnceWithExactly(signedTransaction.txRequestId);
                const setRequestTracerSpy = sinon.spy(bitgo, 'setRequestTracer');
                setRequestTracerSpy.withArgs(sendManyInput.reqId);
                const sendMany = await tssSolWallet.sendMany(sendManyInput);
                sendMany.should.deepEqual('sendTxResponse');
                sinon.assert.calledOnce(setRequestTracerSpy);
                setRequestTracerSpy.restore();
            });
            it('should return transfer from sendMany for apiVersion=full', async function () {
                const wallet = new sdk_core_1.Wallet(bitgo, tsol, {
                    ...walletData,
                    type: 'custodial',
                });
                const signedTxResult = {
                    txRequestId: 'txRequestId',
                };
                const txRequest = {
                    date: new Date().toString(),
                    intent: 'payment',
                    latest: false,
                    policiesChecked: false,
                    state: 'delivered',
                    unsignedTxs: [],
                    userId: 'unit-test',
                    version: 0,
                    walletId: wallet.id(),
                    walletType: wallet.type() ?? 'hot',
                    txRequestId: signedTxResult.txRequestId,
                    transactions: [
                        {
                            state: 'delivered',
                            signedTx: {
                                id: 'txid',
                                tx: 'tx',
                            },
                            unsignedTx: 'something',
                            signatureShares: [],
                        },
                    ],
                };
                const transfer = {
                    id: 'transferId',
                    state: 'signed',
                    txid: 'txid',
                };
                const prebuildAndSignTransaction = sandbox.stub(wallet, 'prebuildAndSignTransaction').resolves(signedTxResult);
                const txRequestNock = nock(bgUrl)
                    .persist()
                    .get(`/api/v2/wallet/${walletData.id}/txrequests?txRequestIds=${signedTxResult.txRequestId}&latest=true`)
                    .reply(200, { txRequests: [txRequest] });
                const createTransferNock = nock(bgUrl)
                    .persist()
                    .post(`/api/v2/wallet/${walletData.id}/txrequests/${signedTxResult.txRequestId}/transfers`)
                    .reply(200, transfer);
                const input = {
                    type: 'transfer',
                    recipients: [
                        {
                            address: 'address',
                            amount: '1000',
                        },
                    ],
                    apiVersion: 'full',
                };
                const sendManyResult = await wallet.sendMany(input);
                prebuildAndSignTransaction.calledOnceWithExactly(input);
                txRequestNock.isDone().should.be.true();
                createTransferNock.isDone().should.be.true();
                sendManyResult.should.deepEqual({
                    txRequest,
                    transfer,
                    txid: 'txid',
                    tx: 'tx',
                    status: 'signed',
                });
            });
            it('should return pendingApproval from sendMany for apiVersion=full', async function () {
                const wallet = new sdk_core_1.Wallet(bitgo, tsol, {
                    ...walletData,
                    type: 'hot',
                });
                const signedTxResult = {
                    txRequestId: 'txRequestId',
                };
                const txRequest = {
                    txRequestId: signedTxResult.txRequestId,
                    date: new Date().toString(),
                    intent: 'payment',
                    latest: false,
                    policiesChecked: false,
                    state: 'pendingApproval',
                    unsignedTxs: [],
                    userId: 'unit-test',
                    version: 0,
                    walletId: wallet.id(),
                    walletType: wallet.type() ?? 'hot',
                    pendingApprovalId: 'some-pending-approval-id',
                    transactions: [
                        {
                            state: 'initialized',
                            unsignedTx: 'something',
                            signatureShares: [],
                        },
                    ],
                };
                const transfer = {
                    id: 'transferId',
                    state: 'signed',
                    txid: 'txid',
                };
                const pendingApproval = {
                    id: 'some-pending-approval-id',
                    wallet: wallet.id(),
                    info: {
                        type: 'transactionRequestFull',
                    },
                    txRequestId: txRequest.txRequestId,
                };
                const prebuildAndSignTransaction = sandbox.stub(wallet, 'prebuildAndSignTransaction').resolves(signedTxResult);
                const txRequestNock = nock(bgUrl)
                    .persist()
                    .get(`/api/v2/wallet/${walletData.id}/txrequests?txRequestIds=${txRequest.txRequestId}&latest=true`)
                    .reply(200, { txRequests: [txRequest] });
                const createTransferNock = nock(bgUrl)
                    .persist()
                    .post(`/api/v2/wallet/${walletData.id}/txrequests/${txRequest.txRequestId}/transfers`)
                    .reply(200, transfer);
                const getPendingApprovalNock = nock(bgUrl)
                    .persist()
                    .get(`/api/v2/${wallet.coin()}/pendingapprovals/${txRequest.pendingApprovalId}`)
                    .reply(200, pendingApproval);
                const input = {
                    type: 'transfer',
                    recipients: [
                        {
                            address: 'address',
                            amount: '1000',
                        },
                    ],
                    apiVersion: 'full',
                };
                const sendManyResult = await wallet.sendMany(input);
                prebuildAndSignTransaction.calledOnceWithExactly(input);
                txRequestNock.isDone().should.be.true();
                createTransferNock.isDone().should.be.true();
                getPendingApprovalNock.isDone().should.be.true();
                sendManyResult.should.deepEqual({ pendingApproval, txRequest });
            });
            it('should fail if txRequestId is missing from prebuild', async function () {
                const signedTransaction = {
                    txHex: 'deadbeef',
                };
                const prebuildAndSignTransaction = sandbox.stub(tssSolWallet, 'prebuildAndSignTransaction');
                prebuildAndSignTransaction.resolves(signedTransaction);
                // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke sendMany
                prebuildAndSignTransaction.calledOnceWithExactly(sendManyInput);
                await tssSolWallet
                    .sendMany(sendManyInput)
                    .should.be.rejectedWith('txRequestId missing from signed transaction');
            });
        });
        describe('Submit transaction', function () {
            it('should submit transaction with txRequestId', async function () {
                const nockSendTx = nock(bgUrl)
                    .persist(false)
                    .post(tssSolWallet.url('/tx/send').replace(bgUrl, ''))
                    .reply(200, { message: 'success' });
                const submittedTx = await tssSolWallet.submitTransaction({
                    txRequestId: 'id',
                });
                submittedTx.should.deepEqual({ message: 'success' });
                nockSendTx.isDone().should.be.true();
            });
            it('should fail when txRequestId and txHex are both provided', async function () {
                await tssSolWallet
                    .submitTransaction({
                    txRequestId: 'id',
                    txHex: 'beef',
                })
                    .should.be.rejectedWith('must supply exactly one of txRequestId, txHex, or halfSigned');
            });
            it('should fail when txRequestId and halfSigned are both provided', async function () {
                await tssSolWallet
                    .submitTransaction({
                    txRequestId: 'id',
                    halfSigned: {
                        txHex: 'beef',
                    },
                })
                    .should.be.rejectedWith('must supply exactly one of txRequestId, txHex, or halfSigned');
            });
            it('should fail when txHex and halfSigned are both provided', async function () {
                await tssSolWallet
                    .submitTransaction({
                    txHex: 'beef',
                    halfSigned: {
                        txHex: 'beef',
                    },
                })
                    .should.be.rejectedWith('must supply either txHex or halfSigned, but not both');
            });
        });
        describe('Transfer tokens', function () {
            const recipients = [
                {
                    address: '0x101c3928946b2e1d99759e8e5d34b5e94c1a8e2f',
                    amount: '0',
                    tokenData: {
                        tokenName: 'erc721:bitgoerc721',
                        tokenContractAddress: '0x8397b091514c1f7bebb9dea6ac267ea23b570605',
                        tokenId: '38',
                        tokenQuantity: '1',
                        decimalPlaces: 0,
                        tokenType: sdk_core_1.TokenType.ERC721,
                    },
                },
            ];
            const feeOptions = {
                maxFeePerGas: 2000000000,
                maxPriorityFeePerGas: 1000000000,
            };
            it('calling prebuildxTransaction should execute prebuildTxWithIntent with proper params', async function () {
                const txRequestFullTokenTransfer = { ...txRequestFull, intent: 'transferToken' };
                const prebuildTxWithIntent = sandbox.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');
                prebuildTxWithIntent.resolves(txRequestFullTokenTransfer);
                // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke prebuildTransaction
                prebuildTxWithIntent.calledOnceWithExactly({
                    reqId,
                    recipients,
                    intentType: 'transferToken',
                    feeOptions,
                }, 'full');
                const txPrebuild = await tssPolygonWallet.prebuildTransaction({
                    isTss: true,
                    recipients,
                    type: 'transfertoken',
                    walletPassphrase: 'passphrase12345',
                    feeOptions,
                });
                txPrebuild.should.deepEqual({
                    walletId: tssPolygonWallet.id(),
                    wallet: tssPolygonWallet,
                    txRequestId: 'id',
                    txHex: 'ababcdcd',
                    buildParams: {
                        recipients,
                        type: 'transfertoken',
                    },
                    feeInfo: {
                        fee: 5000,
                        feeString: '5000',
                    },
                });
            });
            it('should populate intent with EVM-like params', async function () {
                const mpcUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('tpolygon'));
                // @ts-expect-error only pass in params being tested
                const intent = mpcUtils.populateIntent(bitgo.coin('tpolygon'), {
                    intentType: 'transferToken',
                    recipients,
                    feeOptions,
                });
                intent.should.have.property('feeOptions');
                intent.feeOptions.should.have.property('maxFeePerGas', 2000000000);
                intent.feeOptions.should.have.property('maxPriorityFeePerGas', 1000000000);
                intent.should.have.property('recipients');
                intent.recipients.should.have.property('length', 1);
                intent.recipients[0].should.have.property('tokenData');
                intent.recipients[0].tokenData.should.have.property('tokenQuantity', recipients[0].tokenData.tokenQuantity);
                intent.recipients[0].tokenData.should.have.property('tokenType', recipients[0].tokenData.tokenType);
                intent.recipients[0].tokenData.should.have.property('tokenName', recipients[0].tokenData.tokenName);
                intent.recipients[0].tokenData.should.have.property('tokenContractAddress', recipients[0].tokenData.tokenContractAddress);
                intent.recipients[0].tokenData.should.have.property('tokenId', recipients[0].tokenData.tokenId);
                intent.recipients[0].tokenData.should.have.property('decimalPlaces', recipients[0].tokenData.decimalPlaces);
            });
            it('should populate intent with calldata', async function () {
                const recipients = [
                    {
                        address: '0x101c3928946b2e1d99759e8e5d34b5e94c1a8e2f',
                        amount: '0',
                        data: '0x000011112222',
                    },
                ];
                const mpcUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));
                // @ts-expect-error only pass in params being tested
                const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {
                    intentType: 'payment',
                    recipients,
                    feeOptions,
                });
                intent.should.have.property('feeOptions');
                intent.feeOptions.should.have.property('maxFeePerGas', 2000000000);
                intent.feeOptions.should.have.property('maxPriorityFeePerGas', 1000000000);
                intent.should.have.property('recipients');
                intent.recipients.should.have.property('length', 1);
                intent.recipients[0].data.should.equal('0x000011112222');
            });
            it('should not populate intent with tokenData if certain params are undefined', async function () {
                const mpcUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('tpolygon'));
                const recipients = [
                    {
                        address: '0x101c3928946b2e1d99759e8e5d34b5e94c1a8e2f',
                        amount: '0',
                        tokenData: {
                            tokenName: 'erc721:bitgoerc721',
                            tokenContractAddress: '0x8397b091514c1f7bebb9dea6ac267ea23b570605',
                            tokenId: '38',
                            tokenQuantity: '1',
                            decimalPlaces: 0,
                        },
                    },
                ];
                let intent;
                try {
                    intent = mpcUtils.populateIntent(bitgo.coin('tpolygon'), {
                        intentType: 'transferToken',
                        // @ts-expect-error only pass in params be tested for
                        recipients,
                        feeOptions,
                    });
                    intent.should.equal(undefined);
                }
                catch (e) {
                    e.message.should.equal('token type and quantity is required to request a transaction with intent to transfer a token');
                }
            });
        });
        describe('Transfer NFTs', function () {
            it('should populate intent with NFT token details', async function () {
                const params = {
                    reqId,
                    intentType: 'payment',
                    recipients: [
                        {
                            address: '0x9fef749050644625012a2c866973775e7123753b3eef0a1a4037453ac26d79bf',
                            amount: '1',
                            tokenData: {
                                tokenType: sdk_core_1.TokenType.DIGITAL_ASSET,
                                tokenQuantity: '1',
                                tokenContractAddress: '0xbbc561fbfa5d105efd8dfb06ae3e7e5be46331165b99d518f094c701e40603b5',
                                tokenId: '0x675b053d72c24dcc6bc7f38cdd45b4843cfb7af69a25ad21d002c376357e9d69',
                            },
                        },
                    ],
                };
                const baseCoin = bitgo.coin('tapt');
                const mpcUtils = new sdk_core_1.EDDSAUtils.default(bitgo, baseCoin);
                const intent = mpcUtils.populateIntent(baseCoin, params);
                intent.should.have.property('intentType', 'payment');
                intent.recipients.should.deepEqual([
                    {
                        address: {
                            address: '0x9fef749050644625012a2c866973775e7123753b3eef0a1a4037453ac26d79bf',
                        },
                        amount: {
                            value: '1',
                            symbol: 'tapt:nftcollection1',
                        },
                        tokenData: {
                            tokenType: 'Digital Asset',
                            tokenQuantity: '1',
                            tokenContractAddress: '0xbbc561fbfa5d105efd8dfb06ae3e7e5be46331165b99d518f094c701e40603b5',
                            tokenId: '0x675b053d72c24dcc6bc7f38cdd45b4843cfb7af69a25ad21d002c376357e9d69',
                            tokenName: 'tapt:nftcollection1',
                        },
                    },
                ]);
            });
        });
        describe('Wallet Sharing', function () {
            it('should use keychain pub to share tss wallet', async function () {
                const userId = '123';
                const email = 'shareto@sdktest.com';
                const permissions = 'view,spend';
                const toKeychain = utxoLib.bip32.fromSeed(Buffer.from('deadbeef02deadbeef02deadbeef02deadbeef02', 'hex'));
                const path = 'm/999999/1/1';
                const pubkey = toKeychain.derivePath(path).publicKey.toString('hex');
                const walletPassphrase = 'bitgo1234';
                const getSharingKeyNock = nock(bgUrl)
                    .post('/api/v1/user/sharingkey', { email })
                    .reply(200, { userId, pubkey, path });
                // commonPub + commonChaincode
                const commonKeychain = (0, crypto_1.randomBytes)(32).toString('hex') + (0, crypto_1.randomBytes)(32).toString('hex');
                const getKeyNock = nock(bgUrl)
                    .get(`/api/v2/tsol/key/${tssSolWallet.keyIds()[0]}`)
                    .reply(200, {
                    id: tssSolWallet.keyIds()[0],
                    commonKeychain: commonKeychain,
                    source: 'user',
                    encryptedPrv: bitgo.encrypt({ input: 'xprv1', password: walletPassphrase }),
                    coinSpecific: {},
                });
                const stub = sinon.stub(tssSolWallet, 'createShare').callsFake(async (options) => {
                    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
                    options.keychain.pub.should.not.be.undefined();
                    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
                    options.keychain.pub.should.equal(sdk_core_1.TssUtils.getPublicKeyFromCommonKeychain(commonKeychain));
                    return undefined;
                });
                await tssSolWallet.shareWallet({ email, permissions, walletPassphrase });
                stub.calledOnce.should.be.true();
                getSharingKeyNock.isDone().should.be.True();
                getKeyNock.isDone().should.be.True();
            });
        });
    });
    describe('AVAX tests', function () {
        let bgUrl;
        let basecoin;
        let walletData;
        let wallet;
        before(async function () {
            nock.pendingMocks().should.be.empty();
            bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
            walletData = {
                id: '5b34252f1bf349930e34020a00000000',
                keys: [
                    '5b3424f91bf349930e34017500000000',
                    '5b3424f91bf349930e34017600000000',
                    '5b3424f91bf349930e34017700000000',
                ],
                coinSpecific: {},
            };
        });
        it('should fetch cross-chain utxos', async function () {
            basecoin = bitgo.coin('tavaxp');
            walletData.coin = 'tavaxp';
            wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
            const params = { sourceChain: 'C' };
            const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/crossChainUnspents`;
            const scope = nock(bgUrl)
                .get(path)
                .query(params)
                .reply(200, {
                unspent: {
                    outputID: 7,
                    amount: '10000000',
                    txid: 'V3UBZTQj364zNWqt8uMHD5NjxxX8T8qkbeZXURmjnVmLEqzab',
                    threshold: 2,
                    addresses: [
                        'C-fuji199fluegrthqs4tvz40zajfrsx5m7dvy75ajfm6',
                        'C-fuji1gk3m444893ynl0gfvxahjgw3vftnn8sptyd9g5',
                        'C-fuji1ujfzjgwzfygl60qp2l8rmglg3lnm7w4059nca5',
                    ],
                    outputidx: '1111XiaYg',
                    locktime: '0',
                },
                fromWallet: '635092fd4ff3316142df6e6b7a078b92',
                toWallet: '635092fd4ff3316142df6e891f6a7ee6',
                toAddress: '0x125c4451c870f753265b0b1af3cf6ab88ffe4657',
            });
            try {
                await wallet.fetchCrossChainUTXOs(params);
            }
            catch (e) {
                // test is successful if nock is consumed, HMAC errors expected
            }
            scope.isDone().should.be.True();
        });
        it('sendMany should work for C > P export with custodial wallet', async function () {
            basecoin = bitgo.coin('tavaxc');
            walletData.coin = 'tavaxc';
            walletData.type = 'custodial';
            wallet = new sdk_core_1.Wallet(bitgo, basecoin, walletData);
            const address = 'P-fuji1e56pc4966qsevzhwgkym5l0jfma9llkqnrr4gh~P-fuji1kq05zm9nmlq8p3ld55k79dl3qay6c0e3atj56v~P-fuji1rp46z30qg457xc3dpffyxcgzpflxc85mhkjme3';
            const initiateTxParams = {
                recipients: [
                    {
                        amount: '10000000000000000', // 0.01 AVAX
                        address,
                    },
                ],
                hop: true,
                type: 'Export',
            };
            const initiateTxPath = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/initiate`;
            let initiateTxBody;
            const response = nock(bgUrl)
                .post(initiateTxPath, (body) => {
                initiateTxBody = body;
                return true;
            })
                .reply(200);
            const feeEstimationPath = `/api/v2/${wallet.coin()}/tx/fee?hop=true&recipient=${address}&amount=10000000000000000&type=Export`;
            nock(bgUrl).get(feeEstimationPath).reply(200, {
                feeEstimate: '718750000000000',
                gasLimitEstimate: 500000,
            });
            try {
                await wallet.sendMany(initiateTxParams);
            }
            catch (e) {
                console.log(e);
                // test is successful if nock is consumed, HMAC errors expected
            }
            _.isMatch(initiateTxBody, {
                hopParams: {
                    gasPriceMax: 7187500000,
                    gasLimit: 500000,
                },
                type: 'Export',
                recipients: [
                    {
                        amount: '10000000000000000',
                        address,
                    },
                ],
            }).should.be.true();
            response.isDone().should.be.true();
        });
    });
    describe('NFT Tests', function () {
        let ethWallet;
        before(async function () {
            const walletData = {
                id: '598f606cd8fc24710d2ebadb1d9459bb',
                coin: 'hteth',
                keys: [
                    '598f606cd8fc24710d2ebad89dce86c2',
                    '598f606cc8e43aef09fcb785221d9dd2',
                    '5935d59cf660764331bafcade1855fd7',
                ],
                multisigType: 'onchain',
                coinSpecific: {
                    baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                },
            };
            ethWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('hteth'), walletData);
        });
        afterEach(async function () {
            nock.cleanAll();
        });
        it('Should return all nfts in the wallet', async function () {
            const getTokenBalanceNock = nock(bgUrl)
                .get(`/api/v2/hteth/wallet/${ethWallet.id()}?allTokens=true`)
                .reply(200, {
                ...walletData,
                ...nftResponses_1.nftResponse,
            });
            const nfts = await ethWallet.getNftBalances();
            getTokenBalanceNock.isDone().should.be.true();
            nfts.should.length(5);
            nfts.should.containEql({
                type: 'ERC721',
                metadata: {
                    name: 'terc721:bitgoerc721',
                    tokenContractAddress: '0x8397b091514c1f7bebb9dea6ac267ea23b570605',
                },
                collections: {},
                balanceString: '0',
                confirmedBalanceString: '0',
                spendableBalanceString: '0',
                transferCount: 0,
            });
        });
        it('Should throw when attempting to transfer a nft collection not in the wallet', async function () {
            const getTokenBalanceNock = nock(bgUrl)
                .get(`/api/v2/hteth/wallet/${ethWallet.id()}?allTokens=true`)
                .reply(200, {
                ...walletData,
                ...nftResponses_1.nftResponse,
            });
            await ethWallet
                .sendNft({
                walletPassphrase: '123abc',
                otp: '000000',
            }, {
                tokenId: '123',
                type: 'ERC721',
                tokenContractAddress: '0x123badaddress',
                recipientAddress: '0xc15acc27ee41f266877c8f0c61df5bcbc7997df6',
            })
                .should.be.rejectedWith('Collection not found for token contract 0x123badaddress');
            getTokenBalanceNock.isDone().should.be.true();
        });
        it('Should throw when attempting to transfer a ERC-721 nft not owned by the wallet', async function () {
            const getTokenBalanceNock = nock(bgUrl)
                .get(`/api/v2/hteth/wallet/${ethWallet.id()}?allTokens=true`)
                .reply(200, {
                ...walletData,
                ...nftResponses_1.nftResponse,
                ...nftResponses_1.unsupportedNftResponse,
            });
            await ethWallet
                .sendNft({
                walletPassphrase: '123abc',
                otp: '000000',
            }, {
                tokenId: '123',
                type: 'ERC721',
                tokenContractAddress: '0xf5de760f2e916647fd766b4ad9e85ff943ce3a2b',
                recipientAddress: '0xc15acc27ee41f266877c8f0c61df5bcbc7997df6',
            })
                .should.be.rejectedWith('Token 123 not found in collection 0xf5de760f2e916647fd766b4ad9e85ff943ce3a2b or does not have a spendable balance');
            getTokenBalanceNock.isDone().should.be.true();
        });
        it('Should throw when attempting to transfer ERC-1155 tokens when the amount transferred is more than the spendable balance', async function () {
            const getTokenBalanceNock = nock(bgUrl)
                .get(`/api/v2/hteth/wallet/${ethWallet.id()}?allTokens=true`)
                .reply(200, {
                ...walletData,
                ...{
                    unsupportedNfts: {
                        '0xf5de760f2e916647fd766b4ad9e85ff943ce3a2b': {
                            type: 'ERC1155',
                            collections: {
                                1186703: '9',
                                1186705: '1',
                                1294856: '1',
                                1294857: '1',
                                1294858: '1',
                                1294859: '1',
                                1294860: '1',
                            },
                            metadata: {
                                name: 'MultiFaucet NFT',
                                tokenContractAddress: '0xf5de760f2e916647fd766b4ad9e85ff943ce3a2b',
                            },
                        },
                    },
                },
            });
            await ethWallet
                .sendNft({
                walletPassphrase: '123abc',
                otp: '000000',
            }, {
                entries: [
                    {
                        amount: 10,
                        tokenId: '1186703',
                    },
                    {
                        amount: 1,
                        tokenId: '1186705',
                    },
                ],
                type: 'ERC1155',
                tokenContractAddress: '0xf5de760f2e916647fd766b4ad9e85ff943ce3a2b',
                recipientAddress: '0xc15acc27ee41f266877c8f0c61df5bcbc7997df6',
            })
                .should.be.rejectedWith('Amount 10 exceeds spendable balance of 9 for token 1186703');
            getTokenBalanceNock.isDone().should.be.true();
        });
    });
    describe('Ada tests: ', () => {
        let adaWallet;
        const adaBitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
        adaBitgo.initializeTestVars();
        const walletData = {
            id: '598f606cd8fc24710d2ebadb1d9459bb',
            coinSpecific: {
                baseAddress: 'addr_test1q9faa5q3zr38wkd4kd3u8jfshx97jxvwsyvjg3tac826502nmmgpzy8zwavmtvmrc0ynpwvtayvcaqgey3zhmsw44g7shrfrh9',
                pendingChainInitialization: false,
                minimumFunding: 1000000,
                lastChainIndex: { 0: 0 },
            },
            coin: 'tada',
            keys: [
                '598f606cd8fc24710d2ebad89dce86c2',
                '598f606cc8e43aef09fcb785221d9dd2',
                '5935d59cf660764331bafcade1855fd7',
            ],
            multisigType: 'tss',
        };
        before(async function () {
            adaWallet = new sdk_core_1.Wallet(bitgo, bitgo.coin('tada'), walletData);
            nock(bgUrl).get(`/api/v2/${adaWallet.coin()}/key/${adaWallet.keyIds()[0]}`).times(3).reply(200, {
                id: '598f606cd8fc24710d2ebad89dce86c2',
                pub: '5f8WmC2uW9SAk7LMX2r4G1Bx8MMwx8sdgpotyHGodiZo',
                source: 'user',
                encryptedPrv: '{"iv":"hNK3rg82P1T94MaueXFAbA==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"cV4wU4EzPjs=","ct":"9VZX99Ztsb6p75Cxl2lrcXBplmssIAQ9k7ZA81vdDYG4N5dZ36BQNWVfDoelj9O31XyJ+Xri0XKIWUzl0KKLfUERplmtNoOCn5ifJcZwCrOxpHZQe3AJ700o8Wmsrk5H"}',
                coinSpecific: {},
            });
            nock(bgUrl).get(`/api/v2/${adaWallet.coin()}/key/${adaWallet.keyIds()[1]}`).times(2).reply(200, {
                id: '598f606cc8e43aef09fcb785221d9dd2',
                pub: 'G1s43JTzNZzqhUn4aNpwgcc6wb9FUsZQD5JjffG6isyd',
                encryptedPrv: '{"iv":"UFrt/QlIUR1XeQafPBaAlw==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"7VPBYaJXPm8=","ct":"ajFKv2y8yaIBXQ39sAbBWcnbiEEzbjS4AoQtp5cXYqjeDRxt3aCxemPm22pnkJaCijFjJrMHbkmsNhNYzHg5aHFukN+nEAVssyNwHbzlhSnm8/BVN50yAdAAtWreh8cp"}',
                source: 'backup',
                coinSpecific: {},
            });
            nock(bgUrl).get(`/api/v2/${adaWallet.coin()}/key/${adaWallet.keyIds()[2]}`).times(2).reply(200, {
                id: '5935d59cf660764331bafcade1855fd7',
                pub: 'GH1LV1e9FdqGe8U2c8PMEcma3fDeh1ktcGVBrD3AuFqx',
                encryptedPrv: '{"iv":"iIuWOHIOErEDdiJn6g46mg==","v":1,"iter":10000,"ks":256,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"Rzh7RRJksj0=","ct":"rcNICUfp9FakT53l+adB6XKzS1vNTc0Qq9jAtqnxA+ScssiS4Q0l3sgG/0gDy5DaZKtXryKBDUvGsi7b/fYaFCUpAoZn/VZTOhOUN/mo7ZHb4OhOXL29YPPkiryAq9Cr"}',
                source: 'bitgo',
                coinSpecific: {},
            });
        });
        after(async function () {
            sinon.restore();
            nock.cleanAll();
        });
        it('Should send unspents in payment intent when using sendmany', async function () {
            const sendManyParams = {
                type: 'transfer',
                recipients: [
                    {
                        address: 'address',
                        amount: '1000',
                    },
                ],
                unspents: ['unspent1', 'unspent2'],
            };
            nock(bgUrl)
                .post(`/api/v2/wallet/${adaWallet.id()}/txrequests`)
                .reply((url, body) => {
                // validate that the populated intent has unspents
                body.intent.intentType.should.equal('payment');
                body.intent.unspents.should.deepEqual(['unspent1', 'unspent2']);
                return [
                    200,
                    {
                        apiVersion: 'lite',
                        unsignedTxs: [
                            {
                                unsignedTx: {
                                    serializedTxHex: 'serializedTxHex',
                                    feeInfo: 'fee info',
                                },
                            },
                        ],
                    },
                ];
            });
            // stub all steps after txrequest creation
            sinon.stub(adaWallet.baseCoin, 'verifyTransaction').resolves(true);
            sinon.stub(adaWallet, 'signTransaction').resolves({ txRequestId: 'txRequestId' });
            sinon.stub(sdk_core_1.BaseTssUtils.default.prototype, 'sendTxRequest').resolves('sendTxResponse');
            await adaWallet.sendMany(sendManyParams);
        });
        it('Should send senderAddress in payment intent when using sendmany', async function () {
            const sendManyParams = {
                type: 'transfer',
                recipients: [
                    {
                        address: 'address',
                        amount: '1000',
                    },
                ],
                senderAddress: 'senderAddr1',
            };
            nock(bgUrl)
                .post(`/api/v2/wallet/${adaWallet.id()}/txrequests`)
                .reply((url, body) => {
                const createTxRequestBody = body;
                createTxRequestBody.intent.intentType.should.equal('payment');
                createTxRequestBody.intent.senderAddress?.should.equal('senderAddr1');
                return [
                    200,
                    {
                        apiVersion: 'lite',
                        unsignedTxs: [
                            {
                                unsignedTx: {
                                    serializedTxHex: 'serializedTxHex',
                                    feeInfo: 'fee info',
                                },
                            },
                        ],
                    },
                ];
            });
            sinon.stub(adaWallet.baseCoin, 'verifyTransaction').resolves(true);
            sinon.stub(adaWallet, 'signTransaction').resolves({ txRequestId: 'txRequestId' });
            sinon.stub(sdk_core_1.BaseTssUtils.default.prototype, 'sendTxRequest').resolves('sendTxResponse');
            await adaWallet.sendMany(sendManyParams).should.be.resolved();
        });
    });
    describe('ERC20 Token Approval', function () {
        let wallet;
        const walletId = '5b34252f1bf349930e34020a00000000';
        let topethCoin;
        const tokenName = 'topeth:terc18dp';
        const coin = 'topeth';
        beforeEach(function () {
            topethCoin = bitgo.coin('topeth');
            wallet = new sdk_core_1.Wallet(bitgo, topethCoin, {
                id: walletId,
                coin: coin,
                keys: ['keyid1', 'keyid2', 'keyid3'],
            });
        });
        afterEach(function () {
            sinon.restore();
        });
        it('should successfully approve a token', async function () {
            const walletPassphrase = 'password123';
            const expectedApprovalBuild = {
                txHex: '0x123456',
                feeInfo: {
                    fee: '1000000000',
                },
            };
            const expectedSignedTx = {
                txHex: '0x123456signed',
            };
            const expectedSendResult = {
                txid: '0xabcdef',
                tx: '0x123456signed',
                status: 'signed',
            };
            // Mock the token approval build API
            const ethUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
            nock(ethUrl).post(`/api/v2/${coin}/wallet/${walletId}/token/approval/build`).reply(200, expectedApprovalBuild);
            // Mock the getKeychainsAndValidatePassphrase method
            sinon.stub(wallet, 'getKeychainsAndValidatePassphrase').resolves([
                {
                    id: 'keyid1',
                    pub: 'pub1',
                    encryptedPrv: 'encryptedPrv',
                },
            ]);
            // Mock the sign transaction method
            sinon.stub(wallet, 'signTransaction').resolves(expectedSignedTx);
            // Mock the send transaction method
            sinon.stub(wallet, 'sendTransaction').resolves(expectedSendResult);
            const result = await wallet.approveErc20Token(walletPassphrase, tokenName);
            should.exist(result);
            result.should.deepEqual(expectedSendResult);
            // Verify the parameters passed to signTransaction
            const signParams = wallet.signTransaction.firstCall.args[0];
            signParams.should.have.property('txPrebuild', expectedApprovalBuild);
            signParams.should.have.property('keychain');
            signParams.should.have.property('walletPassphrase', walletPassphrase);
        });
        it('should handle token approval build API errors', async function () {
            const walletPassphrase = 'password123';
            const errorMessage = 'token approval build failed';
            // Mock the token approval build API to return an error
            nock(bgUrl).post(`/api/v2/${coin}/wallet/${walletId}/token/approval/build`).replyWithError(errorMessage);
            await wallet.approveErc20Token(walletPassphrase, tokenName).should.be.rejectedWith(errorMessage);
        });
        it('should handle wallet passphrase validation errors', async function () {
            const walletPassphrase = 'wrong-password';
            const expectedApprovalBuild = {
                txHex: '0x123456',
                feeInfo: {
                    fee: '1000000000',
                },
            };
            // Mock the token approval build API
            nock(bgUrl).post(`/api/v2/${coin}/wallet/${walletId}/token/approval/build`).reply(200, expectedApprovalBuild);
            // Mock the getKeychainsAndValidatePassphrase method to throw an error
            const error = new Error('unable to decrypt keychain with the given wallet passphrase');
            error.name = 'wallet_passphrase_incorrect';
            sinon.stub(wallet, 'getKeychainsAndValidatePassphrase').rejects(error);
            await wallet
                .approveErc20Token(walletPassphrase, tokenName)
                .should.be.rejectedWith('unable to decrypt keychain with the given wallet passphrase');
        });
        it('should handle signing errors', async function () {
            const walletPassphrase = 'password123';
            const expectedApprovalBuild = {
                txHex: '0x123456',
                feeInfo: {
                    fee: '1000000000',
                },
            };
            const signingError = new Error('signing failed');
            // Mock the token approval build API
            nock(bgUrl).post(`/api/v2/${coin}/wallet/${walletId}/token/approval/build`).reply(200, expectedApprovalBuild);
            // Mock the getKeychainsAndValidatePassphrase method
            sinon.stub(wallet, 'getKeychainsAndValidatePassphrase').resolves([
                {
                    id: 'keyid1',
                    pub: 'pub1',
                    encryptedPrv: 'encryptedPrv',
                },
            ]);
            // Mock the sign transaction method to throw an error
            sinon.stub(wallet, 'signTransaction').rejects(signingError);
            await wallet.approveErc20Token(walletPassphrase, tokenName).should.be.rejectedWith(signingError);
        });
        it('should handle send transaction errors', async function () {
            const walletPassphrase = 'password123';
            const expectedApprovalBuild = {
                txHex: '0x123456',
                feeInfo: {
                    fee: '1000000000',
                },
            };
            const expectedSignedTx = {
                txHex: '0x123456signed',
            };
            const sendError = new Error('send failed');
            // Mock the token approval build API
            nock(bgUrl).post(`/api/v2/${coin}/wallet/${walletId}/token/approval/build`).reply(200, expectedApprovalBuild);
            // Mock the getKeychainsAndValidatePassphrase method
            sinon.stub(wallet, 'getKeychainsAndValidatePassphrase').resolves([
                {
                    id: 'keyid1',
                    pub: 'pub1',
                    encryptedPrv: 'encryptedPrv',
                },
            ]);
            // Mock the sign transaction method
            sinon.stub(wallet, 'signTransaction').resolves(expectedSignedTx);
            // Mock the send transaction method to throw an error
            sinon.stub(wallet, 'sendTransaction').rejects(sendError);
            await wallet.approveErc20Token(walletPassphrase, tokenName).should.be.rejectedWith(sendError);
        });
    });
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"wallet.js","sourceRoot":"","sources":["../../../../test/v2/unit/wallet.ts"],"names":[],"mappings":"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","sourcesContent":["//\n// Tests for Wallet\n//\n\nimport * as should from 'should';\nimport * as sinon from 'sinon';\nimport '../lib/asserts';\nimport * as nock from 'nock';\nimport * as _ from 'lodash';\n\nimport {\n  common,\n  CustomSigningFunction,\n  ECDSAUtils,\n  EDDSAUtils,\n  RequestTracer,\n  TokenType,\n  TssUtils,\n  TxRequest,\n  Wallet,\n  SignatureShareType,\n  Ecdsa,\n  Keychains,\n  TypedData,\n  TypedMessage,\n  MessageTypes,\n  SignTypedDataVersion,\n  GetUserPrvOptions,\n  ManageUnspentsOptions,\n  SignedMessage,\n  BaseTssUtils,\n  KeyType,\n  SendManyOptions,\n  PopulatedIntent,\n  TxRequestVersion,\n  WalletSignMessageOptions,\n  WalletSignTypedDataOptions,\n  PrebuildTransactionWithIntentOptions,\n} from '@bitgo/sdk-core';\n\nimport { TestBitGo } from '@bitgo/sdk-test';\nimport { BitGo } from '../../../src';\nimport * as utxoLib from '@bitgo/utxo-lib';\nimport { randomBytes } from 'crypto';\nimport { getDefaultWalletKeys, toKeychainObjects } from './coins/utxo/util';\nimport { Tsol } from '@bitgo/sdk-coin-sol';\nimport { Teth } from '@bitgo/sdk-coin-eth';\n\nimport { nftResponse, unsupportedNftResponse } from '../fixtures/nfts/nftResponses';\n\nrequire('should-sinon');\n\nnock.disableNetConnect();\n\ntype CreateTxRequestBody = {\n  intent: PopulatedIntent;\n  apiversion: TxRequestVersion;\n  preview?: boolean;\n};\n\ndescribe('V2 Wallet:', function () {\n  const reqId = new RequestTracer();\n  const bitgo = TestBitGo.decorate(BitGo, { env: 'test' });\n  bitgo.initializeTestVars();\n  const basecoin: any = bitgo.coin('tbtc');\n  const walletData = {\n    id: '5b34252f1bf349930e34020a00000000',\n    coin: 'tbtc',\n    keys: ['5b3424f91bf349930e34017500000000', '5b3424f91bf349930e34017600000000', '5b3424f91bf349930e34017700000000'],\n    coinSpecific: {},\n    multisigType: 'onchain',\n    type: 'hot',\n  };\n  const coldWalletData = {\n    id: '65774419fb4d9690847fbe4b00000000',\n    coin: 'tbtc',\n    keys: ['65774412e54b7516393c9df800000000', '6577442428664ffe791af7ea00000000', '6577442b7317a945756c2fd900000000'],\n    coinSpecific: {},\n    multisigType: 'onchain',\n    type: 'cold',\n  };\n  const wallet = new Wallet(bitgo, basecoin, walletData);\n  const coldWallet = new Wallet(bitgo, basecoin, coldWalletData);\n  const bgUrl = common.Environments[bitgo.getEnv()].uri;\n  const address1 = '0x174cfd823af8ce27ed0afee3fcf3c3ba259116be';\n  const address2 = '0x7e85bdc27c050e3905ebf4b8e634d9ad6edd0de6';\n  const tbtcHotWalletDefaultParams = {\n    txFormat: 'psbt',\n    changeAddressType: ['p2trMusig2', 'p2wsh', 'p2shP2wsh', 'p2sh', 'p2tr'],\n  };\n\n  afterEach(function () {\n    sinon.restore();\n    sinon.reset();\n  });\n\n  describe('Wallet transfers', function () {\n    it('should search in wallet for a transfer', async function () {\n      const params = { limit: 1, searchLabel: 'test' };\n\n      const scope = nock(bgUrl)\n        .get(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/transfer`)\n        .query(params)\n        .reply(200, {\n          coin: 'tbch',\n          transfers: [\n            {\n              wallet: wallet.id(),\n              comment: 'tests',\n            },\n          ],\n        });\n\n      try {\n        await wallet.transfers(params);\n      } catch (e) {\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n\n      scope.isDone().should.be.True();\n    });\n\n    it('should forward all valid parameters', async function () {\n      const params = {\n        limit: 1,\n        address: ['address1', 'address2'],\n        dateGte: 'dateString0',\n        dateLt: 'dateString1',\n        valueGte: 0,\n        valueLt: 300000000,\n        allTokens: true,\n        searchLabel: 'abc',\n        includeHex: true,\n        type: 'transfer_type',\n        state: 'transfer_state',\n      };\n\n      // The actual api request will only send strings, but the SDK function expects numbers for some values\n      const apiParams = _.mapValues(params, (param) => (Array.isArray(param) ? param : String(param)));\n\n      const scope = nock(bgUrl)\n        .get(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/transfer`)\n        .query(_.matches(apiParams))\n        .reply(200);\n\n      await wallet.transfers(params);\n      scope.isDone().should.be.True();\n    });\n\n    it('should accept a string argument for address', async function () {\n      const params = {\n        limit: 1,\n        address: 'stringAddress',\n      };\n\n      const apiParams = {\n        limit: '1',\n        address: 'stringAddress',\n      };\n\n      const scope = nock(bgUrl)\n        .get(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/transfer`)\n        .query(_.matches(apiParams))\n        .reply(200);\n\n      try {\n        await wallet.transfers(params);\n      } catch (e) {\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n\n      scope.isDone().should.be.True();\n    });\n\n    it('should throw errors for invalid expected parameters', async function () {\n      await wallet\n        // @ts-expect-error checking type mismatch\n        .transfers({ address: 13375 })\n        .should.be.rejectedWith('invalid address argument, expecting string or array');\n\n      await wallet\n        // @ts-expect-error checking type mismatch\n        .transfers({ address: [null] })\n        .should.be.rejectedWith('invalid address argument, expecting array of address strings');\n\n      await wallet\n        // @ts-expect-error checking type mismatch\n        .transfers({ dateGte: 20101904 })\n        .should.be.rejectedWith('invalid dateGte argument, expecting string');\n\n      // @ts-expect-error checking type mismatch\n      await wallet.transfers({ dateLt: 20101904 }).should.be.rejectedWith('invalid dateLt argument, expecting string');\n\n      await wallet\n        // @ts-expect-error checking type mismatch\n        .transfers({ valueGte: '10230005' })\n        .should.be.rejectedWith('invalid valueGte argument, expecting number');\n\n      // @ts-expect-error checking type mismatch\n      await wallet.transfers({ valueLt: '-5e8' }).should.be.rejectedWith('invalid valueLt argument, expecting number');\n\n      await wallet\n        // @ts-expect-error checking type mismatch\n        .transfers({ includeHex: '123' })\n        .should.be.rejectedWith('invalid includeHex argument, expecting boolean');\n\n      await wallet\n        // @ts-expect-error checking type mismatch\n        .transfers({ state: 123 })\n        .should.be.rejectedWith('invalid state argument, expecting string or array');\n\n      await wallet\n        // @ts-expect-error checking type mismatch\n        .transfers({ state: [123, 456] })\n        .should.be.rejectedWith('invalid state argument, expecting array of state strings');\n\n      // @ts-expect-error checking type mismatch\n      await wallet.transfers({ type: 123 }).should.be.rejectedWith('invalid type argument, expecting string');\n    });\n  });\n\n  describe('Wallet addresses', function () {\n    it('should search in wallet addresses', async function () {\n      const params = { limit: 1, labelContains: 'test' };\n\n      const scope = nock(bgUrl)\n        .get(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/addresses`)\n        .query(params)\n        .reply(200, {\n          coin: 'tbch',\n          transfers: [\n            {\n              wallet: wallet.id(),\n              comment: 'tests',\n            },\n          ],\n        });\n\n      try {\n        await wallet.addresses(params);\n      } catch (e) {\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n\n      scope.isDone().should.be.True();\n    });\n  });\n\n  it('should verify bch cashaddr format as valid', async function () {\n    const coin = bitgo.coin('tbch');\n    const valid = coin.isValidAddress('bchtest:pzfkxv532t0q5zchck2mhmmf2y02cdejyssq5qrz7a');\n    valid.should.be.True();\n  });\n\n  it('should verify bch legacy format as valid', async function () {\n    const coin = bitgo.coin('tbch');\n    const valid = coin.isValidAddress('2N6gY9r9iuXQQzZiSyngWJeoUuL5mC1x4Ac');\n    valid.should.be.True();\n  });\n\n  describe('TETH Wallet Addresses', function () {\n    let ethWallet;\n\n    before(async function () {\n      const walletData = {\n        id: '598f606cd8fc24710d2ebadb1d9459bb',\n        coin: 'teth',\n        keys: [\n          '598f606cd8fc24710d2ebad89dce86c2',\n          '598f606cc8e43aef09fcb785221d9dd2',\n          '5935d59cf660764331bafcade1855fd7',\n        ],\n      };\n      ethWallet = new Wallet(bitgo, bitgo.coin('teth'), walletData);\n    });\n\n    it('search list addresses should return success', async function () {\n      const params = {\n        includeBalances: true,\n        includeTokens: true,\n        returnBalancesForToken: 'gterc6dp',\n        pendingDeployment: false,\n        includeTotalAddressCount: true,\n      };\n\n      const scope = nock(bgUrl)\n        .get(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/addresses`)\n        .query(params)\n        .reply(200);\n      try {\n        await wallet.addresses(params);\n        throw '';\n      } catch (error) {\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n      scope.isDone().should.be.True();\n    });\n\n    it('should throw errors for invalid expected parameters', async function () {\n      await ethWallet\n        .addresses({ includeBalances: true, returnBalancesForToken: 1 })\n        .should.be.rejectedWith('invalid returnBalancesForToken argument, expecting string');\n\n      await ethWallet\n        .addresses({ pendingDeployment: 1 })\n        .should.be.rejectedWith('invalid pendingDeployment argument, expecting boolean');\n\n      await ethWallet\n        .addresses({ includeBalances: 1 })\n        .should.be.rejectedWith('invalid includeBalances argument, expecting boolean');\n\n      await ethWallet\n        .addresses({ includeTokens: 1 })\n        .should.be.rejectedWith('invalid includeTokens argument, expecting boolean');\n\n      await ethWallet\n        .addresses({ includeTotalAddressCount: 1 })\n        .should.be.rejectedWith('invalid includeTotalAddressCount argument, expecting boolean');\n    });\n\n    it('get forwarder balance', async function () {\n      const forwarders = [\n        {\n          address: '0xbfbcc0fe2b865de877134246af09378e9bc3c91d',\n          balance: '200000',\n        },\n        {\n          address: '0xe59524ed8b47165f4cb0850c9428069a6002e5eb',\n          balance: '10000000000000000',\n        },\n      ];\n\n      nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/forwarders/balances`).reply(200, {\n        forwarders,\n      });\n\n      const forwarderBalance = await ethWallet.getForwarderBalance();\n      forwarderBalance.forwarders[0].address.should.eql(forwarders[0].address);\n      forwarderBalance.forwarders[0].balance.should.eql(forwarders[0].balance);\n      forwarderBalance.forwarders[1].address.should.eql(forwarders[1].address);\n      forwarderBalance.forwarders[1].balance.should.eql(forwarders[1].balance);\n    });\n  });\n\n  describe('Get User Prv', () => {\n    const prv =\n      'xprv9s21ZrQH143K3hekyNj7TciR4XNYe1kMj68W2ipjJGNHETWP7o42AjDnSPgKhdZ4x8NBAvaL72RrXjuXNdmkMqLERZza73oYugGtbLFXG8g';\n    const derivedPrv =\n      'xprv9yoG67Td11uwjXwbV8zEmrySVXERu5FZAsLD9suBeEJbgJqANs8Yng5dEJoii7hag5JermK6PbfxgDmSzW7ewWeLmeJEkmPfmZUSLdETtHx';\n    it('should use the cold derivation seed to derive the proper user private key', async () => {\n      const userPrvOptions = {\n        prv,\n        coldDerivationSeed: '123',\n      };\n      wallet.getUserPrv(userPrvOptions).should.eql(derivedPrv);\n    });\n\n    it('should use the user keychain derivedFromParentWithSeed as the cold derivation seed if none is provided', async () => {\n      const userPrvOptions: GetUserPrvOptions = {\n        prv,\n        keychain: {\n          derivedFromParentWithSeed: '123',\n          id: '456',\n          pub: '789',\n          type: 'independent',\n        },\n      };\n      wallet.getUserPrv(userPrvOptions).should.eql(derivedPrv);\n    });\n\n    it('should prefer the explicit cold derivation seed to the user keychain derivedFromParentWithSeed', async () => {\n      const userPrvOptions: GetUserPrvOptions = {\n        prv,\n        coldDerivationSeed: '123',\n        keychain: {\n          derivedFromParentWithSeed: '456',\n          id: '789',\n          pub: '012',\n          type: 'independent',\n        },\n      };\n      wallet.getUserPrv(userPrvOptions).should.eql(derivedPrv);\n    });\n\n    it('should return the prv provided for TSS SMC', async () => {\n      const tssWalletData = {\n        id: '5b34252f1bf349930e34020a00000000',\n        coin: 'tsol',\n        keys: [\n          '5b3424f91bf349930e34017500000000',\n          '5b3424f91bf349930e34017600000000',\n          '5b3424f91bf349930e34017700000000',\n        ],\n        coinSpecific: {},\n        multisigType: 'tss',\n      };\n      const tsolcoin: any = bitgo.coin('tsol');\n      const wallet = new Wallet(bitgo, tsolcoin, tssWalletData);\n      const prv = 'longstringifiedjson';\n      const keychain = {\n        derivedFromParentWithSeed: 'random seed',\n        id: '123',\n        commonKeychain: 'longstring',\n        type: 'tss' as KeyType,\n      };\n      const userPrvOptions = {\n        prv,\n        keychain,\n      };\n      wallet.getUserPrv(userPrvOptions).should.eql(prv);\n    });\n  });\n\n  describe('UTXO Custom Signer Function', function () {\n    const recipients = [\n      { address: 'abc', amount: 123 },\n      { address: 'def', amount: 456 },\n    ];\n    const rootWalletKey = getDefaultWalletKeys();\n    let customSigningFunction: CustomSigningFunction;\n    let stubs: sinon.SinonStub[];\n\n    beforeEach(function () {\n      customSigningFunction = sinon.stub().returns({\n        txHex: 'this-is-a-tx',\n      });\n      stubs = [\n        sinon.stub(wallet.baseCoin, 'postProcessPrebuild').returnsArg(0),\n        sinon.stub(wallet.baseCoin, 'verifyTransaction').resolves(true),\n        sinon.stub(wallet.baseCoin, 'signTransaction').resolves({ txHex: 'this-is-a-tx' }),\n      ];\n    });\n\n    function nocks(txPrebuild: { txHex: string }) {\n      return nock(bgUrl)\n        .post(wallet.url('/tx/build').replace(bgUrl, ''))\n        .reply(200, txPrebuild)\n        .get(wallet.baseCoin.url('/public/block/latest').replace(bgUrl, ''))\n        .reply(200)\n        .get(wallet.baseCoin.url(`/key/${wallet.keyIds()[0]}`).replace(bgUrl, ''))\n        .reply(200, { pub: 'pub' })\n        .get(wallet.baseCoin.url(`/key/${wallet.keyIds()[1]}`).replace(bgUrl, ''))\n        .reply(200, { pub: 'pub' })\n        .get(wallet.baseCoin.url(`/key/${wallet.keyIds()[2]}`).replace(bgUrl, ''))\n        .reply(200, { pub: 'pub' })\n        .post(wallet.url('/tx/send').replace(bgUrl, ''))\n        .reply(200, { ok: true });\n    }\n\n    it('should use a custom signing function if provided for PSBT with taprootKeyPathSpend input', async function () {\n      const psbt = utxoLib.testutil.constructPsbt(\n        [{ scriptType: 'taprootKeyPathSpend', value: BigInt(1000) }],\n        [{ scriptType: 'p2sh', value: BigInt(900) }],\n        basecoin.network,\n        rootWalletKey,\n        'unsigned'\n      );\n      const scope = nocks({ txHex: psbt.toHex() });\n      const result = await wallet.sendMany({ recipients, customSigningFunction });\n\n      result.should.have.property('ok', true);\n      customSigningFunction.should.have.been.calledTwice();\n      scope.done();\n      stubs.forEach((s) => s.restore());\n    });\n\n    it('should use a custom signing function if provided for PSBT without taprootKeyPathSpend input', async function () {\n      const psbt = utxoLib.testutil.constructPsbt(\n        [{ scriptType: 'p2wsh', value: BigInt(1000) }],\n        [{ scriptType: 'p2sh', value: BigInt(900) }],\n        basecoin.network,\n        rootWalletKey,\n        'unsigned'\n      );\n      const scope = nocks({ txHex: psbt.toHex() });\n      const result = await wallet.sendMany({ recipients, customSigningFunction });\n\n      result.should.have.property('ok', true);\n      customSigningFunction.should.have.been.calledOnce();\n      scope.done();\n      stubs.forEach((s) => s.restore());\n    });\n\n    it('should use a custom signing function if provided for Tx without taprootKeyPathSpend input', async function () {\n      const tx = utxoLib.testutil.constructTxnBuilder(\n        [{ scriptType: 'p2wsh', value: BigInt(1000) }],\n        [{ scriptType: 'p2sh', value: BigInt(900) }],\n        basecoin.network,\n        rootWalletKey,\n        'unsigned'\n      );\n      const scope = nocks({ txHex: tx.buildIncomplete().toHex() });\n      const result = await wallet.sendMany({ recipients, customSigningFunction });\n\n      result.should.have.property('ok', true);\n      customSigningFunction.should.have.been.calledOnce();\n      scope.done();\n      stubs.forEach((s) => s.restore());\n    });\n  });\n\n  describe('TETH Wallet Transactions', function () {\n    let ethWallet;\n\n    before(async function () {\n      const walletData = {\n        id: '598f606cd8fc24710d2ebadb1d9459bb',\n        coin: 'teth',\n        keys: [\n          '598f606cd8fc24710d2ebad89dce86c2',\n          '598f606cc8e43aef09fcb785221d9dd2',\n          '5935d59cf660764331bafcade1855fd7',\n        ],\n        multisigType: 'onchain',\n      };\n      ethWallet = new Wallet(bitgo, bitgo.coin('teth'), walletData);\n    });\n\n    afterEach(async function () {\n      nock.cleanAll();\n    });\n\n    it('should error eip1559 and gasPrice are passed', async function () {\n      const params = {\n        gasPrice: 100,\n        eip1559: {\n          maxPriorityFeePerGas: 10,\n          maxFeePerGas: 10,\n        },\n        amount: 10,\n        address: TestBitGo.V2.TEST_WALLET1_ADDRESS,\n        walletPassphrase: TestBitGo.V2.TEST_WALLET1_PASSCODE,\n      };\n      await ethWallet.send(params).should.be.rejected();\n    });\n\n    it('should search for pending transaction correctly', async function () {\n      const params = { walletId: wallet.id() };\n\n      const scope = nock(bgUrl).get(`/api/v2/${wallet.coin()}/tx/pending/first`).query(params).reply(200);\n      try {\n        await wallet.getFirstPendingTransaction();\n        throw '';\n      } catch (error) {\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n      scope.isDone().should.be.True();\n    });\n\n    it('should try to change the fee correctly', async function () {\n      const params = { txid: '0xffffffff', fee: '10000000' };\n\n      const scope = nock(bgUrl).post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/changeFee`, params).reply(200);\n\n      try {\n        await wallet.changeFee({ txid: '0xffffffff', fee: '10000000' });\n        throw '';\n      } catch (error) {\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n      scope.isDone().should.be.True();\n    });\n\n    it('should try to change the fee correctly using eip1559', async function () {\n      const params = {\n        txid: '0xffffffff',\n        eip1559: {\n          maxPriorityFeePerGas: '1000000000',\n          maxFeePerGas: '25000000000',\n        },\n      };\n\n      const scope = nock(bgUrl).post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/changeFee`, params).reply(200);\n\n      try {\n        await wallet.changeFee(params);\n        throw '';\n      } catch (error) {\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n      scope.isDone().should.be.True();\n    });\n\n    it('should pass data parameter and amount: 0 when using sendTransaction', async function () {\n      const path = `/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/tx/build`;\n      const recipientAddress = '0x7db562c4dd465cc895761c56f83b6af0e32689ba';\n      const recipients = [\n        {\n          address: recipientAddress,\n          amount: 0,\n          data: '0x00110011',\n        },\n      ];\n      const response = nock(bgUrl)\n        .post(path, _.matches({ recipients })) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n\n      const nockKeyChain = nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[0]}`).reply(200, {});\n\n      try {\n        await ethWallet.send({\n          address: recipients[0].address,\n          data: recipients[0].data,\n          amount: recipients[0].amount,\n        });\n      } catch (e) {\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n      response.isDone().should.be.true();\n      nockKeyChain.isDone().should.be.true();\n    });\n\n    it('should pass data parameter and amount: 0 when using sendMany', async function () {\n      const path = `/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/tx/build`;\n      const recipientAddress = '0x7db562c4dd465cc895761c56f83b6af0e32689ba';\n      const recipients = [\n        {\n          address: recipientAddress,\n          amount: 0,\n          data: '0x00110011',\n        },\n      ];\n      const response = nock(bgUrl)\n        .post(path, _.matches({ recipients })) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n\n      const nockKeyChain = nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[0]}`).reply(200, {});\n\n      try {\n        await ethWallet.sendMany({ recipients });\n      } catch (e) {\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n      response.isDone().should.be.true();\n      nockKeyChain.isDone().should.be.true();\n    });\n\n    it('should not pass recipients in sendMany when transaction type is fillNonce', async function () {\n      const recipientAddress = '0x7db562c4dd465cc895761c56f83b6af0e32689ba';\n      const recipients = [\n        {\n          address: recipientAddress,\n          amount: 0,\n        },\n      ];\n      const sendManyParams = { recipients, type: 'fillNonce', isTss: true, nonce: '13' };\n\n      try {\n        await ethWallet.sendMany(sendManyParams);\n      } catch (e) {\n        e.message.should.equal('cannot provide recipients for transaction type fillNonce');\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n    });\n\n    it('should not pass receiveAddress in sendMany when TSS transaction type is transfer or transferToken', async function () {\n      const recipientAddress = '0x7db562c4dd465cc895761c56f83b6af0e32689ba';\n      const recipients = [\n        {\n          address: recipientAddress,\n          amount: 0,\n        },\n      ];\n      const errorMessage = 'cannot use receive address for TSS transactions of type transfer';\n      const sendManyParamsReceiveAddressError = {\n        receiveAddress: 'throw',\n        recipients,\n        type: 'transfer',\n        isTss: true,\n        nonce: '13',\n      };\n      const sendManyParams = { recipients, type: 'transfer', isTss: true, nonce: '13' };\n\n      try {\n        await ethWallet.sendMany(sendManyParamsReceiveAddressError);\n      } catch (e) {\n        e.message.should.equal(errorMessage);\n      }\n\n      try {\n        await ethWallet.sendMany(sendManyParams);\n      } catch (e) {\n        e.message.should.not.equal(errorMessage);\n      }\n    });\n\n    it('should throw error early if password is wrong', async function () {\n      const recipientAddress = '0x7db562c4dd465cc895761c56f83b6af0e32689ba';\n      const recipients = [\n        {\n          address: recipientAddress,\n          amount: 0,\n        },\n      ];\n      const errorMessage = `unable to decrypt keychain with the given wallet passphrase`;\n      const sendManyParamsCorrectPassPhrase = {\n        recipients,\n        type: 'transfer',\n        isTss: true,\n        nonce: '13',\n        walletPassphrase: TestBitGo.V2.TEST_ETH_WALLET_PASSPHRASE,\n      };\n      const nockKeychain = nock(bgUrl)\n        .get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[0]}`)\n        .times(2)\n        .reply(200, {\n          id: '598f606cd8fc24710d2ebad89dce86c2',\n          pub: 'xpub661MyMwAqRbcFXDcWD2vxuebcT1ZpTF4Vke6qmMW8yzddwNYpAPjvYEEL5jLfyYXW2fuxtAxY8TgjPUJLcf1C8qz9N6VgZxArKX4EwB8rH5',\n          ethAddress: '0x26a163ba9739529720c0914c583865dec0d37278',\n          source: 'user',\n          encryptedPrv:\n            '{\"iv\":\"15FsbDVI1zG9OggD8YX+Hg==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"hHbNH3Sz/aU=\",\"ct\":\"WoNVKz7afiRxXI2w/YkzMdMyoQg/B15u1Q8aQgi96jJZ9wk6TIaSEc6bXFH3AHzD9MdJCWJQUpRhoQc/rgytcn69scPTjKeeyVMElGCxZdFVS/psQcNE+lue3//2Zlxj+6t1NkvYO+8yAezSMRBK5OdftXEjNQI=\"}',\n          coinSpecific: {},\n        });\n\n      await ethWallet\n        .sendMany({ ...sendManyParamsCorrectPassPhrase, walletPassphrase: 'wrongPassphrase' })\n        .should.be.rejectedWith(errorMessage);\n\n      try {\n        const customSigningFunction = () => {\n          return 'mock';\n        };\n        // Should not validate passphrase if custom signing function is provided\n        await ethWallet.sendMany({\n          ...sendManyParamsCorrectPassPhrase,\n          walletPassphrase: 'wrongPassphrase',\n          customSigningFunction,\n        });\n      } catch (e) {\n        e.message.should.not.equal(errorMessage);\n      }\n      try {\n        await ethWallet.sendMany({ ...sendManyParamsCorrectPassPhrase });\n      } catch (e) {\n        e.message.should.not.equal(errorMessage);\n      }\n      nockKeychain.isDone().should.be.true();\n    });\n  });\n\n  describe('OFC Create Address', () => {\n    let ofcWallet: Wallet;\n    let nocks;\n    before(async function () {\n      const walletDataOfc = {\n        id: '5b34252f1bf349930e3400b00000000',\n        coin: 'ofc',\n        keys: [\n          '5b3424f91bf349930e34017800000000',\n          '5b3424f91bf349930e34017900000000',\n          '5b3424f91bf349930e34018000000000',\n        ],\n        coinSpecific: {},\n        multisigType: 'onchain',\n      };\n      ofcWallet = new Wallet(bitgo, bitgo.coin('ofc'), walletDataOfc);\n    });\n\n    beforeEach(async function () {\n      nocks = [\n        nock(bgUrl).get(`/api/v2/ofc/key/${ofcWallet.keyIds()[0]}`).reply(200, {\n          id: ofcWallet.keyIds()[0],\n          pub: 'xpub661MyMwAqRbcFXDcWD2vxuebcT1ZpTF4Vke6qmMW8yzddwNYpAPjvYEEL5jLfyYXW2fuxtAxY8TgjPUJLcf1C8qz9N6VgZxArKX4EwB8rH5',\n          source: 'user',\n          encryptedPrv:\n            '{\"iv\":\"15FsbDVI1zG9OggD8YX+Hg==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"hHbNH3Sz/aU=\",\"ct\":\"WoNVKz7afiRxXI2w/YkzMdMyoQg/B15u1Q8aQgi96jJZ9wk6TIaSEc6bXFH3AHzD9MdJCWJQUpRhoQc/rgytcn69scPTjKeeyVMElGCxZdFVS/psQcNE+lue3//2Zlxj+6t1NkvYO+8yAezSMRBK5OdftXEjNQI=\"}',\n          coinSpecific: {},\n        }),\n\n        nock(bgUrl).get(`/api/v2/ofc/key/${ofcWallet.keyIds()[1]}`).reply(200, {\n          id: ofcWallet.keyIds()[1],\n          pub: 'xpub661MyMwAqRbcGhSaXikpuTC9KU88Xx9LrjKSw1JKsvXNgabpTdgjy7LSovh9ZHhcqhAHQu7uthu7FguNGdcC4aXTKK5gqTcPe4WvLYRbCSG',\n          source: 'backup',\n          coinSpecific: {},\n        }),\n\n        nock(bgUrl).get(`/api/v2/ofc/key/${ofcWallet.keyIds()[2]}`).reply(200, {\n          id: ofcWallet.keyIds()[2],\n          pub: 'xpub661MyMwAqRbcFsXShW8R3hJsHNTYTUwzcejnLkY7KCtaJbDqcGkcBF99BrEJSjNZHeHveiYUrsAdwnjUMGwpgmEbiKcZWRuVA9HxnRaA3r3',\n          source: 'bitgo',\n          coinSpecific: {},\n        }),\n      ];\n    });\n\n    afterEach(async function () {\n      nock.cleanAll();\n      nocks.forEach((scope) => scope.isDone().should.be.true());\n    });\n\n    it('should correctly validate arguments to create address on OFC wallet', async function () {\n      await ofcWallet.createAddress().should.be.rejectedWith('onToken is a mandatory parameter for OFC wallets');\n      // @ts-expect-error test passing invalid number argument\n      await ofcWallet.createAddress({ onToken: 42 }).should.be.rejectedWith('onToken has to be a string');\n    });\n\n    it('address creation with valid onToken argument succeeds', async function () {\n      const scope = nock(bgUrl)\n        .post(`/api/v2/ofc/wallet/${ofcWallet.id()}/address`, { onToken: 'ofctbtc' })\n        .reply(200, {\n          id: '638a48c6c3dba40007a3497fa49a080c',\n          address: 'generated address',\n          chain: 0,\n          index: 1,\n          coin: 'tbtc',\n          wallet: ofcWallet.id,\n        });\n      const address = await ofcWallet.createAddress({ onToken: 'ofctbtc' });\n      address.address.should.equal('generated address');\n      scope.isDone().should.be.true();\n    });\n  });\n\n  describe('TETH Create Address', () => {\n    let ethWallet, nocks;\n    const walletData = {\n      id: '598f606cd8fc24710d2ebadb1d9459bb',\n      coinSpecific: {\n        baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n      },\n      coin: 'teth',\n      keys: [\n        '598f606cd8fc24710d2ebad89dce86c2',\n        '598f606cc8e43aef09fcb785221d9dd2',\n        '5935d59cf660764331bafcade1855fd7',\n      ],\n    };\n\n    beforeEach(async function () {\n      ethWallet = new Wallet(bitgo, bitgo.coin('teth'), walletData);\n      nocks = [\n        nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[0]}`).reply(200, {\n          id: '598f606cd8fc24710d2ebad89dce86c2',\n          pub: 'xpub661MyMwAqRbcFXDcWD2vxuebcT1ZpTF4Vke6qmMW8yzddwNYpAPjvYEEL5jLfyYXW2fuxtAxY8TgjPUJLcf1C8qz9N6VgZxArKX4EwB8rH5',\n          ethAddress: '0x26a163ba9739529720c0914c583865dec0d37278',\n          source: 'user',\n          encryptedPrv:\n            '{\"iv\":\"15FsbDVI1zG9OggD8YX+Hg==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"hHbNH3Sz/aU=\",\"ct\":\"WoNVKz7afiRxXI2w/YkzMdMyoQg/B15u1Q8aQgi96jJZ9wk6TIaSEc6bXFH3AHzD9MdJCWJQUpRhoQc/rgytcn69scPTjKeeyVMElGCxZdFVS/psQcNE+lue3//2Zlxj+6t1NkvYO+8yAezSMRBK5OdftXEjNQI=\"}',\n          coinSpecific: {},\n        }),\n\n        nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[1]}`).reply(200, {\n          id: '598f606cc8e43aef09fcb785221d9dd2',\n          pub: 'xpub661MyMwAqRbcGhSaXikpuTC9KU88Xx9LrjKSw1JKsvXNgabpTdgjy7LSovh9ZHhcqhAHQu7uthu7FguNGdcC4aXTKK5gqTcPe4WvLYRbCSG',\n          ethAddress: '0xa1a88a502274073b1bc4fe06ea0f5fe77e151b91',\n          source: 'backup',\n          coinSpecific: {},\n        }),\n\n        nock(bgUrl).get(`/api/v2/${ethWallet.coin()}/key/${ethWallet.keyIds()[2]}`).reply(200, {\n          id: '5935d59cf660764331bafcade1855fd7',\n          pub: 'xpub661MyMwAqRbcFsXShW8R3hJsHNTYTUwzcejnLkY7KCtaJbDqcGkcBF99BrEJSjNZHeHveiYUrsAdwnjUMGwpgmEbiKcZWRuVA9HxnRaA3r3',\n          ethAddress: '0x032821b7ea40ea5d446f47c29a0f777ee035aa10',\n          source: 'bitgo',\n          coinSpecific: {},\n        }),\n      ];\n    });\n\n    afterEach(async function () {\n      nock.cleanAll();\n      nocks.forEach((scope) => scope.isDone().should.be.true());\n    });\n\n    it('should correctly validate arguments to create address', async function () {\n      let message = 'gasPrice has to be an integer or numeric string';\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ gasPrice: {} }).should.be.rejectedWith(message);\n      await wallet.createAddress({ gasPrice: 'abc' }).should.be.rejectedWith(message);\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ gasPrice: null }).should.be.rejectedWith(message);\n\n      message = 'chain has to be an integer';\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ chain: {} }).should.be.rejectedWith(message);\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ chain: 'abc' }).should.be.rejectedWith(message);\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ chain: null }).should.be.rejectedWith(message);\n\n      message = 'count has to be a number between 1 and 250';\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ count: {} }).should.be.rejectedWith(message);\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ count: 'abc' }).should.be.rejectedWith(message);\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ count: null }).should.be.rejectedWith(message);\n      await wallet.createAddress({ count: -1 }).should.be.rejectedWith(message);\n      await wallet.createAddress({ count: 0 }).should.be.rejectedWith(message);\n      await wallet.createAddress({ count: 251 }).should.be.rejectedWith(message);\n\n      message = 'baseAddress has to be a string';\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ baseAddress: {} }).should.be.rejectedWith(message);\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ baseAddress: 123 }).should.be.rejectedWith(message);\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ baseAddress: null }).should.be.rejectedWith(message);\n\n      message = 'allowSkipVerifyAddress has to be a boolean';\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ allowSkipVerifyAddress: {} }).should.be.rejectedWith(message);\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ allowSkipVerifyAddress: 123 }).should.be.rejectedWith(message);\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ allowSkipVerifyAddress: 'abc' }).should.be.rejectedWith(message);\n      // @ts-expect-error checking type mismatch\n      await wallet.createAddress({ allowSkipVerifyAddress: null }).should.be.rejectedWith(message);\n\n      message = 'forwarderVersion has to be an integer 0, 1, 2, 3 or 4';\n      await wallet.createAddress({ forwarderVersion: 5 }).should.be.rejectedWith(message);\n      await wallet.createAddress({ forwarderVersion: -1 }).should.be.rejectedWith(message);\n    });\n\n    it('address creation with forwarder version 3 succeeds', async function () {\n      const scope = nock(bgUrl)\n        .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 3 })\n        .reply(200, {\n          id: '638a48c6c3dba40007a3497fa49a080c',\n          address: '0x5e61b64f38f1b5f85078fb84b27394830b4c8e80',\n          chain: 0,\n          index: 1,\n          coin: 'tpolygon',\n          lastNonce: 0,\n          wallet: '63785f95af7c760007cfae068c2f31ae',\n          coinSpecific: {\n            nonce: -1,\n            updateTime: '2022-12-02T18:49:42.348Z',\n            txCount: 0,\n            pendingChainInitialization: false,\n            creationFailure: [],\n            salt: '0x1',\n            pendingDeployment: true,\n            forwarderVersion: 3,\n            isTss: true,\n          },\n        });\n      const address = await ethWallet.createAddress({ chain: 0, forwarderVersion: 3 });\n      address.coinSpecific.forwarderVersion.should.equal(3);\n      scope.isDone().should.be.true();\n    });\n\n    it('address creation with forwarder version 3 fails due invalid address', async function () {\n      const address = '0x5e61b6'; // invalid address\n      nock(bgUrl)\n        .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 3 })\n        .reply(200, {\n          id: '638a48c6c3dba40007a3497fa49a080c',\n          address: address,\n          chain: 0,\n          index: 1,\n          coin: 'tpolygon',\n          lastNonce: 0,\n          wallet: '63785f95af7c760007cfae068c2f31ae',\n          coinSpecific: {\n            nonce: -1,\n            updateTime: '2022-12-02T18:49:42.348Z',\n            txCount: 0,\n            pendingChainInitialization: false,\n            creationFailure: [],\n            salt: '0x1',\n            pendingDeployment: true,\n            forwarderVersion: 3,\n            isTss: true,\n          },\n        });\n      await ethWallet\n        .createAddress({ chain: 0, forwarderVersion: 3 })\n        .should.be.rejectedWith(`invalid address: ${address}`);\n    });\n\n    it('address creation with forwarder version 2 succeeds', async function () {\n      const scope = nock(bgUrl)\n        .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 2 })\n        .reply(200, {\n          id: '638a48c6c3dba40007a3497fa49a080c',\n          address: '0x5e61b64f38f1b5f85078fb84b27394830b4c8e80',\n          chain: 0,\n          index: 1,\n          coin: 'tpolygon',\n          lastNonce: 0,\n          wallet: '63785f95af7c760007cfae068c2f31ae',\n          coinSpecific: {\n            nonce: -1,\n            updateTime: '2022-12-02T18:49:42.348Z',\n            txCount: 0,\n            pendingChainInitialization: true,\n            creationFailure: [],\n            salt: '0x1',\n            pendingDeployment: true,\n            forwarderVersion: 2,\n            isTss: true,\n          },\n        });\n      const address = await ethWallet.createAddress({ chain: 0, forwarderVersion: 2 });\n      address.coinSpecific.forwarderVersion.should.equal(2);\n      scope.isDone().should.be.true();\n    });\n\n    it('verify address when pendingChainInitialization is true in case of eth v1 forwarder', async function () {\n      const scope = nock(bgUrl)\n        .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 1 })\n        .reply(200, {\n          id: '615c643a98a2a100068e023c639c0f74',\n          address: '0x8c13cd0bb198858f628d5631ba4b2293fc08df49',\n          baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n          chain: 0,\n          index: 3179,\n          coin: 'teth',\n          lastNonce: 0,\n          wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n          coinSpecific: {\n            nonce: -1,\n            updateTime: '2021-10-05T14:42:02.399Z',\n            txCount: 0,\n            pendingChainInitialization: true,\n            creationFailure: [],\n            salt: '0xc6b',\n            pendingDeployment: true,\n            forwarderVersion: 1,\n          },\n        });\n      await ethWallet\n        .createAddress({ chain: 0, forwarderVersion: 1 })\n        .should.be.rejectedWith(\n          'address validation failure: expected 0x32a226cda14e352a47bf4b1658648d8037736f80 but got 0x8c13cd0bb198858f628d5631ba4b2293fc08df49'\n        );\n      scope.isDone().should.be.true();\n    });\n\n    it('verify address when invalid baseAddress is passed', async function () {\n      const scope = nock(bgUrl)\n        .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 1 })\n        .reply(200, {\n          id: '615c643a98a2a100068e023c639c0f74',\n          address: '0x32a226cda14e352a47bf4b1658648d8037736f80',\n          baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n          chain: 0,\n          index: 3179,\n          coin: 'teth',\n          lastNonce: 0,\n          wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n          coinSpecific: {\n            nonce: -1,\n            updateTime: '2021-10-05T14:42:02.399Z',\n            txCount: 0,\n            pendingChainInitialization: true,\n            creationFailure: [],\n            salt: '0xc6b',\n            pendingDeployment: true,\n            forwarderVersion: 1,\n          },\n        });\n      await ethWallet\n        .createAddress({ chain: 0, forwarderVersion: 1, baseAddress: 'asgf' })\n        .should.be.rejectedWith('invalid base address');\n      scope.isDone().should.be.true();\n    });\n\n    it('verify address when incorrect baseAddress is passed', async function () {\n      const scope = nock(bgUrl)\n        .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 1 })\n        .reply(200, {\n          id: '615c643a98a2a100068e023c639c0f74',\n          address: '0x32a226cda14e352a47bf4b1658648d8037736f80',\n          baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n          chain: 0,\n          index: 3179,\n          coin: 'teth',\n          lastNonce: 0,\n          wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n          coinSpecific: {\n            nonce: -1,\n            updateTime: '2021-10-05T14:42:02.399Z',\n            txCount: 0,\n            pendingChainInitialization: true,\n            creationFailure: [],\n            salt: '0xc6b',\n            pendingDeployment: true,\n            forwarderVersion: 1,\n          },\n        });\n      // incorrect address is generated while validating due to incorrect baseAddress\n      await ethWallet\n        .createAddress({ chain: 0, forwarderVersion: 1, baseAddress: '0x8c13cd0bb198858f628d5631ba4b2293fc08df49' })\n        .should.be.rejectedWith(\n          'address validation failure: expected 0x36748926007790e7ee416c6485b32e00cfb177a3 but got 0x32a226cda14e352a47bf4b1658648d8037736f80'\n        );\n      scope.isDone().should.be.true();\n    });\n\n    it('verify address when pendingChainInitialization is true  and allowSkipVerifyAddress is false in case of eth v0 forwarder', async function () {\n      const scope = nock(bgUrl)\n        .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 0 })\n        .reply(200, {\n          id: '615c643a98a2a100068e023c639c0f74',\n          address: '0x32a26cda14e352a47bf4b1658648d8037736f80',\n          baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n          chain: 0,\n          index: 3179,\n          coin: 'teth',\n          lastNonce: 0,\n          wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n          coinSpecific: {\n            nonce: -1,\n            updateTime: '2021-10-05T14:42:02.399Z',\n            txCount: 0,\n            pendingChainInitialization: true,\n            creationFailure: [],\n            salt: '0xc6b',\n            pendingDeployment: true,\n            forwarderVersion: 1,\n          },\n        });\n      await ethWallet\n        .createAddress({ chain: 0, forwarderVersion: 0, allowSkipVerifyAddress: false })\n        .should.be.rejectedWith('address verification skipped for count = 1');\n      scope.isDone().should.be.true();\n    });\n\n    it('verify address with allowSkipVerifyAddress set to false and eth v1 forwarder', async function () {\n      const scope = nock(bgUrl)\n        .post(`/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/address`, { chain: 0, forwarderVersion: 1 })\n        .reply(200, {\n          id: '615c643a98a2a100068e023c639c0f74',\n          address: '0x32a226cda14e352a47bf4b1658648d8037736f80',\n          baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n          chain: 0,\n          index: 3179,\n          coin: 'teth',\n          lastNonce: 0,\n          wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n          coinSpecific: {\n            nonce: -1,\n            updateTime: '2021-10-05T14:42:02.399Z',\n            txCount: 0,\n            pendingChainInitialization: true,\n            creationFailure: [],\n            salt: '0xc6b',\n            pendingDeployment: true,\n            forwarderVersion: 0,\n          },\n        });\n      const newAddress = await ethWallet.createAddress({\n        chain: 0,\n        forwarderVersion: 1,\n        allowSkipVerifyAddress: false,\n      });\n      newAddress.index.should.equal(3179);\n      scope.isDone().should.be.true();\n    });\n  });\n\n  describe('Algorand tests', () => {\n    let algoWallet: Wallet;\n\n    before(async () => {\n      // This is not a real TALGO wallet\n      const walletData = {\n        id: '650204cf43d8b40007cd9e11a872ce65',\n        coin: 'talgo',\n        keys: [\n          '650204b78a75c90007790bce979ae34d',\n          '650204b766c56a00072956c08fb9cdf1',\n          '650204b8ccf1370007b32bb8155dfbec',\n        ],\n        coinSpecific: {\n          rootAddress: '2ULRGE64U7LTMT5M6REB7ORHX5GLJYWHTIV5EAXVLWQTTATVJDGM5KJMII',\n        },\n      };\n      algoWallet = new Wallet(bitgo, bitgo.coin('talgo'), walletData);\n    });\n\n    it('Should build token enablement transactions', async () => {\n      const params = {\n        enableTokens: [\n          {\n            name: 'talgo:USDt-180447',\n          },\n        ],\n      };\n      const txRequestNock = nock(bgUrl)\n        .post(`/api/v2/${algoWallet.coin()}/wallet/${algoWallet.id()}/tx/build`)\n        .reply((uri, body) => {\n          const params = body as any;\n          params.recipients.length.should.equal(1);\n          params.recipients[0].tokenName.should.equal('talgo:USDt-180447');\n          params.type.should.equal('enabletoken');\n          should.not.exist(params.enableTokens);\n          return [200, params];\n        });\n      await algoWallet.buildTokenEnablements(params);\n      txRequestNock.isDone().should.equal(true);\n    });\n\n    afterEach(() => {\n      nock.cleanAll();\n    });\n  });\n\n  describe('Hedera tests', () => {\n    let hbarWallet: Wallet;\n\n    before(async () => {\n      // This is not a real THBAR wallet\n      const walletData = {\n        id: '598f606cd8fc24710d2ebadb1d9459bb',\n        coin: 'thbar',\n        keys: [\n          '598f606cd8fc24710d2ebad89dce86c2',\n          '598f606cc8e43aef09fcb785221d9dd2',\n          '5935d59cf660764331bafcade1855fd7',\n        ],\n        coinSpecific: {\n          baseAddress: '0.0.47841511',\n        },\n      };\n      hbarWallet = new Wallet(bitgo, bitgo.coin('thbar'), walletData);\n    });\n\n    it('Should build token enablement transactions', async () => {\n      const params = {\n        enableTokens: [\n          {\n            name: 'thbar:usdc',\n          },\n        ],\n      };\n      const txRequestNock = nock(bgUrl)\n        .post(`/api/v2/${hbarWallet.coin()}/wallet/${hbarWallet.id()}/tx/build`)\n        .reply((uri, body) => {\n          const params = body as any;\n          params.recipients.length.should.equal(1);\n          params.recipients[0].tokenName.should.equal('thbar:usdc');\n          params.type.should.equal('enabletoken');\n          should.not.exist(params.enableTokens);\n          return [200, params];\n        });\n      await hbarWallet.buildTokenEnablements(params);\n      txRequestNock.isDone().should.equal(true);\n    });\n\n    afterEach(() => {\n      nock.cleanAll();\n    });\n  });\n\n  describe('Solana tests: ', () => {\n    let solWallet: Wallet;\n    const passphrase = '#Bondiola1234';\n    const solBitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    solBitgo.initializeTestVars();\n    const walletData = {\n      id: '598f606cd8fc24710d2ebadb1d9459bb',\n      coinSpecific: {\n        baseAddress: '5f8WmC2uW9SAk7LMX2r4G1Bx8MMwx8sdgpotyHGodiZo',\n        pendingChainInitialization: false,\n        minimumFunding: 2447136,\n        lastChainIndex: { 0: 0 },\n      },\n      coin: 'tsol',\n      keys: [\n        '598f606cd8fc24710d2ebad89dce86c2',\n        '598f606cc8e43aef09fcb785221d9dd2',\n        '5935d59cf660764331bafcade1855fd7',\n      ],\n      multisigType: 'tss',\n    };\n\n    before(async function () {\n      solWallet = new Wallet(bitgo, bitgo.coin('tsol'), walletData);\n      nock(bgUrl).get(`/api/v2/${solWallet.coin()}/key/${solWallet.keyIds()[0]}`).times(3).reply(200, {\n        id: '598f606cd8fc24710d2ebad89dce86c2',\n        pub: '5f8WmC2uW9SAk7LMX2r4G1Bx8MMwx8sdgpotyHGodiZo',\n        source: 'user',\n        encryptedPrv:\n          '{\"iv\":\"hNK3rg82P1T94MaueXFAbA==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"cV4wU4EzPjs=\",\"ct\":\"9VZX99Ztsb6p75Cxl2lrcXBplmssIAQ9k7ZA81vdDYG4N5dZ36BQNWVfDoelj9O31XyJ+Xri0XKIWUzl0KKLfUERplmtNoOCn5ifJcZwCrOxpHZQe3AJ700o8Wmsrk5H\"}',\n        coinSpecific: {},\n      });\n\n      nock(bgUrl).get(`/api/v2/${solWallet.coin()}/key/${solWallet.keyIds()[1]}`).times(2).reply(200, {\n        id: '598f606cc8e43aef09fcb785221d9dd2',\n        pub: 'G1s43JTzNZzqhUn4aNpwgcc6wb9FUsZQD5JjffG6isyd',\n        encryptedPrv:\n          '{\"iv\":\"UFrt/QlIUR1XeQafPBaAlw==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"7VPBYaJXPm8=\",\"ct\":\"ajFKv2y8yaIBXQ39sAbBWcnbiEEzbjS4AoQtp5cXYqjeDRxt3aCxemPm22pnkJaCijFjJrMHbkmsNhNYzHg5aHFukN+nEAVssyNwHbzlhSnm8/BVN50yAdAAtWreh8cp\"}',\n        source: 'backup',\n        coinSpecific: {},\n      });\n\n      nock(bgUrl).get(`/api/v2/${solWallet.coin()}/key/${solWallet.keyIds()[2]}`).times(2).reply(200, {\n        id: '5935d59cf660764331bafcade1855fd7',\n        pub: 'GH1LV1e9FdqGe8U2c8PMEcma3fDeh1ktcGVBrD3AuFqx',\n        encryptedPrv:\n          '{\"iv\":\"iIuWOHIOErEDdiJn6g46mg==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"Rzh7RRJksj0=\",\"ct\":\"rcNICUfp9FakT53l+adB6XKzS1vNTc0Qq9jAtqnxA+ScssiS4Q0l3sgG/0gDy5DaZKtXryKBDUvGsi7b/fYaFCUpAoZn/VZTOhOUN/mo7ZHb4OhOXL29YPPkiryAq9Cr\"}',\n        source: 'bitgo',\n        coinSpecific: {},\n      });\n    });\n\n    after(async function () {\n      nock.cleanAll();\n    });\n\n    describe('prebuildAndSignTransaction: ', function () {\n      // TODO (STLX-15018): fix test\n      xit('should successfully sign a consolidation transfer', async function () {\n        const txParams = {\n          prebuildTx: {\n            walletId: walletData.id,\n            txHex:\n              'AQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABAAIE9MWWV2ct01mg5Gm4EqcJ9SAn2XuD+FuAHcHFTkc1Tgut3DgTsiSgTQ0dmzj5JJg6qYTpn8FxOYPFCFTMoZi46gAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABUpTWpkpIQZNJOhxYNo4fHw1td28kruB5B+oQEEFRI0Qc+q0Zg6OOpV8eCDVLfYziox7YBA7+QPLX4IRhDCSKwICAgABDAIAAACghgEAAAAAAAMAFVRlc3QgaW50ZWdyYXRpb24gbWVtbw==',\n            txInfo: {\n              feePayer: 'HUVE5NfJyGfU1djZsVLA6fxSTS1E2iRqcTRVNC9K2z7c',\n              lamportsPerSignature: 5000,\n              nonce: '27E3MXFvXMUNYeMJeX1pAbERGsJfUbkaZTfgMgpmNN5g',\n              numSignatures: 0,\n              instructionsData: [\n                {\n                  type: 'Transfer',\n                  params: {\n                    fromAddress: 'HUVE5NfJyGfU1djZsVLA6fxSTS1E2iRqcTRVNC9K2z7c',\n                    toAddress: 'ChgJ5tgDwBUsk9RNMm2iLiwP8RodwgZ6uqrC5paJsXVT',\n                    amount: '100000',\n                  },\n                },\n                {\n                  type: 'Memo',\n                  params: {\n                    memo: 'Test integration memo',\n                  },\n                },\n              ],\n            },\n            buildParams: {\n              memo: {\n                type: 'Memo',\n                value: 'Test integration memo',\n              },\n              recipients: [\n                {\n                  address: 'ChgJ5tgDwBUsk9RNMm2iLiwP8RodwgZ6uqrC5paJsXVT',\n                  amount: '100000',\n                },\n              ],\n              type: 'transfer',\n            },\n            consolidateId: '1234',\n            consolidationDetails: {\n              senderAddressIndex: 1,\n            },\n          },\n          walletPassphrase: passphrase,\n        };\n        // Build and sign the transaction\n        const preBuiltSignedTx = await solWallet.prebuildAndSignTransaction(txParams);\n        preBuiltSignedTx.should.have.property('txHex');\n      });\n    });\n\n    it('Should build token enablement transactions correctly', async function () {\n      const params = {\n        enableTokens: [{ name: 'tsol:usdc' }, { name: 'tsol:srm' }, { name: 'tsol:gmt' }],\n      };\n      const txRequestNock = nock(bgUrl)\n        .post(`/api/v2/wallet/${solWallet.id()}/txrequests`)\n        .reply((url, body) => {\n          const bodyParams = body as any;\n          bodyParams.intent.intentType.should.equal('enableToken');\n          bodyParams.intent.recipients.length.should.equal(0);\n          bodyParams.intent.enableTokens.should.deepEqual(params.enableTokens);\n          return [\n            200,\n            {\n              apiVersion: 'full',\n              transactions: [\n                {\n                  unsignedTx: {\n                    serializedTxHex: 'fake transaction',\n                    feeInfo: 'fake fee info',\n                  },\n                },\n              ],\n            },\n          ];\n        });\n      await solWallet.buildTokenEnablements(params);\n      txRequestNock.isDone().should.equal(true);\n    });\n  });\n\n  describe('Accelerate Transaction', function () {\n    it('fails if acceleration ids are not passed', async function () {\n      await wallet.accelerateTransaction({}).should.be.rejectedWith({ code: 'cpfptxids_or_rbftxids_required' });\n    });\n\n    it('fails if cpfpTxIds is not an array', async function () {\n      // @ts-expect-error checking type mismatch\n      await wallet.accelerateTransaction({ cpfpTxIds: {} }).should.be.rejectedWith({ code: 'cpfptxids_not_array' });\n    });\n\n    it('fails if cpfpTxIds is not of length 1', async function () {\n      await wallet.accelerateTransaction({ cpfpTxIds: [] }).should.be.rejectedWith({ code: 'cpfptxids_not_array' });\n      await wallet\n        .accelerateTransaction({ cpfpTxIds: ['id1', 'id2'] })\n        .should.be.rejectedWith({ code: 'cpfptxids_not_array' });\n    });\n\n    it('fails if cpfpFeeRate is not passed and neither is noCpfpFeeRate', async function () {\n      await wallet.accelerateTransaction({ cpfpTxIds: ['id'] }).should.be.rejectedWith({ code: 'cpfpfeerate_not_set' });\n    });\n\n    it('fails if cpfpFeeRate is not an integer', async function () {\n      await wallet\n        // @ts-expect-error checking type mismatch\n        .accelerateTransaction({ cpfpTxIds: ['id'], cpfpFeeRate: 'one' })\n        .should.be.rejectedWith({ code: 'cpfpfeerate_not_nonnegative_integer' });\n    });\n\n    it('fails if cpfpFeeRate is negative', async function () {\n      await wallet\n        .accelerateTransaction({ cpfpTxIds: ['id'], cpfpFeeRate: -1 })\n        .should.be.rejectedWith({ code: 'cpfpfeerate_not_nonnegative_integer' });\n    });\n\n    it('fails if maxFee is not passed and neither is noMaxFee', async function () {\n      await wallet\n        .accelerateTransaction({ cpfpTxIds: ['id'], noCpfpFeeRate: true })\n        .should.be.rejectedWith({ code: 'maxfee_not_set' });\n    });\n\n    it('fails if maxFee is not an integer', async function () {\n      await wallet\n        // @ts-expect-error checking type mismatch\n        .accelerateTransaction({ cpfpTxIds: ['id'], noCpfpFeeRate: true, maxFee: 'one' })\n        .should.be.rejectedWith({ code: 'maxfee_not_nonnegative_integer' });\n    });\n\n    it('fails if maxFee is negative', async function () {\n      await wallet\n        .accelerateTransaction({ cpfpTxIds: ['id'], noCpfpFeeRate: true, maxFee: -1 })\n        .should.be.rejectedWith({ code: 'maxfee_not_nonnegative_integer' });\n    });\n\n    it('fails if both rbfTxids and cpfpTxids is set', async function () {\n      await wallet\n        .accelerateTransaction({ cpfpTxIds: ['id1'], rbfTxIds: ['id2'] })\n        .should.be.rejectedWith({ code: 'cannot_specify_both_cpfp_and_rbf_txids' });\n    });\n\n    it('fails if rbfTxIds is set but feeMultiplier is missing', async function () {\n      await wallet\n        .accelerateTransaction({ rbfTxIds: ['id'] })\n        .should.be.rejectedWith({ code: 'feemultiplier_not_set' });\n    });\n\n    it('fails if fee multiplier is less than or equal to 1', async function () {\n      await wallet\n        .accelerateTransaction({ rbfTxIds: ['id'], feeMultiplier: 1 })\n        .should.be.rejectedWith({ code: 'feemultiplier_greater_than_one' });\n\n      await wallet\n        .accelerateTransaction({ rbfTxIds: ['id2'], feeMultiplier: 0.5 })\n        .should.be.rejectedWith({ code: 'feemultiplier_greater_than_one' });\n    });\n\n    it('submits a transaction with all cpfp specific parameters', async function () {\n      const params = {\n        cpfpTxIds: ['id'],\n        cpfpFeeRate: 1,\n        maxFee: 1,\n      };\n\n      const prebuildReturn = Object.assign({ txHex: '123' }, params);\n      const prebuildStub = sinon.stub(wallet, 'prebuildAndSignTransaction').resolves(prebuildReturn);\n\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/send`;\n      nock(bgUrl).post(path, _.matches(prebuildReturn)).reply(200);\n\n      await wallet.accelerateTransaction(params);\n\n      prebuildStub.should.have.been.calledOnceWith(params);\n\n      sinon.restore();\n    });\n  });\n\n  describe('fanout input maxNumInputsToUse and unspents verification', function () {\n    const address = '5b34252f1bf349930e34020a';\n    const maxNumInputsToUse = 2;\n    const unspents = [\n      'cc30565750e2aeb818625aaedaf89db5c614e5977b9645cee1d7289f616fb1d8:0',\n      '8c45164787a954ab07864af9b05b34fbde3a8e430a8c65b0e60e4e543d8e1b6c:2',\n    ];\n    let basecoin;\n    let wallet;\n\n    before(async function () {\n      basecoin = bitgo.coin('tbtc');\n      const walletData = {\n        id: '5b34252f1bf349930e34020a',\n        coin: 'tbtc',\n        keys: ['5b3424f91bf349930e340175'],\n      };\n      wallet = new Wallet(bitgo, basecoin, walletData);\n    });\n\n    it('should pass maxNumInputsToUse parameter when calling fanout unspents', async function () {\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/fanoutUnspents`;\n      const response = nock(bgUrl)\n        .post(path, _.matches({ maxNumInputsToUse })) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n\n      try {\n        await wallet.fanoutUnspents({ address, maxNumInputsToUse });\n      } catch (e) {\n        // the fanoutUnspents method will probably throw an exception for not having all of the correct nocks\n        // we only care about /fanoutUnspents and whether maxNumInputsToUse is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n    });\n\n    it('should pass unspents parameter when calling fanout unspents', async function () {\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/fanoutUnspents`;\n      const response = nock(bgUrl)\n        .post(path, _.matches({ unspents })) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n\n      try {\n        await wallet.fanoutUnspents({ address, unspents });\n      } catch (e) {\n        // the fanoutUnspents method will probably throw an exception for not having all of the correct nocks\n        // we only care about /fanoutUnspents and whether unspents is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n    });\n\n    it('should only build tx (not sign/send) while fanning out unspents', async function () {\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/fanoutUnspents`;\n      const response = nock(bgUrl).post(path, _.matches({ unspents })).reply(200);\n\n      const unusedNocks = nock(bgUrl);\n      unusedNocks.get(`/api/v2/${wallet.coin()}/key/${wallet.keyIds()[0]}`).reply(200);\n      unusedNocks.post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/send`).reply(200);\n\n      try {\n        await wallet.fanoutUnspents({ address, unspents }, ManageUnspentsOptions.BUILD_ONLY);\n      } catch (e) {\n        // the fanoutUnspents method will probably throw an exception for not having all of the correct nocks\n        // we only care about /fanoutUnspents and whether unspents is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n      unusedNocks.pendingMocks().length.should.eql(2);\n      nock.cleanAll();\n    });\n  });\n\n  describe('manage unspents', function () {\n    let rootWalletKey;\n    let walletPassphrase;\n    let basecoin;\n    let wallet;\n    let keysObj;\n\n    before(async function () {\n      rootWalletKey = getDefaultWalletKeys();\n      walletPassphrase = 'fixthemoneyfixtheworld';\n      keysObj = toKeychainObjects(rootWalletKey, walletPassphrase);\n      basecoin = bitgo.coin('tbtc');\n      const walletData = {\n        id: '5b34252f1bf349930e34020a',\n        coin: 'tbtc',\n        keys: keysObj.map((k) => k.id),\n      };\n      wallet = new Wallet(bitgo, basecoin, walletData);\n    });\n\n    it('should pass for bulk consolidating unspents', async function () {\n      const psbts = (['p2wsh', 'p2shP2wsh'] as const).map((scriptType) =>\n        utxoLib.testutil.constructPsbt(\n          [{ scriptType, value: BigInt(1000) }],\n          [{ scriptType, value: BigInt(900) }],\n          basecoin.network,\n          rootWalletKey,\n          'unsigned'\n        )\n      );\n      const txHexes = psbts.map((psbt) => ({ txHex: psbt.toHex() }));\n\n      const nocks: nock.Scope[] = [];\n      nocks.push(\n        nock(bgUrl).post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/consolidateUnspents`).reply(200, txHexes)\n      );\n\n      nocks.push(\n        ...keysObj.map((k, i) => nock(bgUrl).get(`/api/v2/${wallet.coin()}/key/${wallet.keyIds()[i]}`).reply(200, k))\n      );\n\n      nocks.push(\n        ...psbts.map((psbt) =>\n          nock(bgUrl)\n            .post(\n              `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/send`,\n              _.matches({ txHex: psbt.signAllInputsHD(rootWalletKey.user).toHex() })\n            )\n            .reply(200)\n        )\n      );\n\n      await wallet.consolidateUnspents({ bulk: true, walletPassphrase });\n\n      nocks.forEach((n) => {\n        console.log(n);\n        n.isDone().should.be.true();\n      });\n    });\n\n    it('should pass for single consolidating unspents', async function () {\n      const psbt = utxoLib.testutil.constructPsbt(\n        [{ scriptType: 'p2wsh', value: BigInt(1000) }],\n        [{ scriptType: 'p2shP2wsh', value: BigInt(900) }],\n        basecoin.network,\n        rootWalletKey,\n        'unsigned'\n      );\n\n      const nocks: nock.Scope[] = [];\n      nocks.push(\n        nock(bgUrl)\n          .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/consolidateUnspents`)\n          .reply(200, { txHex: psbt.toHex() })\n      );\n\n      nocks.push(\n        ...keysObj.map((k, i) => nock(bgUrl).get(`/api/v2/${wallet.coin()}/key/${wallet.keyIds()[i]}`).reply(200, k))\n      );\n\n      nocks.push(\n        nock(bgUrl)\n          .post(\n            `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/send`,\n            _.matches({ txHex: psbt.signAllInputsHD(rootWalletKey.user).toHex() })\n          )\n          .reply(200)\n      );\n\n      await wallet.consolidateUnspents({ walletPassphrase });\n\n      nocks.forEach((n) => {\n        n.isDone().should.be.true();\n      });\n    });\n  });\n  describe('max recipient', function () {\n    const address = '5b34252f1bf349930e34020a';\n    const recipients = [\n      {\n        address,\n        amount: 'max',\n      },\n    ];\n    let basecoin;\n    let wallet;\n\n    before(async function () {\n      basecoin = bitgo.coin('tbtc');\n      const walletData = {\n        id: '5b34252f1bf349930e34020a',\n        coin: 'tbtc',\n        keys: ['5b3424f91bf349930e340175'],\n      };\n      wallet = new Wallet(bitgo, basecoin, walletData);\n    });\n\n    it('should pass maxFeeRate parameter when building transactions', async function () {\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`;\n      const response = nock(bgUrl)\n        .post(\n          path,\n          _.matches({\n            recipients,\n          })\n        ) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n\n      try {\n        await wallet.prebuildTransaction({ recipients });\n      } catch (e) {\n        // the prebuildTransaction method will probably throw an exception for not having all of the correct nocks\n        // we only care about /tx/build and whether maxFeeRate is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n    });\n  });\n\n  describe('maxFeeRate verification', function () {\n    const address = '5b34252f1bf349930e34020a';\n    const recipients = [\n      {\n        address,\n        amount: 0,\n      },\n    ];\n    const maxFeeRate = 10000;\n    let basecoin;\n    let wallet;\n\n    before(async function () {\n      basecoin = bitgo.coin('tbtc');\n      const walletData = {\n        id: '5b34252f1bf349930e34020a',\n        coin: 'tbtc',\n        keys: ['5b3424f91bf349930e340175'],\n      };\n      wallet = new Wallet(bitgo, basecoin, walletData);\n    });\n\n    it('should pass maxFeeRate parameter when building transactions', async function () {\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`;\n      const response = nock(bgUrl)\n        .post(path, _.matches({ recipients, maxFeeRate })) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n\n      try {\n        await wallet.prebuildTransaction({ recipients, maxFeeRate });\n      } catch (e) {\n        // the prebuildTransaction method will probably throw an exception for not having all of the correct nocks\n        // we only care about /tx/build and whether maxFeeRate is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n    });\n\n    it('should pass maxFeeRate parameter when consolidating unspents', async function () {\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/consolidateUnspents`;\n      const response = nock(bgUrl)\n        .post(path, _.matches({ maxFeeRate })) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n\n      nock(bgUrl).get(`/api/v2/${wallet.coin()}/key/${wallet.keyIds()[0]}`).reply(200);\n\n      try {\n        await wallet.consolidateUnspents({ recipients, maxFeeRate });\n      } catch (e) {\n        // the consolidateUnspents method will probably throw an exception for not having all of the correct nocks\n        // we only care about /consolidateUnspents and whether maxFeeRate is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n    });\n\n    it('should only build tx (not sign/send) while consolidating unspents', async function () {\n      const toBeUsedNock = nock(bgUrl);\n      toBeUsedNock.post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/consolidateUnspents`).reply(200);\n\n      const unusedNocks = nock(bgUrl);\n      unusedNocks.get(`/api/v2/${wallet.coin()}/key/${wallet.keyIds()[0]}`).reply(200);\n      unusedNocks.post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/send`).reply(200);\n\n      await wallet.consolidateUnspents({ recipients }, ManageUnspentsOptions.BUILD_ONLY);\n\n      toBeUsedNock.isDone().should.be.true();\n      unusedNocks.pendingMocks().length.should.eql(2);\n      nock.cleanAll();\n    });\n\n    it('should pass maxFeeRate parameter when calling sweep wallets', async function () {\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/sweepWallet`;\n      const response = nock(bgUrl)\n        .post(path, _.matches({ address, maxFeeRate })) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n\n      try {\n        await wallet.sweep({ address, maxFeeRate });\n      } catch (e) {\n        // the sweep method will probably throw an exception for not having all of the correct nocks\n        // we only care about /sweepWallet and whether maxFeeRate is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n    });\n\n    it('should pass maxFeeRate parameter when calling fanout unspents', async function () {\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/fanoutUnspents`;\n      const response = nock(bgUrl)\n        .post(path, _.matches({ maxFeeRate })) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n\n      try {\n        await wallet.fanoutUnspents({ address, maxFeeRate });\n      } catch (e) {\n        // the fanoutUnspents method will probably throw an exception for not having all of the correct nocks\n        // we only care about /fanoutUnspents and whether maxFeeRate is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n    });\n  });\n\n  describe('allowPartialSweep verification', function () {\n    const address = '5b34252f1bf349930e34020a';\n    const allowPartialSweep = true;\n    let basecoin;\n    let wallet;\n\n    before(async function () {\n      basecoin = bitgo.coin('tbtc');\n      const walletData = {\n        id: '5b34252f1bf349930e34020a',\n        coin: 'tbtc',\n        keys: ['5b3424f91bf349930e340175'],\n      };\n      wallet = new Wallet(bitgo, basecoin, walletData);\n    });\n\n    it('should pass allowPartialSweep parameter when calling sweep wallets', async function () {\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/sweepWallet`;\n      const response = nock(bgUrl)\n        .post(path, _.matches({ address, allowPartialSweep })) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n\n      try {\n        await wallet.sweep({ address, allowPartialSweep });\n      } catch (e) {\n        // the sweep method will probably throw an exception for not having all of the correct nocks\n        // we only care about /sweepWallet and whether allowPartialSweep is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n    });\n  });\n\n  describe('sweep wallet', function () {\n    let basecoin;\n    let wallet;\n\n    before(async function () {\n      basecoin = bitgo.coin('ttrx');\n      const walletData = {\n        id: '5b34252f1bf349930e34020a',\n        coin: 'ttrx',\n        keys: ['5b3424f91bf349930e340175'],\n      };\n      wallet = new Wallet(bitgo, basecoin, walletData);\n    });\n\n    it('should use maximum spendable balance of wallet to sweep funds ', async function () {\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/maximumSpendable`;\n      const response = nock(bgUrl).get(path).reply(200, {\n        coin: 'ttrx',\n        maximumSpendable: 65000,\n      });\n      const body = {\n        coin: 'ttrx',\n        address: '2MwvR24yqym2CgHMp7zwvdeqBa4F8KTqunS',\n      };\n      try {\n        await wallet.sweep(body);\n      } catch (e) {\n        // the sweep method will probably throw an exception for not having all of the correct nocks\n        // we only care about maximum spendable balance being used to sweep funds\n      }\n\n      response.isDone().should.be.true();\n    });\n  });\n\n  describe('Transaction prebuilds', function () {\n    let ethWallet;\n\n    before(async function () {\n      const walletData = {\n        id: '598f606cd8fc24710d2ebadb1d9459bb',\n        coin: 'teth',\n        keys: [\n          '598f606cd8fc24710d2ebad89dce86c2',\n          '598f606cc8e43aef09fcb785221d9dd2',\n          '5935d59cf660764331bafcade1855fd7',\n        ],\n      };\n      ethWallet = new Wallet(bitgo, bitgo.coin('teth'), walletData);\n    });\n\n    it('should return reqId if it was passed in the params', async function () {\n      const params = { offlineVerification: true };\n      const scope = nock(bgUrl)\n        .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, tbtcHotWalletDefaultParams)\n        .query(params)\n        .reply(200, {});\n      const blockHeight = 100;\n      sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);\n      sinon.stub(basecoin, 'postProcessPrebuild').resolves({});\n      const txRequest = await wallet.prebuildTransaction({ ...params, reqId: reqId });\n      txRequest.reqId?.should.containEql(reqId);\n      scope.done();\n    });\n\n    it('should pass offlineVerification=true query param if passed truthy value', async function () {\n      const params = { offlineVerification: true };\n      const scope = nock(bgUrl)\n        .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, tbtcHotWalletDefaultParams)\n        .query(params)\n        .reply(200, {});\n      const blockHeight = 100;\n      const blockHeightStub = sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);\n      const postProcessStub = sinon.stub(basecoin, 'postProcessPrebuild').resolves({});\n      await wallet.prebuildTransaction(params);\n      blockHeightStub.should.have.been.calledOnce();\n      postProcessStub.should.have.been.calledOnceWith({\n        blockHeight: 100,\n        wallet: wallet,\n        buildParams: tbtcHotWalletDefaultParams,\n      });\n      scope.done();\n      blockHeightStub.restore();\n      postProcessStub.restore();\n    });\n\n    it('should not pass the offlineVerification query param if passed a falsey value', async function () {\n      const params = { offlineVerification: false };\n      nock(bgUrl)\n        .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, tbtcHotWalletDefaultParams)\n        .query({})\n        .reply(200, {});\n      const blockHeight = 100;\n      const blockHeightStub = sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);\n      const postProcessStub = sinon.stub(basecoin, 'postProcessPrebuild').resolves({});\n      await wallet.prebuildTransaction(params);\n      blockHeightStub.should.have.been.calledOnce();\n      postProcessStub.should.have.been.calledOnceWith({\n        blockHeight: 100,\n        wallet: wallet,\n        buildParams: tbtcHotWalletDefaultParams,\n      });\n      blockHeightStub.restore();\n      postProcessStub.restore();\n    });\n\n    it('should pass script outputs with the proper structure to wallet platform', async function () {\n      const script = '6a11223344556677889900';\n      nock(bgUrl)\n        .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, {\n          ...tbtcHotWalletDefaultParams,\n          recipients: [{ script, amount: 1e6 }],\n        })\n        .query({})\n        .reply(200, {});\n\n      const blockHeight = 100;\n      const blockHeightStub = sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);\n      const postProcessStub = sinon.stub(basecoin, 'postProcessPrebuild').resolves({});\n      await wallet.prebuildTransaction({ recipients: [{ address: `scriptPubKey:${script}`, amount: 1e6 }] });\n      blockHeightStub.should.have.been.calledOnce();\n      postProcessStub.should.have.been.calledOnceWith({\n        blockHeight: 100,\n        wallet: wallet,\n        buildParams: { ...tbtcHotWalletDefaultParams, recipients: [{ script, amount: 1e6 }] },\n      });\n      blockHeightStub.restore();\n      postProcessStub.restore();\n    });\n\n    it('prebuild should call build and getLatestBlockHeight for utxo coins', async function () {\n      const params = {};\n      nock(bgUrl)\n        .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, tbtcHotWalletDefaultParams)\n        .query(params)\n        .reply(200, {});\n      const blockHeight = 100;\n      const blockHeightStub = sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);\n      const postProcessStub = sinon.stub(basecoin, 'postProcessPrebuild').resolves({});\n      await wallet.prebuildTransaction(params);\n      blockHeightStub.should.have.been.calledOnce();\n      postProcessStub.should.have.been.calledOnceWith({\n        blockHeight: 100,\n        wallet: wallet,\n        buildParams: tbtcHotWalletDefaultParams,\n      });\n      blockHeightStub.restore();\n      postProcessStub.restore();\n    });\n\n    it('prebuild should not have changeAddressType array in post body when changeAddressType is defined', async function () {\n      const expectedBuildPostBodyParams = {\n        changeAddressType: 'p2trMusig2',\n        txFormat: 'psbt',\n      };\n\n      nock(bgUrl)\n        .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`, expectedBuildPostBodyParams)\n        .query({})\n        .reply(200, {});\n      const blockHeight = 100;\n      const blockHeightStub = sinon.stub(basecoin, 'getLatestBlockHeight').resolves(blockHeight);\n      const postProcessStub = sinon.stub(basecoin, 'postProcessPrebuild').resolves({});\n      await wallet.prebuildTransaction({ changeAddressType: 'p2trMusig2' });\n      blockHeightStub.should.have.been.calledOnce();\n      postProcessStub.should.have.been.calledOnceWith({\n        blockHeight: 100,\n        wallet: wallet,\n        buildParams: expectedBuildPostBodyParams,\n      });\n      blockHeightStub.restore();\n      postProcessStub.restore();\n    });\n\n    it('prebuild should call build but not getLatestBlockHeight for account coins', async function () {\n      ['txrp', 'txlm', 'teth'].forEach(async function (coin) {\n        const accountcoin = bitgo.coin(coin);\n        const walletData = {\n          id: '5b34252f1bf349930e34021a',\n          coin,\n          keys: ['5b3424f91bf349930e340175'],\n        };\n        const accountWallet = new Wallet(bitgo, accountcoin, walletData);\n        const params = {};\n        nock(bgUrl)\n          .post(`/api/v2/${accountWallet.coin()}/wallet/${accountWallet.id()}/tx/build`)\n          .query(params)\n          .reply(200, {});\n        const postProcessStub = sinon.stub(accountcoin, 'postProcessPrebuild').resolves({});\n        await accountWallet.prebuildTransaction(params);\n        postProcessStub.should.have.been.calledOnceWith({\n          wallet: accountWallet,\n          buildParams: {},\n        });\n        postProcessStub.restore();\n      });\n    });\n\n    it('should have isBatch = true in the txPrebuild if txParams has more than one recipient', async function () {\n      const txParams = {\n        recipients: [\n          { amount: '1000000000000000', address: address1 },\n          { amount: '1000000000000000', address: address2 },\n        ],\n        walletContractAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n        walletPassphrase: 'moon',\n      };\n\n      const totalAmount = '2000000000000000';\n\n      nock(bgUrl)\n        .post(\n          `/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/tx/build`,\n          _.matches({ recipients: txParams.recipients })\n        )\n        .reply(200, {\n          recipients: [\n            {\n              address: '0xc0aaf2649e7b0f3950164681eca2b1a8f654a478',\n              amount: '2000000000000000',\n              data: '0xc00c4e9e000000000000000000000000000000000000000000000000000000000000004000000000000000000000000000000000000000000000000000000000000000a00000000000000000000000000000000000000000000000000000000000000002000000000000000000000000174cfd823af8ce27ed0afee3fcf3c3ba259116be0000000000000000000000007e85bdc27c050e3905ebf4b8e634d9ad6edd0de6000000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000038d7ea4c6800000000000000000000000000000000000000000000000000000038d7ea4c68000',\n            },\n          ],\n          nextContractSequenceId: 10896,\n          gasPrice: 20000000000,\n          gasLimit: 500000,\n          isBatch: true,\n          coin: 'teth',\n        });\n\n      const txPrebuild = await ethWallet.prebuildTransaction(txParams);\n      txPrebuild.isBatch.should.equal(true);\n      txPrebuild.recipients[0].address.should.equal(\n        (bitgo.coin('teth') as any).staticsCoin.network.batcherContractAddress\n      );\n      txPrebuild.recipients[0].amount.should.equal(totalAmount);\n    });\n\n    it('should have isBatch = false and hopTransaction field should not be there in the txPrebuild  for normal eth tx', async function () {\n      const txParams = {\n        recipients: [{ amount: '1000000000000000', address: address1 }],\n        walletContractAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n        walletPassphrase: 'moon',\n      };\n\n      nock(bgUrl)\n        .post(\n          `/api/v2/${ethWallet.coin()}/wallet/${ethWallet.id()}/tx/build`,\n          _.matches({ recipients: txParams.recipients })\n        )\n        .reply(200, {\n          recipients: [\n            {\n              amount: '1000000000000000',\n              address: '0x174cfd823af8ce27ed0afee3fcf3c3ba259116be',\n            },\n          ],\n          nextContractSequenceId: 10897,\n          gasPrice: 20000000000,\n          gasLimit: 500000,\n          isBatch: false,\n          coin: 'teth',\n        });\n\n      const txPrebuild = await ethWallet.prebuildTransaction(txParams);\n      txPrebuild.isBatch.should.equal(false);\n      txPrebuild.should.not.have.property('hopTransaction');\n      txPrebuild.recipients[0].address.should.equal(address1);\n      txPrebuild.recipients[0].amount.should.equal('1000000000000000');\n    });\n\n    it('should pass unspent reservation parameter through when building transactions', async function () {\n      const reservation = {\n        expireTime: '2029-08-12',\n      };\n      const recipients = [\n        {\n          address: 'aaa',\n          amount: '1000',\n        },\n      ];\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`;\n      const response = nock(bgUrl)\n        .post(path, _.matches({ recipients, reservation })) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n      try {\n        await wallet.prebuildTransaction({ recipients, reservation });\n      } catch (e) {\n        // the prebuildTransaction method will probably throw an exception for not having all of the correct nocks\n        // we only care about /tx/build and whether reservation is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n    });\n\n    it('should pass gas limit parameter through when building transaction for sui', async function () {\n      const params = { gasLimit: 100 };\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/build`;\n      const response = nock(bgUrl)\n        .post(path, _.matches(params)) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200);\n      try {\n        await wallet.prebuildTransaction(params);\n      } catch (e) {\n        // the prebuildTransaction method will probably throw an exception for not having all of the correct nocks\n        // we only care about /tx/build and whether reservation is an allowed parameter\n      }\n\n      response.isDone().should.be.true();\n    });\n  });\n\n  describe('Maximum Spendable', function maximumSpendable() {\n    let bgUrl;\n\n    before(async function () {\n      nock.pendingMocks().should.be.empty();\n      bgUrl = common.Environments[bitgo.getEnv()].uri;\n    });\n\n    it('arguments', async function () {\n      const optionalParams = {\n        limit: 25,\n        minValue: '0',\n        maxValue: '9999999999999',\n        minHeight: 0,\n        minConfirms: 2,\n        enforceMinConfirmsForChange: false,\n        feeRate: 10000,\n        maxFeeRate: 100000,\n        recipientAddress: '2NCUFDLiUz9CVnmdVqQe9acVonoM89e76df',\n      };\n\n      // The actual api request will only send strings, but the SDK function expects numbers for some values\n      const apiParams = _.mapValues(optionalParams, (param) => String(param));\n\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/maximumSpendable`;\n      const response = nock(bgUrl)\n        .get(path)\n        .query(_.matches(apiParams)) // use _.matches to do a partial match on request body object instead of strict matching\n        .reply(200, {\n          coin: 'tbch',\n          maximumSpendable: 65000,\n        });\n\n      try {\n        await wallet.maximumSpendable(optionalParams);\n      } catch (e) {\n        // test is successful if nock is consumed\n      }\n\n      response.isDone().should.be.true();\n    });\n  });\n\n  describe('Wallet Sharing', function () {\n    it('should share to cold wallet without passing skipKeychain', async function () {\n      const userId = '123';\n      const email = 'shareto@sdktest.com';\n      const permissions = 'view,spend';\n\n      const getSharingKeyNock = nock(bgUrl).post('/api/v1/user/sharingkey', { email }).reply(200, { userId });\n\n      const getKeyNock = nock(bgUrl)\n        .get(`/api/v2/tbtc/key/${coldWallet.keyIds()[0]}`)\n        .reply(200, {})\n        .get(`/api/v2/tbtc/key/${coldWallet.keyIds()[1]}`)\n        .reply(200, {})\n        .get(`/api/v2/tbtc/key/${coldWallet.keyIds()[2]}`)\n        .reply(200, {});\n\n      const createShareNock = nock(bgUrl)\n        .post(`/api/v2/tbtc/wallet/${coldWallet.id()}/share`, {\n          user: userId,\n          permissions,\n          skipKeychain: true,\n        })\n        .reply(200, {});\n\n      await coldWallet.shareWallet({ email, permissions });\n\n      getSharingKeyNock.isDone().should.be.True();\n      getKeyNock.isDone().should.be.True();\n      createShareNock.isDone().should.be.True();\n    });\n\n    describe('Hot Wallet Sharing', function () {\n      const userId = '123';\n      const email = 'shareto@sdktest.com';\n      const permissions = 'view,spend';\n      const toKeychain = utxoLib.bip32.fromSeed(Buffer.from('deadbeef02deadbeef02deadbeef02deadbeef02', 'hex'));\n      const path = 'm/999999/1/1';\n      const pubkey = toKeychain.derivePath(path).publicKey.toString('hex');\n      const walletPassphrase = 'bitgo1234';\n      const pub = 'Zo1ggzTUKMY5bYnDvT5mtVeZxzf2FaLTbKkmvGUhUQk';\n\n      const lightningCoin: any = bitgo.coin('tlnbtc');\n      const lightningWalletData = {\n        id: '5b34252f1bf349930e34020a00000001',\n        coin: 'tlnbtc',\n        keys: ['5b3424f91bf349930e34017500000001'],\n        coinSpecific: { keys: ['5b3424f91bf349930e34017600000000', '5b3424f91bf349930e34017700000000'] },\n        type: 'hot',\n      };\n      const lightningWallet = new Wallet(bitgo, lightningCoin, lightningWalletData);\n\n      for (const hotWallet of [wallet, lightningWallet] as const) {\n        it(`should use keychain pub to share ${hotWallet.coin()} hot wallet`, async function () {\n          const getSharingKeyNock = nock(bgUrl)\n            .post('/api/v1/user/sharingkey', { email })\n            .reply(200, { userId, pubkey, path });\n\n          const getKeyNocks: nock.Scope[] = [];\n          if (hotWallet.baseCoin.getFamily() === 'lnbtc') {\n            for (let i = 0; i < 2; i++) {\n              const keyId = lightningWalletData.coinSpecific.keys[i];\n              const getKeyNock = nock(bgUrl)\n                .get(`/api/v2/tlnbtc/key/${keyId}`)\n                .reply(200, {\n                  id: keyId,\n                  pub: i === 0 ? pub : 'Zo1ggzTUKMY5bYnDvT5mtVeZxzf2FaLTbKkmvGUhUQm',\n                  source: 'user',\n                  encryptedPrv: bitgo.encrypt({ input: 'xprv' + i, password: walletPassphrase }),\n                  coinSpecific:\n                    i === 0\n                      ? { [hotWallet.baseCoin.getChain()]: { purpose: 'userAuth' } }\n                      : { [hotWallet.baseCoin.getChain()]: { purpose: 'nodeAuth' } },\n                });\n              getKeyNocks.push(getKeyNock);\n            }\n          } else {\n            const getKeyNock = nock(bgUrl)\n              .get(`/api/v2/tbtc/key/${wallet.keyIds()[0]}`)\n              .reply(200, {\n                id: wallet.keyIds()[0],\n                pub,\n                source: 'user',\n                encryptedPrv: bitgo.encrypt({ input: 'xprv1', password: walletPassphrase }),\n                coinSpecific: {},\n              });\n            getKeyNocks.push(getKeyNock);\n          }\n\n          const stub = sinon.stub(hotWallet, 'createShare').callsFake(async (options) => {\n            // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n            options!.keychain!.pub!.should.not.be.undefined();\n            // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n            options!.keychain!.pub!.should.equal(pub);\n            return undefined;\n          });\n          await hotWallet.shareWallet({ email, permissions, walletPassphrase });\n\n          stub.calledOnce.should.be.true();\n          getSharingKeyNock.isDone().should.be.True();\n          getKeyNocks.every((v) => v.isDone().should.be.True());\n        });\n      }\n    });\n\n    it('should provide skipKeychain to wallet share api for hot wallet', async function () {\n      const userId = '123';\n      const email = 'shareto@sdktest.com';\n      const permissions = 'view,spend';\n      const toKeychain = utxoLib.bip32.fromSeed(Buffer.from('deadbeef02deadbeef02deadbeef02deadbeef02', 'hex'));\n      const path = 'm/999999/1/1';\n      const pubkey = toKeychain.derivePath(path).publicKey.toString('hex');\n\n      const getSharingKeyNock = nock(bgUrl)\n        .post('/api/v1/user/sharingkey', { email })\n        .reply(200, { userId, pubkey, path });\n      const createShareNock = nock(bgUrl)\n        .post(`/api/v2/tbtc/wallet/${wallet.id()}/share`, {\n          user: userId,\n          permissions,\n          skipKeychain: true,\n        })\n        .reply(200, {});\n\n      await wallet.shareWallet({ email, permissions, skipKeychain: true });\n\n      createShareNock.isDone().should.be.True();\n      getSharingKeyNock.isDone().should.be.True();\n    });\n\n    it('should decrypt webauthn encryptedPrv for wallet share (spend)', async function () {\n      const userId = '123';\n      const email = 'shareto@sdktest.com';\n      const permissions = 'view,spend';\n      const toKeychain = utxoLib.bip32.fromSeed(Buffer.from('deadbeef02deadbeef02deadbeef02deadbeef02', 'hex'));\n      const path = 'm/999999/1/1';\n      const pubkey = toKeychain.derivePath(path).publicKey.toString('hex');\n      const privateKey = 'xprv1';\n      const walletPassphrase1 = 'bitgo1234';\n      const walletPassphrase2 = 'bitgo5678';\n\n      const getSharingKeyNock = nock(bgUrl)\n        .post('/api/v1/user/sharingkey', { email })\n        .reply(200, { userId, pubkey, path });\n\n      const pub = 'Zo1ggzTUKMY5bYnDvT5mtVeZxzf2FaLTbKkmvGUhUQk';\n      const getKeyNock = nock(bgUrl)\n        .get(`/api/v2/tbtc/key/${wallet.keyIds()[0]}`)\n        .reply(200, {\n          id: wallet.keyIds()[0],\n          pub,\n          source: 'user',\n          encryptedPrv: bitgo.encrypt({ input: privateKey, password: walletPassphrase1 }),\n          webauthnDevices: [\n            {\n              otpDeviceId: '123',\n              authenticatorInfo: {\n                credID: 'credID',\n                fmt: 'packed',\n                publicKey: 'some value',\n              },\n              prfSalt: '456',\n              encryptedPrv: bitgo.encrypt({ input: privateKey, password: walletPassphrase2 }),\n            },\n          ],\n          coinSpecific: {},\n        });\n\n      const stub = sinon.stub(wallet, 'createShare').callsFake(async (options) => {\n        options!.keychain!.encryptedPrv!.should.not.be.undefined();\n        return undefined;\n      });\n      await wallet.shareWallet({ email, permissions, walletPassphrase: walletPassphrase2 });\n      stub.calledOnce.should.be.true();\n      getSharingKeyNock.isDone().should.be.True();\n      getKeyNock.isDone().should.be.True();\n    });\n  });\n\n  describe('Wallet Freezing', function () {\n    it('should freeze wallet for specified duration in seconds', async function () {\n      const params = { duration: 60 };\n      const scope = nock(bgUrl).post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/freeze`, params).reply(200, {});\n      await wallet.freeze(params);\n      scope.isDone().should.be.True();\n    });\n  });\n\n  describe('TSS Wallets', function () {\n    const sandbox = sinon.createSandbox();\n    const tsol = bitgo.coin('tsol');\n    const walletData = {\n      id: '5b34252f1bf349930e34020a00000000',\n      coin: 'tsol',\n      keys: [\n        '598f606cd8fc24710d2ebad89dce86c2',\n        '598f606cc8e43aef09fcb785221d9dd2',\n        '5935d59cf660764331bafcade1855fd7',\n      ],\n      coinSpecific: {},\n      multisigType: 'tss',\n    };\n\n    const ethWalletData = {\n      id: '598f606cd8fc24710d2ebadb1d9459bb',\n      coin: 'teth',\n      keys: [\n        '598f606cd8fc24710d2ebad89dce86c2',\n        '598f606cc8e43aef09fcb785221d9dd2',\n        '5935d59cf660764331bafcade1855fd7',\n      ],\n      multisigType: 'tss',\n      coinSpecific: { addressVersion: 1 },\n      type: 'hot',\n    };\n\n    const polygonWalletData = {\n      id: '632826520ee1e5000729017354acaeab',\n      coin: 'tpolygon',\n      keys: [\n        '598f606cd8fc24710d2ebad89dce86c2',\n        '598f606cc8e43aef09fcb785221d9dd2',\n        '5935d59cf660764331bafcade1855fd7',\n      ],\n      multisigType: 'tss',\n    };\n\n    const tssSolWallet = new Wallet(bitgo, tsol, walletData);\n\n    let tssEthWallet = new Wallet(bitgo, bitgo.coin('teth'), ethWalletData);\n    const tssPolygonWallet = new Wallet(bitgo, bitgo.coin('tpolygon'), polygonWalletData);\n    const custodialTssSolWallet = new Wallet(bitgo, tsol, {\n      ...walletData,\n      type: 'custodial',\n    });\n\n    const txRequest: TxRequest = {\n      txRequestId: 'id',\n      transactions: [],\n      intent: {\n        intentType: 'payment',\n      },\n      date: new Date().toISOString(),\n      latest: true,\n      state: 'pendingUserSignature',\n      userId: 'userId',\n      walletType: 'hot',\n      policiesChecked: false,\n      version: 1,\n      walletId: 'walletId',\n      unsignedTxs: [\n        {\n          serializedTxHex: 'ababcdcd',\n          signableHex: 'deadbeef',\n          feeInfo: {\n            fee: 5000,\n            feeString: '5000',\n          },\n          derivationPath: 'm/0',\n        },\n      ],\n    };\n\n    const txRequestFull: TxRequest = {\n      txRequestId: 'id',\n      intent: {\n        intentType: 'payment',\n      },\n      date: new Date().toISOString(),\n      latest: true,\n      state: 'pendingUserSignature',\n      userId: 'userId',\n      walletId: 'walletId',\n      signatureShares: [],\n      version: 1,\n      policiesChecked: false,\n      walletType: 'hot',\n      transactions: [\n        {\n          state: 'pendingSignature',\n          unsignedTx: {\n            serializedTxHex: 'ababcdcd',\n            signableHex: 'deadbeef',\n            feeInfo: {\n              fee: 5000,\n              feeString: '5000',\n            },\n            derivationPath: 'm/0',\n          },\n          signatureShares: [],\n          commitmentShares: [],\n        },\n      ],\n      unsignedTxs: [],\n      apiVersion: 'full',\n    };\n\n    afterEach(function () {\n      sandbox.verifyAndRestore();\n    });\n\n    describe('preBuildAndSignTransaction', async function () {\n      const params = {\n        walletPassphrase: 'passphrase12345',\n        prebuildTx: { walletId: tssEthWallet.id(), txRequestId: 'randomId' },\n        type: 'transfer',\n      };\n\n      ['eddsa', 'ecdsa'].forEach((keyCurve: string) => {\n        describe(keyCurve, () => {\n          const wallet = keyCurve === 'eddsa' ? tssSolWallet : tssEthWallet;\n\n          beforeEach(function () {\n            sandbox\n              .stub(Keychains.prototype, 'getKeysForSigning')\n              .resolves([{ commonKeychain: 'test', id: '', pub: '', type: 'independent' }]);\n            if (keyCurve === 'eddsa') {\n              sandbox.stub(Tsol.prototype, 'verifyTransaction').resolves(true);\n            } else {\n              sandbox.stub(Teth.prototype, 'verifyTransaction').resolves(true);\n            }\n          });\n\n          afterEach(function () {\n            sandbox.verifyAndRestore();\n          });\n\n          it('it should succeed but not sign if the txRequest is pending approval', async function () {\n            const getTxRequestStub = sandbox.stub(BaseTssUtils.default.prototype, 'getTxRequest').resolves({\n              ...txRequestFull,\n              state: 'pendingApproval',\n            });\n\n            const signTransactionSpy = sandbox.spy(Wallet.prototype, 'signTransaction');\n\n            const result = (await wallet.prebuildAndSignTransaction(params)) as TxRequest;\n            result.should.have.property('state');\n            result.state.should.equal('pendingApproval');\n            getTxRequestStub.calledOnce.should.be.true();\n            signTransactionSpy.notCalled.should.be.true();\n          });\n\n          it('it should succeed and sign if the txRequest is not pending approval', async function () {\n            const getTxRequestStub = sandbox.stub(BaseTssUtils.default.prototype, 'getTxRequest');\n            getTxRequestStub.resolves(txRequestFull);\n\n            const signTransactionStub = sandbox.stub(Wallet.prototype, 'signTransaction');\n            signTransactionStub.resolves({ ...txRequestFull, state: 'signed' });\n\n            const result = (await wallet.prebuildAndSignTransaction(params)) as TxRequest;\n            result.should.have.property('state');\n            result.state.should.equal('signed');\n            getTxRequestStub.calledOnce.should.be.true();\n            signTransactionStub.calledOnce.should.be.true();\n          });\n        });\n      });\n    });\n\n    describe('Transaction prebuilds', function () {\n      it('should build a single recipient transfer transaction', async function () {\n        const recipients = [\n          {\n            address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',\n            amount: '1000',\n          },\n        ];\n\n        const prebuildTxWithIntent = sandbox.stub(TssUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequest);\n        // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke prebuildTransaction\n        prebuildTxWithIntent.calledOnceWithExactly({\n          reqId,\n          recipients,\n          intentType: 'payment',\n        });\n\n        const txPrebuild = await tssSolWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'transfer',\n        });\n\n        txPrebuild.should.deepEqual({\n          walletId: tssSolWallet.id(),\n          wallet: tssSolWallet,\n          txRequestId: 'id',\n          txHex: 'ababcdcd',\n          buildParams: {\n            recipients,\n            type: 'transfer',\n          },\n          feeInfo: {\n            fee: 5000,\n            feeString: '5000',\n          },\n        });\n      });\n\n      it('should build a single recipient transfer with pending approval id if transaction is having one', async function () {\n        const recipients = [\n          {\n            address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',\n            amount: '1000',\n          },\n        ];\n\n        const prebuildTxWithIntent = sandbox.stub(TssUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves({ ...txRequest, state: 'pendingApproval', pendingApprovalId: 'some-id' });\n        prebuildTxWithIntent.calledOnceWithExactly({\n          reqId,\n          recipients,\n          intentType: 'payment',\n        });\n\n        const txPrebuild = await custodialTssSolWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'transfer',\n        });\n\n        txPrebuild.should.deepEqual({\n          walletId: custodialTssSolWallet.id(),\n          wallet: custodialTssSolWallet,\n          txRequestId: 'id',\n          txHex: 'ababcdcd',\n          pendingApprovalId: 'some-id',\n          buildParams: {\n            recipients,\n            type: 'transfer',\n          },\n          feeInfo: {\n            fee: 5000,\n            feeString: '5000',\n          },\n        });\n      });\n\n      it('should build a multiple recipient transfer transaction with memo', async function () {\n        const recipients = [\n          {\n            address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',\n            amount: '1000',\n          },\n          {\n            address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',\n            amount: '2000',\n          },\n        ];\n\n        const prebuildTxWithIntent = sandbox.stub(TssUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequest);\n        // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke prebuildTransaction\n        prebuildTxWithIntent.calledOnceWithExactly({\n          reqId,\n          recipients,\n          intentType: 'payment',\n          memo: {\n            type: 'type',\n            value: 'test memo',\n          },\n        });\n\n        const txPrebuild = await tssSolWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'transfer',\n          memo: {\n            type: 'type',\n            value: 'test memo',\n          },\n        });\n\n        txPrebuild.should.deepEqual({\n          walletId: tssSolWallet.id(),\n          wallet: tssSolWallet,\n          txRequestId: 'id',\n          txHex: 'ababcdcd',\n          buildParams: {\n            recipients,\n            memo: {\n              type: 'type',\n              value: 'test memo',\n            },\n            type: 'transfer',\n          },\n          feeInfo: {\n            fee: 5000,\n            feeString: '5000',\n          },\n        });\n      });\n\n      it('should build an enable token transaction', async function () {\n        const recipients = [];\n        const tokenName = 'tcoin:tokenName';\n        const prebuildTxWithIntent = sandbox.stub(TssUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequest);\n        // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke prebuildTransaction\n        prebuildTxWithIntent.calledOnceWithExactly({\n          reqId,\n          recipients,\n          intentType: 'createAccount',\n          memo: {\n            type: 'type',\n            value: 'test memo',\n          },\n          tokenName,\n        });\n\n        const txPrebuild = await tssSolWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'enabletoken',\n          memo: {\n            type: 'type',\n            value: 'test memo',\n          },\n          tokenName,\n        });\n\n        txPrebuild.should.deepEqual({\n          walletId: tssSolWallet.id(),\n          wallet: tssSolWallet,\n          txRequestId: 'id',\n          txHex: 'ababcdcd',\n          buildParams: {\n            recipients,\n            memo: {\n              type: 'type',\n              value: 'test memo',\n            },\n            type: 'enabletoken',\n            tokenName,\n          },\n          feeInfo: {\n            fee: 5000,\n            feeString: '5000',\n          },\n        });\n      });\n\n      it('should build an enable token transaction for cold wallets', async function () {\n        const recipients = [];\n        const tokenName = 'tcoin:tokenName';\n        const prebuildTxWithIntent = sandbox.stub(TssUtils.prototype, 'prebuildTxWithIntent');\n        txRequest.walletType = 'cold';\n        prebuildTxWithIntent.resolves(txRequest);\n        prebuildTxWithIntent.calledOnceWithExactly({\n          reqId,\n          recipients,\n          intentType: 'createAccount',\n          memo: {\n            type: 'type',\n            value: 'test memo',\n          },\n          tokenName,\n        });\n\n        const txPrebuild = await tssSolWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'enabletoken',\n          memo: {\n            type: 'type',\n            value: 'test memo',\n          },\n          tokenName,\n        });\n\n        txPrebuild.should.deepEqual({\n          walletId: tssSolWallet.id(),\n          wallet: tssSolWallet,\n          txRequestId: 'id',\n          txHex: 'ababcdcd',\n          buildParams: {\n            recipients,\n            memo: {\n              type: 'type',\n              value: 'test memo',\n            },\n            type: 'enabletoken',\n            tokenName,\n          },\n          feeInfo: {\n            fee: 5000,\n            feeString: '5000',\n          },\n        });\n      });\n\n      it('should fail for non-transfer transaction types', async function () {\n        await tssSolWallet\n          .prebuildTransaction({\n            reqId,\n            recipients: [\n              {\n                address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',\n                amount: '1000',\n              },\n            ],\n            type: 'stake',\n          })\n          .should.be.rejectedWith('transaction type not supported: stake');\n      });\n\n      it('should fail for full api version compatibility', async function () {\n        await custodialTssSolWallet\n          .prebuildTransaction({\n            reqId,\n            apiVersion: 'lite',\n            recipients: [\n              {\n                address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',\n                amount: '1000',\n              },\n            ],\n            type: 'transfer',\n          })\n          .should.be.rejectedWith('For non self-custodial (hot) tss wallets, parameter `apiVersion` must be `full`.');\n      });\n\n      it('should build a single recipient transfer transaction for full', async function () {\n        const recipients = [\n          {\n            address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',\n            amount: '1000',\n          },\n        ];\n\n        const prebuildTxWithIntent = sandbox.stub(TssUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequestFull);\n        // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke prebuildTransaction\n        prebuildTxWithIntent.calledOnceWithExactly(\n          {\n            reqId,\n            recipients,\n            intentType: 'payment',\n          },\n          'full'\n        );\n\n        const txPrebuild = await custodialTssSolWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'transfer',\n        });\n\n        txPrebuild.should.deepEqual({\n          walletId: tssSolWallet.id(),\n          wallet: custodialTssSolWallet,\n          txRequestId: 'id',\n          txHex: 'ababcdcd',\n          buildParams: {\n            recipients,\n            type: 'transfer',\n          },\n          feeInfo: {\n            fee: 5000,\n            feeString: '5000',\n          },\n        });\n      });\n\n      it('should call prebuildTxWithIntent with the correct params for eth transfers', async function () {\n        const recipients = [\n          {\n            address: '0xAB100912e133AA06cEB921459aaDdBd62381F5A3',\n            amount: '1000',\n          },\n        ];\n\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n\n        const prebuildTxWithIntent = sandbox.stub(ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequestFull);\n\n        await tssEthWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'transfer',\n          feeOptions,\n        });\n\n        sinon.assert.calledOnce(prebuildTxWithIntent);\n        const args = prebuildTxWithIntent.args[0];\n        args[0]!.recipients!.should.deepEqual(recipients);\n        args[0]!.feeOptions!.should.deepEqual(feeOptions);\n        args[0]!.intentType.should.equal('payment');\n        args[1]!.should.equal('full');\n      });\n\n      it('should call prebuildTxWithIntent with the correct params for eth transfertokens', async function () {\n        const recipients = [\n          {\n            address: '0xAB100912e133AA06cEB921459aaDdBd62381F5A3',\n            amount: '1000',\n            tokenName: 'gterc18dp',\n          },\n        ];\n\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n\n        const prebuildTxWithIntent = sandbox.stub(ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequestFull);\n\n        await tssEthWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'transfertoken',\n          isTss: true,\n          feeOptions,\n        });\n\n        sinon.assert.calledOnce(prebuildTxWithIntent);\n        const args = prebuildTxWithIntent.args[0];\n        args[0]!.recipients!.should.deepEqual(recipients);\n        args[0]!.feeOptions!.should.deepEqual(feeOptions);\n        args[0]!.isTss!.should.equal(true);\n        args[0]!.intentType.should.equal('transferToken');\n        args[1]!.should.equal('full');\n      });\n\n      it('should call prebuildTxWithIntent with the correct params for eth accelerations', async function () {\n        const recipients = [\n          {\n            address: '0xAB100912e133AA06cEB921459aaDdBd62381F5A3',\n            amount: '1000',\n            tokenName: 'gterc18dp',\n          },\n        ];\n\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n\n        const lowFeeTxid = '0x6ea07f9420f4676be6478ab1660eb92444a7c663e0e24bece929f715e882e0cf';\n\n        const prebuildTxWithIntent = sandbox.stub(ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequestFull);\n\n        await tssEthWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'acceleration',\n          feeOptions,\n          lowFeeTxid,\n        });\n\n        sinon.assert.calledOnce(prebuildTxWithIntent);\n        const args = prebuildTxWithIntent.args[0];\n        args[0]!.should.not.have.property('recipients');\n        args[0]!.feeOptions!.should.deepEqual(feeOptions);\n        args[0]!.lowFeeTxid!.should.equal(lowFeeTxid);\n        args[0]!.intentType.should.equal('acceleration');\n        args[1]!.should.equal('full');\n      });\n\n      it('should call prebuildTxWithIntent with the correct params for eth accelerations for receive address', async function () {\n        const recipients = [\n          {\n            address: '0xAB100912e133AA06cEB921459aaDdBd62381F5A3',\n            amount: '1000',\n            tokenName: 'gterc18dp',\n          },\n        ];\n\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n\n        const lowFeeTxid = '0x6ea07f9420f4676be6478ab1660eb92444a7c663e0e24bece929f715e882e0cf';\n        const receiveAddress = '0x062176bc9345da3e8ee90361b0cf6ff883ba7206';\n\n        const prebuildTxWithIntent = sandbox.stub(ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequestFull);\n\n        await tssEthWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'acceleration',\n          feeOptions,\n          lowFeeTxid,\n          receiveAddress,\n        });\n\n        sinon.assert.calledOnce(prebuildTxWithIntent);\n        const args = prebuildTxWithIntent.args[0];\n        args[0]!.should.not.have.property('recipients');\n        args[0]!.feeOptions!.should.deepEqual(feeOptions);\n        args[0]!.lowFeeTxid!.should.equal(lowFeeTxid);\n        args[0]!.receiveAddress!.should.equal(receiveAddress);\n        args[0]!.intentType.should.equal('acceleration');\n        args[1]!.should.equal('full');\n      });\n\n      it('should call prebuildTxWithIntent with the correct params for eth fillNonce', async function () {\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n\n        const prebuildTxWithIntent = sandbox.stub(ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequestFull);\n\n        const nonce = '1';\n        const comment = 'fillNonce comment';\n\n        await tssEthWallet.prebuildTransaction({\n          reqId,\n          type: 'fillNonce',\n          feeOptions,\n          nonce,\n          comment,\n        });\n\n        sinon.assert.calledOnce(prebuildTxWithIntent);\n        const args = prebuildTxWithIntent.args[0];\n        args[0]!.should.not.have.property('recipients');\n        args[0]!.feeOptions!.should.deepEqual(feeOptions);\n        args[0]!.nonce!.should.equal(nonce);\n        args[0]!.intentType.should.equal('fillNonce');\n        args[0]!.comment!.should.equal(comment);\n        args[1]!.should.equal('full');\n      });\n\n      it('should call prebuildTxWithIntent with the correct params for eth fillNonce for receive address nonce filling tx', async function () {\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n\n        const prebuildTxWithIntent = sandbox.stub(ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequestFull);\n\n        const nonce = '1';\n        const comment = 'fillNonce comment';\n        const receiveAddress = '0x062176bc9345da3e8ee90361b0cf6ff883ba7206';\n\n        await tssEthWallet.prebuildTransaction({\n          reqId,\n          type: 'fillNonce',\n          feeOptions,\n          nonce,\n          receiveAddress,\n          comment,\n        });\n\n        sinon.assert.calledOnce(prebuildTxWithIntent);\n        const args = prebuildTxWithIntent.args[0];\n        args[0]!.should.not.have.property('recipients');\n        args[0]!.feeOptions!.should.deepEqual(feeOptions);\n        args[0]!.nonce!.should.equal(nonce);\n        args[0]!.intentType.should.equal('fillNonce');\n        args[0]!.comment!.should.equal(comment);\n        args[0]!.receiveAddress!.should.equal(receiveAddress);\n        args[1]!.should.equal('full');\n      });\n\n      it('should call prebuildTxWithIntent with the correct feeOptions when passing using the legacy format', async function () {\n        const recipients = [\n          {\n            address: '0xAB100912e133AA06cEB921459aaDdBd62381F5A3',\n            amount: '1000',\n          },\n        ];\n\n        const expectedFeeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n          gasLimit: undefined,\n        };\n\n        const prebuildTxWithIntent = sandbox.stub(ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequestFull);\n\n        await tssEthWallet.prebuildTransaction({\n          reqId,\n          recipients,\n          type: 'transfer',\n          eip1559: {\n            maxFeePerGas: expectedFeeOptions.maxFeePerGas.toString(),\n            maxPriorityFeePerGas: expectedFeeOptions.maxPriorityFeePerGas.toString(),\n          },\n        });\n\n        sinon.assert.calledOnce(prebuildTxWithIntent);\n        const args = prebuildTxWithIntent.args[0];\n        args[0]!.feeOptions!.should.deepEqual(expectedFeeOptions);\n      });\n\n      it('populate intent should return valid eth acceleration intent', async function () {\n        const mpcUtils = new ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));\n\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n        const lowFeeTxid = '0x6ea07f9420f4676be6478ab1660eb92444a7c663e0e24bece929f715e882e0cf';\n\n        const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {\n          reqId,\n          intentType: 'acceleration',\n          lowFeeTxid,\n          feeOptions,\n        });\n\n        intent.should.have.property('recipients', undefined);\n        intent.feeOptions!.should.deepEqual(feeOptions);\n        intent.txid!.should.equal(lowFeeTxid);\n        intent.intentType.should.equal('acceleration');\n      });\n\n      it('populate intent should return valid eth acceleration intent for receive address', async function () {\n        const mpcUtils = new ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));\n\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n        const lowFeeTxid = '0x6ea07f9420f4676be6478ab1660eb92444a7c663e0e24bece929f715e882e0cf';\n        const receiveAddress = '0x062176bc9345da3e8ee90361b0cf6ff883ba7206';\n\n        const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {\n          reqId,\n          intentType: 'acceleration',\n          lowFeeTxid,\n          receiveAddress,\n          feeOptions,\n        });\n\n        intent.should.have.property('recipients', undefined);\n        intent.feeOptions!.should.deepEqual(feeOptions);\n        intent.txid!.should.equal(lowFeeTxid);\n        intent.receiveAddress!.should.equal(receiveAddress);\n        intent.intentType.should.equal('acceleration');\n      });\n\n      it('populate intent should return valid eth fillNonce intent', async function () {\n        const mpcUtils = new ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n        const nonce = '1';\n\n        const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {\n          reqId,\n          intentType: 'fillNonce',\n          nonce,\n          feeOptions,\n        });\n\n        intent.should.have.property('recipients', undefined);\n        intent.feeOptions!.should.deepEqual(feeOptions);\n        intent.nonce!.should.equal(nonce);\n        intent.intentType.should.equal('fillNonce');\n      });\n\n      it('populate intent should return valid eth fillNonce intent for receive address nonce filling tx', async function () {\n        const mpcUtils = new ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n        const nonce = '1';\n        const receiveAddress = '0x062176bc9345da3e8ee90361b0cf6ff883ba7206';\n\n        const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {\n          reqId,\n          intentType: 'fillNonce',\n          nonce,\n          receiveAddress,\n          feeOptions,\n        });\n\n        intent.should.have.property('recipients', undefined);\n        intent.feeOptions!.should.deepEqual(feeOptions);\n        intent.nonce!.should.equal(nonce);\n        intent.receiveAddress!.should.equal(receiveAddress);\n        intent.intentType.should.equal('fillNonce');\n      });\n\n      it('should populate intent with custodianTransactionId', async function () {\n        const mpcUtils = new ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));\n        const feeOptions = {\n          maxFeePerGas: 3000000000,\n          maxPriorityFeePerGas: 2000000000,\n        };\n        const nonce = '1';\n\n        const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {\n          custodianTransactionId: 'unittest',\n          reqId,\n          intentType: 'fillNonce',\n          nonce,\n          feeOptions,\n          isTss: true,\n        });\n\n        intent.custodianTransactionId!.should.equal('unittest');\n        intent.should.have.property('recipients', undefined);\n        intent.feeOptions!.should.deepEqual(feeOptions);\n        intent.nonce!.should.equal(nonce);\n        intent.isTss!.should.equal(true);\n        intent.intentType.should.equal('fillNonce');\n      });\n\n      it('should build a single recipient transfer transaction providing apiVersion parameter as \"full\" ', async function () {\n        const recipients = [\n          {\n            address: '6DadkZcx9JZgeQUDbHh12cmqCpaqehmVxv6sGy49jrah',\n            amount: '1000',\n          },\n        ];\n\n        const prebuildTxWithIntent = sandbox.stub(TssUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequestFull);\n        prebuildTxWithIntent.calledOnceWithExactly(\n          {\n            reqId,\n            recipients,\n            intentType: 'payment',\n          },\n          'full'\n        );\n\n        const txPrebuild = await custodialTssSolWallet.prebuildTransaction({\n          reqId,\n          apiVersion: 'full',\n          recipients,\n          type: 'transfer',\n        });\n\n        txPrebuild.should.deepEqual({\n          walletId: tssSolWallet.id(),\n          wallet: custodialTssSolWallet,\n          txRequestId: 'id',\n          txHex: 'ababcdcd',\n          buildParams: {\n            apiVersion: 'full',\n            recipients,\n            type: 'transfer',\n          },\n          feeInfo: {\n            fee: 5000,\n            feeString: '5000',\n          },\n        });\n      });\n    });\n\n    describe('Transaction signing', function () {\n      it('should sign transaction', async function () {\n        const signTxRequest = sandbox.stub(TssUtils.prototype, 'signTxRequest');\n        signTxRequest.resolves(txRequest);\n        // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke signTransaction\n        signTxRequest.calledOnceWithExactly({ txRequest, prv: 'secretKey', reqId });\n\n        const txPrebuild = {\n          walletId: tssSolWallet.id(),\n          wallet: tssSolWallet,\n          txRequestId: 'id',\n          txHex: 'ababcdcd',\n        };\n        const signedTransaction = await tssSolWallet.signTransaction({\n          reqId,\n          txPrebuild,\n          prv: 'sercretKey',\n        });\n        signedTransaction.should.deepEqual(txRequest);\n      });\n\n      it('should fail to sign transaction without txRequestId', async function () {\n        const txPrebuild = {\n          walletId: tssSolWallet.id(),\n          wallet: tssSolWallet,\n          txHex: 'ababcdcd',\n        };\n        await tssSolWallet\n          .signTransaction({\n            reqId,\n            txPrebuild,\n            prv: 'sercretKey',\n          })\n          .should.be.rejectedWith('txRequestId required to sign transactions with TSS');\n      });\n    });\n\n    describe('getUserKeyAndSignTssTransaction', function () {\n      ['eddsa', 'ecdsa'].forEach((keyCurve: string) => {\n        describe(keyCurve, () => {\n          const wallet = keyCurve === 'eddsa' ? tssSolWallet : tssEthWallet;\n          let getKeysStub: sinon.SinonStub;\n          let signTransactionStub: sinon.SinonStub;\n          beforeEach(function () {\n            getKeysStub = sandbox.stub(Keychains.prototype, 'getKeysForSigning');\n\n            signTransactionStub = sandbox\n              .stub(Wallet.prototype, 'signTransaction')\n              .resolves({ ...txRequestFull, state: 'signed' });\n          });\n\n          afterEach(function () {\n            sandbox.verifyAndRestore();\n          });\n          it('should sign transaction', async function () {\n            getKeysStub.resolves([\n              {\n                commonKeychain: 'test',\n                id: '',\n                pub: '',\n                type: 'tss',\n                encryptedPrv:\n                  '{\"iv\":\"15FsbDVI1zG9OggD8YX+Hg==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"hHbNH3Sz/aU=\",\"ct\":\"WoNVKz7afiRxXI2w/YkzMdMyoQg/B15u1Q8aQgi96jJZ9wk6TIaSEc6bXFH3AHzD9MdJCWJQUpRhoQc/rgytcn69scPTjKeeyVMElGCxZdFVS/psQcNE+lue3//2Zlxj+6t1NkvYO+8yAezSMRBK5OdftXEjNQI=\"}',\n              },\n            ]);\n            const params = {\n              walletPassphrase: TestBitGo.V2.TEST_ETH_WALLET_PASSPHRASE as string,\n              txRequestId: 'id',\n            };\n\n            const response = await wallet.getUserKeyAndSignTssTransaction(params);\n            response.should.deepEqual({ ...txRequestFull, state: 'signed' });\n\n            getKeysStub.calledOnce.should.be.true();\n            signTransactionStub.calledOnce.should.be.true();\n          });\n\n          it('should throw if the keychain doesnt have the encryptedKey', async function () {\n            getKeysStub.resolves([{ commonKeychain: 'test', id: '', pub: '', type: 'tss' }]);\n            const params = {\n              walletPassphrase: TestBitGo.V2.TEST_ETH_WALLET_PASSPHRASE as string,\n              txRequestId: 'id',\n            };\n\n            await wallet\n              .getUserKeyAndSignTssTransaction(params)\n              .should.be.rejectedWith('the user keychain does not have property encryptedPrv');\n\n            getKeysStub.calledOnce.should.be.true();\n            signTransactionStub.notCalled.should.be.true();\n          });\n\n          it('should throw if password is invalid', async function () {\n            getKeysStub.resolves([\n              {\n                commonKeychain: 'test',\n                id: '',\n                pub: '',\n                type: 'tss',\n                encryptedPrv:\n                  '{\"iv\":\"15FsbDVI1zG9OggD8YX+Hg==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"hHbNH3Sz/aU=\",\"ct\":\"WoNVKz7afiRxXI2w/YkzMdMyoQg/B15u1Q8aQgi96jJZ9wk6TIaSEc6bXFH3AHzD9MdJCWJQUpRhoQc/rgytcn69scPTjKeeyVMElGCxZdFVS/psQcNE+lue3//2Zlxj+6t1NkvYO+8yAezSMRBK5OdftXEjNQI=\"}',\n              },\n            ]);\n            const params = {\n              walletPassphrase: 'randompass',\n              txRequestId: 'id',\n            };\n\n            await wallet\n              .getUserKeyAndSignTssTransaction(params)\n              .should.be.rejectedWith(`unable to decrypt keychain with the given wallet passphrase`);\n\n            getKeysStub.calledOnce.should.be.true();\n            signTransactionStub.notCalled.should.be.true();\n          });\n        });\n      });\n    });\n\n    describe('signAndSendTxRequest', function () {\n      const exampleSignedTx = {\n        txHex: '0x123',\n        txid: '0x456',\n        status: 'signed',\n      };\n\n      afterEach(async function () {\n        sandbox.restore();\n      });\n\n      it('should sign lite transaction', async function () {\n        const getUserKeyAndSignTssTxSpy = sandbox.stub(tssSolWallet, 'getUserKeyAndSignTssTransaction');\n        getUserKeyAndSignTssTxSpy.resolves(exampleSignedTx);\n        const submitTxSpy = sandbox.stub(tssSolWallet, 'submitTransaction');\n        submitTxSpy.resolves(exampleSignedTx);\n\n        const signedTx = await tssSolWallet.signAndSendTxRequest({\n          walletPassphrase: 'passphrase',\n          txRequestId: 'id',\n          isTxRequestFull: false,\n        });\n\n        sandbox.assert.calledOnce(getUserKeyAndSignTssTxSpy);\n        sandbox.assert.calledOnce(submitTxSpy);\n        signedTx.should.deepEqual(exampleSignedTx);\n      });\n\n      it('should sign full transaction', async function () {\n        const deleteSignatureSharesSpy = sandbox.stub(TssUtils.prototype, 'deleteSignatureShares');\n        const getUserKeyAndSignTssTxSpy = sandbox.stub(tssSolWallet, 'getUserKeyAndSignTssTransaction');\n        getUserKeyAndSignTssTxSpy.resolves(exampleSignedTx);\n\n        const signedTx = await tssSolWallet.signAndSendTxRequest({\n          walletPassphrase: 'passphrase',\n          txRequestId: 'id',\n          isTxRequestFull: true,\n        });\n\n        sandbox.assert.calledOnce(deleteSignatureSharesSpy);\n        sandbox.assert.calledOnce(getUserKeyAndSignTssTxSpy);\n        signedTx.should.deepEqual(exampleSignedTx);\n      });\n    });\n\n    describe('Message Signing', function () {\n      const txHash = '0xrrrsss1b';\n      const txRequestForMessageSigning: TxRequest = {\n        txRequestId: reqId.toString(),\n        transactions: [],\n        intent: {\n          intentType: 'signMessage',\n        },\n        date: new Date().toISOString(),\n        latest: true,\n        state: 'pendingUserSignature',\n        userId: 'userId',\n        walletType: 'hot',\n        policiesChecked: false,\n        version: 1,\n        walletId: 'walletId',\n        unsignedTxs: [],\n        unsignedMessages: [],\n        messages: [\n          {\n            state: 'signed',\n            messageRaw: 'hello world',\n            derivationPath: 'm/0',\n            signatureShares: [{ from: SignatureShareType.USER, to: SignatureShareType.USER, share: '' }],\n            combineSigShare: '0:rrr:sss:3',\n            txHash,\n          },\n        ],\n      };\n      let signTxRequestForMessage;\n      const messageSigningCoins = ['teth', 'tpolygon'];\n      const messageRaw = 'test';\n      const expected: SignedMessage = {\n        txRequestId: reqId.toString(),\n        txHash,\n        signature: txHash,\n        messageRaw,\n        coin: 'teth',\n        messageEncoded: Buffer.from('\\u0019Ethereum Signed Message:\\n4test').toString('hex'),\n      };\n\n      beforeEach(async function () {\n        signTxRequestForMessage = sandbox.stub(ECDSAUtils.EcdsaUtils.prototype, 'signTxRequestForMessage');\n        signTxRequestForMessage.resolves(txRequestForMessageSigning);\n        sandbox\n          .stub(Keychains.prototype, 'getKeysForSigning')\n          .resolves([{ commonKeychain: 'test', id: '', pub: '', type: 'independent' }]);\n        sinon.stub(Ecdsa.prototype, 'verify').resolves(true);\n      });\n\n      afterEach(async function () {\n        sinon.restore();\n        nock.cleanAll();\n      });\n\n      it('should throw error for unsupported coins', async function () {\n        await tssSolWallet\n          .signMessage({\n            reqId,\n            message: { messageRaw },\n            prv: 'secretKey',\n          })\n          .should.be.rejectedWith('Message signing not supported for Testnet Solana');\n      });\n\n      messageSigningCoins.map((coinName) => {\n        const expectedWithCoinField = { ...expected, coin: 'teth' };\n\n        tssEthWallet = new Wallet(bitgo, bitgo.coin(coinName), ethWalletData);\n        const txRequestId = txRequestForMessageSigning.txRequestId;\n\n        it('should sign message', async function () {\n          const signMessageTssSpy = sinon.spy(tssEthWallet, 'signMessageTss' as any);\n          nock(bgUrl)\n            .get(\n              `/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${\n                txRequestForMessageSigning.txRequestId\n              }&latest=true`\n            )\n            .reply(200, { txRequests: [txRequestForMessageSigning] });\n\n          const signMessage = await tssEthWallet.signMessage({\n            reqId,\n            message: { messageRaw, txRequestId },\n            prv: 'secretKey',\n          });\n          signMessage.should.deepEqual(expectedWithCoinField);\n          const actualArg = signMessageTssSpy.getCalls()[0].args[0] as WalletSignMessageOptions;\n          actualArg.message?.messageEncoded?.should.equal(\n            Buffer.from(`\\u0019Ethereum Signed Message:\\n${messageRaw.length}${messageRaw}`).toString('hex')\n          );\n        });\n\n        it('should sign message when custodianMessageId is provided', async function () {\n          const signMessageTssSpy = sinon.spy(tssEthWallet, 'signMessageTss' as any);\n          nock(bgUrl).post(`/api/v2/wallet/${tssEthWallet.id()}/txrequests`).reply(200, txRequestForMessageSigning);\n\n          const signMessage = await tssEthWallet.signMessage({\n            custodianMessageId: 'unittest',\n            reqId,\n            message: { messageRaw },\n            prv: 'secretKey',\n          });\n          signMessage.should.deepEqual(expectedWithCoinField);\n          const actualArg = signMessageTssSpy.getCalls()[0].args[0] as WalletSignMessageOptions;\n          actualArg.message?.messageEncoded?.should.equal(\n            Buffer.from(`\\u0019Ethereum Signed Message:\\n${messageRaw.length}${messageRaw}`).toString('hex')\n          );\n        });\n\n        it('should sign message when txRequestId not provided', async function () {\n          const signMessageTssSpy = sinon.spy(tssEthWallet, 'signMessageTss' as any);\n          nock(bgUrl).post(`/api/v2/wallet/${tssEthWallet.id()}/txrequests`).reply(200, txRequestForMessageSigning);\n\n          const signMessage = await tssEthWallet.signMessage({\n            reqId,\n            message: { messageRaw },\n            prv: 'secretKey',\n          });\n          signMessage.should.deepEqual(expectedWithCoinField);\n          const actualArg = signMessageTssSpy.getCalls()[0].args[0] as WalletSignMessageOptions;\n          actualArg.message?.messageEncoded?.should.equal(\n            Buffer.from(`\\u0019Ethereum Signed Message:\\n${messageRaw.length}${messageRaw}`).toString('hex')\n          );\n        });\n\n        it('should fail to sign message with empty prv', async function () {\n          await tssEthWallet\n            .signMessage({\n              reqId,\n              message: { messageRaw, txRequestId },\n              prv: '',\n            })\n            .should.be.rejectedWith('keychain does not have property encryptedPrv');\n        });\n      });\n    });\n\n    describe('Typed Data Signing', function () {\n      const txHash =\n        '1901493fbf2ae1c27c3ced26a89070c6ab5d3fbf37ed778de9378e7703b7d1f116b3883077a61826129b98b622e54fc68c5008d1b1c16552e1eda6916f870d719220';\n      const txRequestForTypedDataSigning: TxRequest = {\n        txRequestId: reqId.toString(),\n        transactions: [],\n        intent: {\n          intentType: 'signMessage',\n        },\n        date: new Date().toISOString(),\n        latest: true,\n        state: 'pendingUserSignature',\n        userId: 'userId',\n        walletType: 'hot',\n        policiesChecked: false,\n        version: 1,\n        walletId: 'walletId',\n        unsignedTxs: [],\n        unsignedMessages: [],\n        messages: [\n          {\n            state: 'signed',\n            messageRaw: 'hello world',\n            derivationPath: 'm/0',\n            signatureShares: [{ from: SignatureShareType.USER, to: SignatureShareType.USER, share: '' }],\n            combineSigShare: '0:rrr:sss:3',\n            txHash,\n          },\n        ],\n      };\n      let signTxRequestForMessage;\n      const messageSigningCoins = ['teth', 'tpolygon'];\n      const types: MessageTypes = {\n        EIP712Domain: [\n          {\n            name: 'name',\n            type: 'string',\n          },\n          {\n            name: 'version',\n            type: 'string',\n          },\n          {\n            name: 'chainId',\n            type: 'uint256',\n          },\n          {\n            name: 'verifyingContract',\n            type: 'address',\n          },\n        ],\n        Message: [{ name: 'data', type: 'string' }],\n      };\n      const typedMessage: TypedMessage<MessageTypes> = {\n        domain: {\n          name: 'bitgo',\n          version: '1',\n          chainId: 1,\n          verifyingContract: '0x0000000000000000000000000000000000000000',\n        },\n        primaryType: 'Message',\n        types,\n        message: { data: 'bitgo says hello!' },\n      };\n      const typedDataBase: TypedData = {\n        typedDataRaw: JSON.stringify(typedMessage),\n        version: SignTypedDataVersion.V3,\n      };\n\n      beforeEach(async function () {\n        signTxRequestForMessage = sandbox.stub(ECDSAUtils.EcdsaUtils.prototype, 'signTxRequestForMessage');\n        signTxRequestForMessage.resolves(txRequestForTypedDataSigning);\n        sandbox\n          .stub(Keychains.prototype, 'getKeysForSigning')\n          .resolves([{ commonKeychain: 'test', id: '', pub: '', type: 'independent' }]);\n        sinon.stub(Ecdsa.prototype, 'verify').resolves(true);\n      });\n\n      afterEach(async function () {\n        sinon.restore();\n        nock.cleanAll();\n      });\n\n      it('should throw error for unsupported coins', async function () {\n        await tssSolWallet\n          .signTypedData({\n            reqId,\n            typedData: typedDataBase,\n            prv: 'secretKey',\n          })\n          .should.be.rejectedWith('Sign typed data not supported for Testnet Solana');\n      });\n\n      it('should throw error for sign typed data V1', async function () {\n        const typedData = { ...typedDataBase };\n        typedData.version = SignTypedDataVersion.V1;\n        nock(bgUrl)\n          .get(\n            `/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${\n              txRequestForTypedDataSigning.txRequestId\n            }&latest=true`\n          )\n          .reply(200, { txRequests: [txRequestForTypedDataSigning] });\n\n        await tssEthWallet\n          .signTypedData({\n            reqId,\n            typedData,\n            prv: 'secretKey',\n          })\n          .should.be.rejectedWith('SignTypedData v1 is not supported due to security concerns');\n      });\n      messageSigningCoins.map((coinName) => {\n        tssEthWallet = new Wallet(bitgo, bitgo.coin(coinName), ethWalletData);\n        const txRequestId = txRequestForTypedDataSigning.txRequestId;\n        typedDataBase.txRequestId = txRequestId;\n        const expected: SignedMessage = {\n          txRequestId,\n          messageRaw: JSON.stringify(typedMessage),\n          signature: txHash,\n          txHash,\n          coin: 'teth',\n          messageEncoded: txHash,\n        };\n\n        describe(`sign typed data V3 for ${coinName}`, async function () {\n          const typedData = { ...typedDataBase };\n          typedData.version = SignTypedDataVersion.V3;\n\n          it('should sign typed data V3', async function () {\n            const signTypedDataTssSpy = sinon.spy(tssEthWallet, 'signTypedDataTss' as any);\n            nock(bgUrl)\n              .get(\n                `/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${\n                  txRequestForTypedDataSigning.txRequestId\n                }&latest=true`\n              )\n              .reply(200, { txRequests: [txRequestForTypedDataSigning] });\n\n            const signedTypedData = await tssEthWallet.signTypedData({\n              reqId,\n              typedData,\n              prv: 'secretKey',\n            });\n            signedTypedData.should.deepEqual(expected);\n            const actualArg = signTypedDataTssSpy.getCalls()[0].args[0] as WalletSignTypedDataOptions;\n            actualArg.typedData?.typedDataEncoded?.toString('hex').should.equal(txHash);\n          });\n\n          it('should sign typed data V3 when custodianMessageID is provided', async function () {\n            typedData.txRequestId = txRequestId;\n            const signTypedDataTssSpy = sinon.spy(tssEthWallet, 'signTypedDataTss' as any);\n            nock(bgUrl)\n              .get(\n                `/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${\n                  txRequestForTypedDataSigning.txRequestId\n                }&latest=true`\n              )\n              .reply(200, { txRequests: [txRequestForTypedDataSigning] });\n\n            const signedTypedData = await tssEthWallet.signTypedData({\n              custodianMessageId: 'unittest',\n              reqId,\n              typedData,\n              prv: 'secretKey',\n            });\n            signedTypedData.should.deepEqual(expected);\n            const actualArg = signTypedDataTssSpy.getCalls()[0].args[0] as WalletSignTypedDataOptions;\n            actualArg.typedData?.typedDataEncoded?.toString('hex').should.equal(txHash);\n          });\n\n          it('should fail to sign typed data V3 with empty prv', async function () {\n            await tssEthWallet\n              .signTypedData({\n                reqId,\n                typedData: typedDataBase,\n                prv: '',\n              })\n              .should.be.rejectedWith('keychain does not have property encryptedPrv');\n          });\n\n          it('should sign typed data V3 when txRequestId not provided ', async function () {\n            delete typedData.txRequestId;\n            const signedTypedDataTssSpy = sinon.spy(tssEthWallet, 'signTypedDataTss' as any);\n            nock(bgUrl).post(`/api/v2/wallet/${tssEthWallet.id()}/txrequests`).reply(200, txRequestForTypedDataSigning);\n\n            const signedTypedData = await tssEthWallet.signTypedData({\n              reqId,\n              typedData,\n              prv: 'secretKey',\n            });\n            signedTypedData.should.deepEqual(expected);\n            const actualArg = signedTypedDataTssSpy.getCalls()[0].args[0] as WalletSignTypedDataOptions;\n            actualArg.typedData?.typedDataEncoded?.toString('hex').should.equal(txHash);\n          });\n        });\n\n        describe(`sign typed data V4 for ${coinName}`, async function () {\n          const typedData = { ...typedDataBase };\n          typedData.version = SignTypedDataVersion.V4;\n          it('should sign typed data V4', async function () {\n            typedData.txRequestId = txRequestId;\n            nock(bgUrl)\n              .get(\n                `/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${\n                  txRequestForTypedDataSigning.txRequestId\n                }&latest=true`\n              )\n              .reply(200, { txRequests: [txRequestForTypedDataSigning] });\n\n            const signedTypedData = await tssEthWallet.signTypedData({\n              reqId,\n              typedData,\n              prv: 'secretKey',\n            });\n            signedTypedData.should.deepEqual(expected);\n          });\n\n          it('should sign typed data V4 when custodianMessageID is provided', async function () {\n            typedData.txRequestId = txRequestId;\n            nock(bgUrl)\n              .get(\n                `/api/v2/wallet/${tssEthWallet.id()}/txrequests?txRequestIds=${\n                  txRequestForTypedDataSigning.txRequestId\n                }&latest=true`\n              )\n              .reply(200, { txRequests: [txRequestForTypedDataSigning] });\n\n            const signedTypedData = await tssEthWallet.signTypedData({\n              custodianMessageId: 'unittest',\n              reqId,\n              typedData,\n              prv: 'secretKey',\n            });\n            signedTypedData.should.deepEqual(expected);\n          });\n\n          it('should fail to sign typed data V4 with empty prv', async function () {\n            await tssEthWallet\n              .signTypedData({\n                reqId,\n                typedData: typedDataBase,\n                prv: '',\n              })\n              .should.be.rejectedWith('keychain does not have property encryptedPrv');\n          });\n\n          it('should sign typed data V4 when txRequestId not provided ', async function () {\n            delete typedData.txRequestId;\n            const signedTypedDataTssSpy = sinon.spy(tssEthWallet, 'signTypedDataTss' as any);\n            nock(bgUrl).post(`/api/v2/wallet/${tssEthWallet.id()}/txrequests`).reply(200, txRequestForTypedDataSigning);\n\n            const signedTypedData = await tssEthWallet.signTypedData({\n              reqId,\n              typedData,\n              prv: 'secretKey',\n            });\n            signedTypedData.should.deepEqual(expected);\n            const actualArg = signedTypedDataTssSpy.getCalls()[0].args[0] as WalletSignTypedDataOptions;\n            actualArg.typedData?.typedDataEncoded?.toString('hex').should.equal(txHash);\n          });\n        });\n      });\n    });\n\n    describe('Send Many', function () {\n      const sendManyInput = {\n        type: 'transfer',\n        recipients: [\n          {\n            address: 'address',\n            amount: '1000',\n          },\n        ],\n        reqId: new RequestTracer(),\n      };\n\n      afterEach(function () {\n        nock.cleanAll();\n      });\n\n      it('should send many', async function () {\n        const signedTransaction = {\n          txRequestId: 'txRequestId',\n        };\n\n        const prebuildAndSignTransaction = sandbox.stub(tssSolWallet, 'prebuildAndSignTransaction');\n        prebuildAndSignTransaction.resolves(signedTransaction);\n        // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke sendMany\n        prebuildAndSignTransaction.calledOnceWithExactly(sendManyInput);\n\n        const sendTxRequest = sandbox.stub(TssUtils.prototype, 'sendTxRequest');\n        sendTxRequest.resolves('sendTxResponse');\n        // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke sendMany\n        sendTxRequest.calledOnceWithExactly(signedTransaction.txRequestId);\n\n        const sendMany = await tssSolWallet.sendMany(sendManyInput);\n        sendMany.should.deepEqual('sendTxResponse');\n      });\n\n      it('should send many and call setRequestTracer', async function () {\n        const signedTransaction = {\n          txRequestId: 'txRequestId',\n        };\n\n        const prebuildAndSignTransaction = sandbox.stub(tssSolWallet, 'prebuildAndSignTransaction');\n        prebuildAndSignTransaction.resolves(signedTransaction);\n        prebuildAndSignTransaction.calledOnceWithExactly(sendManyInput);\n\n        const sendTxRequest = sandbox.stub(TssUtils.prototype, 'sendTxRequest');\n        sendTxRequest.resolves('sendTxResponse');\n        sendTxRequest.calledOnceWithExactly(signedTransaction.txRequestId);\n\n        const setRequestTracerSpy = sinon.spy(bitgo, 'setRequestTracer');\n        setRequestTracerSpy.withArgs(sendManyInput.reqId);\n\n        const sendMany = await tssSolWallet.sendMany(sendManyInput);\n        sendMany.should.deepEqual('sendTxResponse');\n        sinon.assert.calledOnce(setRequestTracerSpy);\n        setRequestTracerSpy.restore();\n      });\n\n      it('should return transfer from sendMany for apiVersion=full', async function () {\n        const wallet = new Wallet(bitgo, tsol, {\n          ...walletData,\n          type: 'custodial',\n        });\n        const signedTxResult = {\n          txRequestId: 'txRequestId',\n        };\n        const txRequest: TxRequest = {\n          date: new Date().toString(),\n          intent: 'payment',\n          latest: false,\n          policiesChecked: false,\n          state: 'delivered',\n          unsignedTxs: [],\n          userId: 'unit-test',\n          version: 0,\n          walletId: wallet.id(),\n          walletType: wallet.type() ?? 'hot',\n          txRequestId: signedTxResult.txRequestId,\n          transactions: [\n            {\n              state: 'delivered',\n              signedTx: {\n                id: 'txid',\n                tx: 'tx',\n              },\n              unsignedTx: 'something' as any,\n              signatureShares: [],\n            },\n          ],\n        };\n        const transfer = {\n          id: 'transferId',\n          state: 'signed',\n          txid: 'txid',\n        };\n\n        const prebuildAndSignTransaction = sandbox.stub(wallet, 'prebuildAndSignTransaction').resolves(signedTxResult);\n\n        const txRequestNock = nock(bgUrl)\n          .persist()\n          .get(`/api/v2/wallet/${walletData.id}/txrequests?txRequestIds=${signedTxResult.txRequestId}&latest=true`)\n          .reply(200, { txRequests: [txRequest] });\n\n        const createTransferNock = nock(bgUrl)\n          .persist()\n          .post(`/api/v2/wallet/${walletData.id}/txrequests/${signedTxResult.txRequestId}/transfers`)\n          .reply(200, transfer);\n\n        const input: SendManyOptions = {\n          type: 'transfer',\n          recipients: [\n            {\n              address: 'address',\n              amount: '1000',\n            },\n          ],\n          apiVersion: 'full',\n        };\n        const sendManyResult = await wallet.sendMany(input);\n        prebuildAndSignTransaction.calledOnceWithExactly(input);\n        txRequestNock.isDone().should.be.true();\n        createTransferNock.isDone().should.be.true();\n\n        sendManyResult.should.deepEqual({\n          txRequest,\n          transfer,\n          txid: 'txid',\n          tx: 'tx',\n          status: 'signed',\n        });\n      });\n\n      it('should return pendingApproval from sendMany for apiVersion=full', async function () {\n        const wallet = new Wallet(bitgo, tsol, {\n          ...walletData,\n          type: 'hot',\n        });\n        const signedTxResult = {\n          txRequestId: 'txRequestId',\n        };\n        const txRequest: TxRequest = {\n          txRequestId: signedTxResult.txRequestId,\n          date: new Date().toString(),\n          intent: 'payment',\n          latest: false,\n          policiesChecked: false,\n          state: 'pendingApproval',\n          unsignedTxs: [],\n          userId: 'unit-test',\n          version: 0,\n          walletId: wallet.id(),\n          walletType: wallet.type() ?? 'hot',\n          pendingApprovalId: 'some-pending-approval-id',\n          transactions: [\n            {\n              state: 'initialized',\n              unsignedTx: 'something' as any,\n              signatureShares: [],\n            },\n          ],\n        };\n        const transfer = {\n          id: 'transferId',\n          state: 'signed',\n          txid: 'txid',\n        };\n        const pendingApproval = {\n          id: 'some-pending-approval-id',\n          wallet: wallet.id(),\n          info: {\n            type: 'transactionRequestFull',\n          },\n          txRequestId: txRequest.txRequestId,\n        };\n\n        const prebuildAndSignTransaction = sandbox.stub(wallet, 'prebuildAndSignTransaction').resolves(signedTxResult);\n\n        const txRequestNock = nock(bgUrl)\n          .persist()\n          .get(`/api/v2/wallet/${walletData.id}/txrequests?txRequestIds=${txRequest.txRequestId}&latest=true`)\n          .reply(200, { txRequests: [txRequest] });\n\n        const createTransferNock = nock(bgUrl)\n          .persist()\n          .post(`/api/v2/wallet/${walletData.id}/txrequests/${txRequest.txRequestId}/transfers`)\n          .reply(200, transfer);\n\n        const getPendingApprovalNock = nock(bgUrl)\n          .persist()\n          .get(`/api/v2/${wallet.coin()}/pendingapprovals/${txRequest.pendingApprovalId}`)\n          .reply(200, pendingApproval);\n\n        const input: SendManyOptions = {\n          type: 'transfer',\n          recipients: [\n            {\n              address: 'address',\n              amount: '1000',\n            },\n          ],\n          apiVersion: 'full',\n        };\n        const sendManyResult = await wallet.sendMany(input);\n        prebuildAndSignTransaction.calledOnceWithExactly(input);\n        txRequestNock.isDone().should.be.true();\n        createTransferNock.isDone().should.be.true();\n        getPendingApprovalNock.isDone().should.be.true();\n\n        sendManyResult.should.deepEqual({ pendingApproval, txRequest });\n      });\n\n      it('should fail if txRequestId is missing from prebuild', async function () {\n        const signedTransaction = {\n          txHex: 'deadbeef',\n        };\n\n        const prebuildAndSignTransaction = sandbox.stub(tssSolWallet, 'prebuildAndSignTransaction');\n        prebuildAndSignTransaction.resolves(signedTransaction);\n        // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke sendMany\n        prebuildAndSignTransaction.calledOnceWithExactly(sendManyInput);\n\n        await tssSolWallet\n          .sendMany(sendManyInput)\n          .should.be.rejectedWith('txRequestId missing from signed transaction');\n      });\n    });\n\n    describe('Submit transaction', function () {\n      it('should submit transaction with txRequestId', async function () {\n        const nockSendTx = nock(bgUrl)\n          .persist(false)\n          .post(tssSolWallet.url('/tx/send').replace(bgUrl, ''))\n          .reply(200, { message: 'success' });\n\n        const submittedTx = await tssSolWallet.submitTransaction({\n          txRequestId: 'id',\n        });\n        submittedTx.should.deepEqual({ message: 'success' });\n        nockSendTx.isDone().should.be.true();\n      });\n\n      it('should fail when txRequestId and txHex are both provided', async function () {\n        await tssSolWallet\n          .submitTransaction({\n            txRequestId: 'id',\n            txHex: 'beef',\n          })\n          .should.be.rejectedWith('must supply exactly one of txRequestId, txHex, or halfSigned');\n      });\n\n      it('should fail when txRequestId and halfSigned are both provided', async function () {\n        await tssSolWallet\n          .submitTransaction({\n            txRequestId: 'id',\n            halfSigned: {\n              txHex: 'beef',\n            },\n          })\n          .should.be.rejectedWith('must supply exactly one of txRequestId, txHex, or halfSigned');\n      });\n\n      it('should fail when txHex and halfSigned are both provided', async function () {\n        await tssSolWallet\n          .submitTransaction({\n            txHex: 'beef',\n            halfSigned: {\n              txHex: 'beef',\n            },\n          })\n          .should.be.rejectedWith('must supply either txHex or halfSigned, but not both');\n      });\n    });\n\n    describe('Transfer tokens', function () {\n      const recipients = [\n        {\n          address: '0x101c3928946b2e1d99759e8e5d34b5e94c1a8e2f',\n          amount: '0',\n          tokenData: {\n            tokenName: 'erc721:bitgoerc721',\n            tokenContractAddress: '0x8397b091514c1f7bebb9dea6ac267ea23b570605',\n            tokenId: '38',\n            tokenQuantity: '1',\n            decimalPlaces: 0,\n            tokenType: TokenType.ERC721,\n          },\n        },\n      ];\n\n      const feeOptions = {\n        maxFeePerGas: 2000000000,\n        maxPriorityFeePerGas: 1000000000,\n      };\n\n      it('calling prebuildxTransaction should execute prebuildTxWithIntent with proper params', async function () {\n        const txRequestFullTokenTransfer = { ...txRequestFull, intent: 'transferToken' };\n        const prebuildTxWithIntent = sandbox.stub(ECDSAUtils.EcdsaUtils.prototype, 'prebuildTxWithIntent');\n        prebuildTxWithIntent.resolves(txRequestFullTokenTransfer);\n        // TODO(BG-59686): this is not doing anything if we don't check the return value, we should also move this check to happen after we invoke prebuildTransaction\n        prebuildTxWithIntent.calledOnceWithExactly(\n          {\n            reqId,\n            recipients,\n            intentType: 'transferToken',\n            feeOptions,\n          },\n          'full'\n        );\n\n        const txPrebuild = await tssPolygonWallet.prebuildTransaction({\n          isTss: true,\n          recipients,\n          type: 'transfertoken',\n          walletPassphrase: 'passphrase12345',\n          feeOptions,\n        });\n\n        txPrebuild.should.deepEqual({\n          walletId: tssPolygonWallet.id(),\n          wallet: tssPolygonWallet,\n          txRequestId: 'id',\n          txHex: 'ababcdcd',\n          buildParams: {\n            recipients,\n            type: 'transfertoken',\n          },\n          feeInfo: {\n            fee: 5000,\n            feeString: '5000',\n          },\n        });\n      });\n\n      it('should populate intent with EVM-like params', async function () {\n        const mpcUtils = new ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('tpolygon'));\n        // @ts-expect-error only pass in params being tested\n        const intent = mpcUtils.populateIntent(bitgo.coin('tpolygon'), {\n          intentType: 'transferToken',\n          recipients,\n          feeOptions,\n        });\n        intent.should.have.property('feeOptions');\n        intent.feeOptions!.should.have.property('maxFeePerGas', 2000000000);\n        intent.feeOptions!.should.have.property('maxPriorityFeePerGas', 1000000000);\n        intent.should.have.property('recipients');\n        intent.recipients!.should.have.property('length', 1);\n        intent.recipients![0].should.have.property('tokenData');\n        intent.recipients![0].tokenData!.should.have.property('tokenQuantity', recipients[0].tokenData.tokenQuantity);\n        intent.recipients![0].tokenData!.should.have.property('tokenType', recipients[0].tokenData.tokenType);\n        intent.recipients![0].tokenData!.should.have.property('tokenName', recipients[0].tokenData.tokenName);\n        intent.recipients![0].tokenData!.should.have.property(\n          'tokenContractAddress',\n          recipients[0].tokenData.tokenContractAddress\n        );\n        intent.recipients![0].tokenData!.should.have.property('tokenId', recipients[0].tokenData.tokenId);\n        intent.recipients![0].tokenData!.should.have.property('decimalPlaces', recipients[0].tokenData.decimalPlaces);\n      });\n\n      it('should populate intent with calldata', async function () {\n        const recipients = [\n          {\n            address: '0x101c3928946b2e1d99759e8e5d34b5e94c1a8e2f',\n            amount: '0',\n            data: '0x000011112222',\n          },\n        ];\n\n        const mpcUtils = new ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('hteth'));\n        // @ts-expect-error only pass in params being tested\n        const intent = mpcUtils.populateIntent(bitgo.coin('hteth'), {\n          intentType: 'payment',\n          recipients,\n          feeOptions,\n        });\n\n        intent.should.have.property('feeOptions');\n        intent.feeOptions!.should.have.property('maxFeePerGas', 2000000000);\n        intent.feeOptions!.should.have.property('maxPriorityFeePerGas', 1000000000);\n        intent.should.have.property('recipients');\n        intent.recipients!.should.have.property('length', 1);\n        intent.recipients![0].data!.should.equal('0x000011112222');\n      });\n\n      it('should not populate intent with tokenData if certain params are undefined', async function () {\n        const mpcUtils = new ECDSAUtils.EcdsaUtils(bitgo, bitgo.coin('tpolygon'));\n        const recipients = [\n          {\n            address: '0x101c3928946b2e1d99759e8e5d34b5e94c1a8e2f',\n            amount: '0',\n            tokenData: {\n              tokenName: 'erc721:bitgoerc721',\n              tokenContractAddress: '0x8397b091514c1f7bebb9dea6ac267ea23b570605',\n              tokenId: '38',\n              tokenQuantity: '1',\n              decimalPlaces: 0,\n            },\n          },\n        ];\n        let intent;\n        try {\n          intent = mpcUtils.populateIntent(bitgo.coin('tpolygon'), {\n            intentType: 'transferToken',\n            // @ts-expect-error only pass in params be tested for\n            recipients,\n            feeOptions,\n          });\n          intent.should.equal(undefined);\n        } catch (e: any) {\n          e.message.should.equal(\n            'token type and quantity is required to request a transaction with intent to transfer a token'\n          );\n        }\n      });\n    });\n\n    describe('Transfer NFTs', function () {\n      it('should populate intent with NFT token details', async function () {\n        const params: PrebuildTransactionWithIntentOptions = {\n          reqId,\n          intentType: 'payment',\n          recipients: [\n            {\n              address: '0x9fef749050644625012a2c866973775e7123753b3eef0a1a4037453ac26d79bf',\n              amount: '1',\n              tokenData: {\n                tokenType: TokenType.DIGITAL_ASSET,\n                tokenQuantity: '1',\n                tokenContractAddress: '0xbbc561fbfa5d105efd8dfb06ae3e7e5be46331165b99d518f094c701e40603b5',\n                tokenId: '0x675b053d72c24dcc6bc7f38cdd45b4843cfb7af69a25ad21d002c376357e9d69',\n              },\n            },\n          ],\n        };\n\n        const baseCoin = bitgo.coin('tapt');\n        const mpcUtils = new EDDSAUtils.default(bitgo, baseCoin);\n        const intent = mpcUtils.populateIntent(baseCoin, params);\n\n        intent.should.have.property('intentType', 'payment');\n        intent.recipients!.should.deepEqual([\n          {\n            address: {\n              address: '0x9fef749050644625012a2c866973775e7123753b3eef0a1a4037453ac26d79bf',\n            },\n            amount: {\n              value: '1',\n              symbol: 'tapt:nftcollection1',\n            },\n            tokenData: {\n              tokenType: 'Digital Asset',\n              tokenQuantity: '1',\n              tokenContractAddress: '0xbbc561fbfa5d105efd8dfb06ae3e7e5be46331165b99d518f094c701e40603b5',\n              tokenId: '0x675b053d72c24dcc6bc7f38cdd45b4843cfb7af69a25ad21d002c376357e9d69',\n              tokenName: 'tapt:nftcollection1',\n            },\n          },\n        ]);\n      });\n    });\n\n    describe('Wallet Sharing', function () {\n      it('should use keychain pub to share tss wallet', async function () {\n        const userId = '123';\n        const email = 'shareto@sdktest.com';\n        const permissions = 'view,spend';\n        const toKeychain = utxoLib.bip32.fromSeed(Buffer.from('deadbeef02deadbeef02deadbeef02deadbeef02', 'hex'));\n        const path = 'm/999999/1/1';\n        const pubkey = toKeychain.derivePath(path).publicKey.toString('hex');\n        const walletPassphrase = 'bitgo1234';\n\n        const getSharingKeyNock = nock(bgUrl)\n          .post('/api/v1/user/sharingkey', { email })\n          .reply(200, { userId, pubkey, path });\n\n        // commonPub + commonChaincode\n        const commonKeychain = randomBytes(32).toString('hex') + randomBytes(32).toString('hex');\n        const getKeyNock = nock(bgUrl)\n          .get(`/api/v2/tsol/key/${tssSolWallet.keyIds()[0]}`)\n          .reply(200, {\n            id: tssSolWallet.keyIds()[0],\n            commonKeychain: commonKeychain,\n            source: 'user',\n            encryptedPrv: bitgo.encrypt({ input: 'xprv1', password: walletPassphrase }),\n            coinSpecific: {},\n          });\n\n        const stub = sinon.stub(tssSolWallet, 'createShare').callsFake(async (options) => {\n          // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n          options!.keychain!.pub!.should.not.be.undefined();\n          // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n          options!.keychain!.pub!.should.equal(TssUtils.getPublicKeyFromCommonKeychain(commonKeychain));\n          return undefined;\n        });\n        await tssSolWallet.shareWallet({ email, permissions, walletPassphrase });\n\n        stub.calledOnce.should.be.true();\n        getSharingKeyNock.isDone().should.be.True();\n        getKeyNock.isDone().should.be.True();\n      });\n    });\n  });\n\n  describe('AVAX tests', function () {\n    let bgUrl;\n    let basecoin;\n    let walletData;\n    let wallet;\n\n    before(async function () {\n      nock.pendingMocks().should.be.empty();\n      bgUrl = common.Environments[bitgo.getEnv()].uri;\n      walletData = {\n        id: '5b34252f1bf349930e34020a00000000',\n        keys: [\n          '5b3424f91bf349930e34017500000000',\n          '5b3424f91bf349930e34017600000000',\n          '5b3424f91bf349930e34017700000000',\n        ],\n        coinSpecific: {},\n      };\n    });\n\n    it('should fetch cross-chain utxos', async function () {\n      basecoin = bitgo.coin('tavaxp');\n      walletData.coin = 'tavaxp';\n      wallet = new Wallet(bitgo, basecoin, walletData);\n\n      const params = { sourceChain: 'C' };\n      const path = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/crossChainUnspents`;\n      const scope = nock(bgUrl)\n        .get(path)\n        .query(params)\n        .reply(200, {\n          unspent: {\n            outputID: 7,\n            amount: '10000000',\n            txid: 'V3UBZTQj364zNWqt8uMHD5NjxxX8T8qkbeZXURmjnVmLEqzab',\n            threshold: 2,\n            addresses: [\n              'C-fuji199fluegrthqs4tvz40zajfrsx5m7dvy75ajfm6',\n              'C-fuji1gk3m444893ynl0gfvxahjgw3vftnn8sptyd9g5',\n              'C-fuji1ujfzjgwzfygl60qp2l8rmglg3lnm7w4059nca5',\n            ],\n            outputidx: '1111XiaYg',\n            locktime: '0',\n          },\n          fromWallet: '635092fd4ff3316142df6e6b7a078b92',\n          toWallet: '635092fd4ff3316142df6e891f6a7ee6',\n          toAddress: '0x125c4451c870f753265b0b1af3cf6ab88ffe4657',\n        });\n\n      try {\n        await wallet.fetchCrossChainUTXOs(params);\n      } catch (e) {\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n\n      scope.isDone().should.be.True();\n    });\n\n    it('sendMany should work for C > P export with custodial wallet', async function () {\n      basecoin = bitgo.coin('tavaxc');\n      walletData.coin = 'tavaxc';\n      walletData.type = 'custodial';\n      wallet = new Wallet(bitgo, basecoin, walletData);\n\n      const address =\n        'P-fuji1e56pc4966qsevzhwgkym5l0jfma9llkqnrr4gh~P-fuji1kq05zm9nmlq8p3ld55k79dl3qay6c0e3atj56v~P-fuji1rp46z30qg457xc3dpffyxcgzpflxc85mhkjme3';\n      const initiateTxParams = {\n        recipients: [\n          {\n            amount: '10000000000000000', // 0.01 AVAX\n            address,\n          },\n        ],\n        hop: true,\n        type: 'Export',\n      };\n\n      const initiateTxPath = `/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/initiate`;\n      let initiateTxBody;\n      const response = nock(bgUrl)\n        .post(initiateTxPath, (body) => {\n          initiateTxBody = body;\n          return true;\n        })\n        .reply(200);\n\n      const feeEstimationPath = `/api/v2/${wallet.coin()}/tx/fee?hop=true&recipient=${address}&amount=10000000000000000&type=Export`;\n      nock(bgUrl).get(feeEstimationPath).reply(200, {\n        feeEstimate: '718750000000000',\n        gasLimitEstimate: 500000,\n      });\n\n      try {\n        await wallet.sendMany(initiateTxParams);\n      } catch (e) {\n        console.log(e);\n        // test is successful if nock is consumed, HMAC errors expected\n      }\n      _.isMatch(initiateTxBody, {\n        hopParams: {\n          gasPriceMax: 7187500000,\n          gasLimit: 500000,\n        },\n        type: 'Export',\n        recipients: [\n          {\n            amount: '10000000000000000',\n            address,\n          },\n        ],\n      }).should.be.true();\n\n      response.isDone().should.be.true();\n    });\n  });\n\n  describe('NFT Tests', function () {\n    let ethWallet: Wallet;\n\n    before(async function () {\n      const walletData = {\n        id: '598f606cd8fc24710d2ebadb1d9459bb',\n        coin: 'hteth',\n        keys: [\n          '598f606cd8fc24710d2ebad89dce86c2',\n          '598f606cc8e43aef09fcb785221d9dd2',\n          '5935d59cf660764331bafcade1855fd7',\n        ],\n        multisigType: 'onchain',\n        coinSpecific: {\n          baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n        },\n      };\n      ethWallet = new Wallet(bitgo, bitgo.coin('hteth'), walletData);\n    });\n\n    afterEach(async function () {\n      nock.cleanAll();\n    });\n\n    it('Should return all nfts in the wallet', async function () {\n      const getTokenBalanceNock = nock(bgUrl)\n        .get(`/api/v2/hteth/wallet/${ethWallet.id()}?allTokens=true`)\n        .reply(200, {\n          ...walletData,\n          ...nftResponse,\n        });\n      const nfts = await ethWallet.getNftBalances();\n      getTokenBalanceNock.isDone().should.be.true();\n\n      nfts.should.length(5);\n      nfts.should.containEql({\n        type: 'ERC721',\n        metadata: {\n          name: 'terc721:bitgoerc721',\n          tokenContractAddress: '0x8397b091514c1f7bebb9dea6ac267ea23b570605',\n        },\n        collections: {},\n        balanceString: '0',\n        confirmedBalanceString: '0',\n        spendableBalanceString: '0',\n        transferCount: 0,\n      });\n    });\n\n    it('Should throw when attempting to transfer a nft collection not in the wallet', async function () {\n      const getTokenBalanceNock = nock(bgUrl)\n        .get(`/api/v2/hteth/wallet/${ethWallet.id()}?allTokens=true`)\n        .reply(200, {\n          ...walletData,\n          ...nftResponse,\n        });\n\n      await ethWallet\n        .sendNft(\n          {\n            walletPassphrase: '123abc',\n            otp: '000000',\n          },\n          {\n            tokenId: '123',\n            type: 'ERC721',\n            tokenContractAddress: '0x123badaddress',\n            recipientAddress: '0xc15acc27ee41f266877c8f0c61df5bcbc7997df6',\n          }\n        )\n        .should.be.rejectedWith('Collection not found for token contract 0x123badaddress');\n      getTokenBalanceNock.isDone().should.be.true();\n    });\n\n    it('Should throw when attempting to transfer a ERC-721 nft not owned by the wallet', async function () {\n      const getTokenBalanceNock = nock(bgUrl)\n        .get(`/api/v2/hteth/wallet/${ethWallet.id()}?allTokens=true`)\n        .reply(200, {\n          ...walletData,\n          ...nftResponse,\n          ...unsupportedNftResponse,\n        });\n\n      await ethWallet\n        .sendNft(\n          {\n            walletPassphrase: '123abc',\n            otp: '000000',\n          },\n          {\n            tokenId: '123',\n            type: 'ERC721',\n            tokenContractAddress: '0xf5de760f2e916647fd766b4ad9e85ff943ce3a2b',\n            recipientAddress: '0xc15acc27ee41f266877c8f0c61df5bcbc7997df6',\n          }\n        )\n        .should.be.rejectedWith(\n          'Token 123 not found in collection 0xf5de760f2e916647fd766b4ad9e85ff943ce3a2b or does not have a spendable balance'\n        );\n      getTokenBalanceNock.isDone().should.be.true();\n    });\n\n    it('Should throw when attempting to transfer ERC-1155 tokens when the amount transferred is more than the spendable balance', async function () {\n      const getTokenBalanceNock = nock(bgUrl)\n        .get(`/api/v2/hteth/wallet/${ethWallet.id()}?allTokens=true`)\n        .reply(200, {\n          ...walletData,\n          ...{\n            unsupportedNfts: {\n              '0xf5de760f2e916647fd766b4ad9e85ff943ce3a2b': {\n                type: 'ERC1155',\n                collections: {\n                  1186703: '9',\n                  1186705: '1',\n                  1294856: '1',\n                  1294857: '1',\n                  1294858: '1',\n                  1294859: '1',\n                  1294860: '1',\n                },\n                metadata: {\n                  name: 'MultiFaucet NFT',\n                  tokenContractAddress: '0xf5de760f2e916647fd766b4ad9e85ff943ce3a2b',\n                },\n              },\n            },\n          },\n        });\n\n      await ethWallet\n        .sendNft(\n          {\n            walletPassphrase: '123abc',\n            otp: '000000',\n          },\n          {\n            entries: [\n              {\n                amount: 10,\n                tokenId: '1186703',\n              },\n              {\n                amount: 1,\n                tokenId: '1186705',\n              },\n            ],\n            type: 'ERC1155',\n            tokenContractAddress: '0xf5de760f2e916647fd766b4ad9e85ff943ce3a2b',\n            recipientAddress: '0xc15acc27ee41f266877c8f0c61df5bcbc7997df6',\n          }\n        )\n        .should.be.rejectedWith('Amount 10 exceeds spendable balance of 9 for token 1186703');\n      getTokenBalanceNock.isDone().should.be.true();\n    });\n  });\n\n  describe('Ada tests: ', () => {\n    let adaWallet: Wallet;\n    const adaBitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    adaBitgo.initializeTestVars();\n    const walletData = {\n      id: '598f606cd8fc24710d2ebadb1d9459bb',\n      coinSpecific: {\n        baseAddress:\n          'addr_test1q9faa5q3zr38wkd4kd3u8jfshx97jxvwsyvjg3tac826502nmmgpzy8zwavmtvmrc0ynpwvtayvcaqgey3zhmsw44g7shrfrh9',\n        pendingChainInitialization: false,\n        minimumFunding: 1000000,\n        lastChainIndex: { 0: 0 },\n      },\n      coin: 'tada',\n      keys: [\n        '598f606cd8fc24710d2ebad89dce86c2',\n        '598f606cc8e43aef09fcb785221d9dd2',\n        '5935d59cf660764331bafcade1855fd7',\n      ],\n      multisigType: 'tss',\n    };\n\n    before(async function () {\n      adaWallet = new Wallet(bitgo, bitgo.coin('tada'), walletData);\n      nock(bgUrl).get(`/api/v2/${adaWallet.coin()}/key/${adaWallet.keyIds()[0]}`).times(3).reply(200, {\n        id: '598f606cd8fc24710d2ebad89dce86c2',\n        pub: '5f8WmC2uW9SAk7LMX2r4G1Bx8MMwx8sdgpotyHGodiZo',\n        source: 'user',\n        encryptedPrv:\n          '{\"iv\":\"hNK3rg82P1T94MaueXFAbA==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"cV4wU4EzPjs=\",\"ct\":\"9VZX99Ztsb6p75Cxl2lrcXBplmssIAQ9k7ZA81vdDYG4N5dZ36BQNWVfDoelj9O31XyJ+Xri0XKIWUzl0KKLfUERplmtNoOCn5ifJcZwCrOxpHZQe3AJ700o8Wmsrk5H\"}',\n        coinSpecific: {},\n      });\n\n      nock(bgUrl).get(`/api/v2/${adaWallet.coin()}/key/${adaWallet.keyIds()[1]}`).times(2).reply(200, {\n        id: '598f606cc8e43aef09fcb785221d9dd2',\n        pub: 'G1s43JTzNZzqhUn4aNpwgcc6wb9FUsZQD5JjffG6isyd',\n        encryptedPrv:\n          '{\"iv\":\"UFrt/QlIUR1XeQafPBaAlw==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"7VPBYaJXPm8=\",\"ct\":\"ajFKv2y8yaIBXQ39sAbBWcnbiEEzbjS4AoQtp5cXYqjeDRxt3aCxemPm22pnkJaCijFjJrMHbkmsNhNYzHg5aHFukN+nEAVssyNwHbzlhSnm8/BVN50yAdAAtWreh8cp\"}',\n        source: 'backup',\n        coinSpecific: {},\n      });\n\n      nock(bgUrl).get(`/api/v2/${adaWallet.coin()}/key/${adaWallet.keyIds()[2]}`).times(2).reply(200, {\n        id: '5935d59cf660764331bafcade1855fd7',\n        pub: 'GH1LV1e9FdqGe8U2c8PMEcma3fDeh1ktcGVBrD3AuFqx',\n        encryptedPrv:\n          '{\"iv\":\"iIuWOHIOErEDdiJn6g46mg==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\":\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"Rzh7RRJksj0=\",\"ct\":\"rcNICUfp9FakT53l+adB6XKzS1vNTc0Qq9jAtqnxA+ScssiS4Q0l3sgG/0gDy5DaZKtXryKBDUvGsi7b/fYaFCUpAoZn/VZTOhOUN/mo7ZHb4OhOXL29YPPkiryAq9Cr\"}',\n        source: 'bitgo',\n        coinSpecific: {},\n      });\n    });\n\n    after(async function () {\n      sinon.restore();\n      nock.cleanAll();\n    });\n\n    it('Should send unspents in payment intent when using sendmany', async function () {\n      const sendManyParams = {\n        type: 'transfer',\n        recipients: [\n          {\n            address: 'address',\n            amount: '1000',\n          },\n        ],\n        unspents: ['unspent1', 'unspent2'],\n      };\n\n      nock(bgUrl)\n        .post(`/api/v2/wallet/${adaWallet.id()}/txrequests`)\n        .reply((url, body: any) => {\n          // validate that the populated intent has unspents\n          body.intent.intentType.should.equal('payment');\n          body.intent.unspents.should.deepEqual(['unspent1', 'unspent2']);\n\n          return [\n            200,\n            {\n              apiVersion: 'lite',\n              unsignedTxs: [\n                {\n                  unsignedTx: {\n                    serializedTxHex: 'serializedTxHex',\n                    feeInfo: 'fee info',\n                  },\n                },\n              ],\n            },\n          ];\n        });\n\n      // stub all steps after txrequest creation\n      sinon.stub(adaWallet.baseCoin, 'verifyTransaction').resolves(true);\n      sinon.stub(adaWallet, 'signTransaction').resolves({ txRequestId: 'txRequestId' });\n      sinon.stub(BaseTssUtils.default.prototype, 'sendTxRequest').resolves('sendTxResponse');\n      await adaWallet.sendMany(sendManyParams);\n    });\n\n    it('Should send senderAddress in payment intent when using sendmany', async function () {\n      const sendManyParams = {\n        type: 'transfer',\n        recipients: [\n          {\n            address: 'address',\n            amount: '1000',\n          },\n        ],\n        senderAddress: 'senderAddr1',\n      };\n\n      nock(bgUrl)\n        .post(`/api/v2/wallet/${adaWallet.id()}/txrequests`)\n        .reply((url, body: nock.Body) => {\n          const createTxRequestBody = body as CreateTxRequestBody;\n          createTxRequestBody.intent.intentType.should.equal('payment');\n          createTxRequestBody.intent.senderAddress?.should.equal('senderAddr1');\n\n          return [\n            200,\n            {\n              apiVersion: 'lite',\n              unsignedTxs: [\n                {\n                  unsignedTx: {\n                    serializedTxHex: 'serializedTxHex',\n                    feeInfo: 'fee info',\n                  },\n                },\n              ],\n            },\n          ];\n        });\n\n      sinon.stub(adaWallet.baseCoin, 'verifyTransaction').resolves(true);\n      sinon.stub(adaWallet, 'signTransaction').resolves({ txRequestId: 'txRequestId' });\n      sinon.stub(BaseTssUtils.default.prototype, 'sendTxRequest').resolves('sendTxResponse');\n      await adaWallet.sendMany(sendManyParams).should.be.resolved();\n    });\n  });\n\n  describe('ERC20 Token Approval', function () {\n    let wallet;\n    const walletId = '5b34252f1bf349930e34020a00000000';\n    let topethCoin;\n    const tokenName = 'topeth:terc18dp';\n    const coin = 'topeth';\n\n    beforeEach(function () {\n      topethCoin = bitgo.coin('topeth');\n      wallet = new Wallet(bitgo, topethCoin, {\n        id: walletId,\n        coin: coin,\n        keys: ['keyid1', 'keyid2', 'keyid3'],\n      });\n    });\n\n    afterEach(function () {\n      sinon.restore();\n    });\n\n    it('should successfully approve a token', async function () {\n      const walletPassphrase = 'password123';\n      const expectedApprovalBuild = {\n        txHex: '0x123456',\n        feeInfo: {\n          fee: '1000000000',\n        },\n      };\n\n      const expectedSignedTx = {\n        txHex: '0x123456signed',\n      };\n\n      const expectedSendResult = {\n        txid: '0xabcdef',\n        tx: '0x123456signed',\n        status: 'signed',\n      };\n\n      // Mock the token approval build API\n      const ethUrl = common.Environments[bitgo.getEnv()].uri;\n      nock(ethUrl).post(`/api/v2/${coin}/wallet/${walletId}/token/approval/build`).reply(200, expectedApprovalBuild);\n\n      // Mock the getKeychainsAndValidatePassphrase method\n      sinon.stub(wallet, 'getKeychainsAndValidatePassphrase').resolves([\n        {\n          id: 'keyid1',\n          pub: 'pub1',\n          encryptedPrv: 'encryptedPrv',\n        },\n      ]);\n\n      // Mock the sign transaction method\n      sinon.stub(wallet, 'signTransaction').resolves(expectedSignedTx);\n\n      // Mock the send transaction method\n      sinon.stub(wallet, 'sendTransaction').resolves(expectedSendResult);\n\n      const result = await wallet.approveErc20Token(walletPassphrase, tokenName);\n\n      should.exist(result);\n      result.should.deepEqual(expectedSendResult);\n\n      // Verify the parameters passed to signTransaction\n      const signParams = wallet.signTransaction.firstCall.args[0];\n      signParams.should.have.property('txPrebuild', expectedApprovalBuild);\n      signParams.should.have.property('keychain');\n      signParams.should.have.property('walletPassphrase', walletPassphrase);\n    });\n\n    it('should handle token approval build API errors', async function () {\n      const walletPassphrase = 'password123';\n      const errorMessage = 'token approval build failed';\n\n      // Mock the token approval build API to return an error\n      nock(bgUrl).post(`/api/v2/${coin}/wallet/${walletId}/token/approval/build`).replyWithError(errorMessage);\n\n      await wallet.approveErc20Token(walletPassphrase, tokenName).should.be.rejectedWith(errorMessage);\n    });\n\n    it('should handle wallet passphrase validation errors', async function () {\n      const walletPassphrase = 'wrong-password';\n      const expectedApprovalBuild = {\n        txHex: '0x123456',\n        feeInfo: {\n          fee: '1000000000',\n        },\n      };\n\n      // Mock the token approval build API\n      nock(bgUrl).post(`/api/v2/${coin}/wallet/${walletId}/token/approval/build`).reply(200, expectedApprovalBuild);\n\n      // Mock the getKeychainsAndValidatePassphrase method to throw an error\n      const error = new Error('unable to decrypt keychain with the given wallet passphrase');\n      error.name = 'wallet_passphrase_incorrect';\n      sinon.stub(wallet, 'getKeychainsAndValidatePassphrase').rejects(error);\n\n      await wallet\n        .approveErc20Token(walletPassphrase, tokenName)\n        .should.be.rejectedWith('unable to decrypt keychain with the given wallet passphrase');\n    });\n\n    it('should handle signing errors', async function () {\n      const walletPassphrase = 'password123';\n      const expectedApprovalBuild = {\n        txHex: '0x123456',\n        feeInfo: {\n          fee: '1000000000',\n        },\n      };\n      const signingError = new Error('signing failed');\n\n      // Mock the token approval build API\n      nock(bgUrl).post(`/api/v2/${coin}/wallet/${walletId}/token/approval/build`).reply(200, expectedApprovalBuild);\n\n      // Mock the getKeychainsAndValidatePassphrase method\n      sinon.stub(wallet, 'getKeychainsAndValidatePassphrase').resolves([\n        {\n          id: 'keyid1',\n          pub: 'pub1',\n          encryptedPrv: 'encryptedPrv',\n        },\n      ]);\n\n      // Mock the sign transaction method to throw an error\n      sinon.stub(wallet, 'signTransaction').rejects(signingError);\n\n      await wallet.approveErc20Token(walletPassphrase, tokenName).should.be.rejectedWith(signingError);\n    });\n\n    it('should handle send transaction errors', async function () {\n      const walletPassphrase = 'password123';\n      const expectedApprovalBuild = {\n        txHex: '0x123456',\n        feeInfo: {\n          fee: '1000000000',\n        },\n      };\n      const expectedSignedTx = {\n        txHex: '0x123456signed',\n      };\n      const sendError = new Error('send failed');\n\n      // Mock the token approval build API\n      nock(bgUrl).post(`/api/v2/${coin}/wallet/${walletId}/token/approval/build`).reply(200, expectedApprovalBuild);\n\n      // Mock the getKeychainsAndValidatePassphrase method\n      sinon.stub(wallet, 'getKeychainsAndValidatePassphrase').resolves([\n        {\n          id: 'keyid1',\n          pub: 'pub1',\n          encryptedPrv: 'encryptedPrv',\n        },\n      ]);\n\n      // Mock the sign transaction method\n      sinon.stub(wallet, 'signTransaction').resolves(expectedSignedTx);\n\n      // Mock the send transaction method to throw an error\n      sinon.stub(wallet, 'sendTransaction').rejects(sendError);\n\n      await wallet.approveErc20Token(walletPassphrase, tokenName).should.be.rejectedWith(sendError);\n    });\n  });\n});\n"]}

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


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