PHP WebShell

Текущая директория: /opt/BitGoJS/modules/sdk-coin-eth/dist/test/unit

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

"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    var desc = Object.getOwnPropertyDescriptor(m, k);
    if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
      desc = { enumerable: true, get: function() { return m[k]; } };
    }
    Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
    Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
    o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
    var ownKeys = function(o) {
        ownKeys = Object.getOwnPropertyNames || function (o) {
            var ar = [];
            for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
            return ar;
        };
        return ownKeys(o);
    };
    return function (mod) {
        if (mod && mod.__esModule) return mod;
        var result = {};
        if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
        __setModuleDefault(result, mod);
        return result;
    };
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const sdk_test_1 = require("@bitgo/sdk-test");
const nock_1 = __importDefault(require("nock"));
const sinon_1 = __importDefault(require("sinon"));
const secp256k1_1 = require("@bitgo/secp256k1");
const secp256k1 = __importStar(require("secp256k1"));
const sdk_core_1 = require("@bitgo/sdk-core");
const sdk_api_1 = require("@bitgo/sdk-api");
const src_1 = require("../../src");
const assert_1 = __importDefault(require("assert"));
const getBuilder_1 = require("./getBuilder");
const testData = __importStar(require("../resources/eth"));
const mockData = __importStar(require("../fixtures/eth"));
const should_1 = __importDefault(require("should"));
nock_1.default.enableNetConnect();
describe('ETH:', function () {
    let bitgo;
    let hopTxBitgoSignature;
    let sandbox;
    const address1 = '0x174cfd823af8ce27ed0afee3fcf3c3ba259116be';
    const address2 = '0x7e85bdc27c050e3905ebf4b8e634d9ad6edd0de6';
    const hopContractAddress = '0x47ce7cc86efefef19f8fb516b11735d183da8635';
    const hopDestinationAddress = '0x9c7e8ce6825bD48278B3Ab59228EE26f8BE7925b';
    const hopTx = '0xf86b808504a817c8ff8252ff949c7e8ce6825bd48278b3ab59228ee26f8be7925b87038d7ea4c68000801ca011bc22c664570133dfca4f08a0b8d02339cf467046d6a4152f04f368d0eaf99ea01d6dc5cf0c897c8d4c3e1df53d0d042784c424536a4cc5b802552b7d64fee8b5';
    const hopTxid = '0x4af65143bc77da2b50f35b3d13cacb4db18f026bf84bc0743550bc57b9b53351';
    const userReqSig = '0x404db307f6147f0d8cd338c34c13906ef46a6faa7e0e119d5194ef05aec16e6f3d710f9b7901460f97e924066b62efd74443bd34402c6d40b49c203a559ff2c8';
    before(function () {
        const bitgoKeyXprv = 'xprv9s21ZrQH143K3tpWBHWe31sLoXNRQ9AvRYJgitkKxQ4ATFQMwvr7hHNqYRUnS7PsjzB7aK1VxqHLuNQjj1sckJ2Jwo2qxmsvejwECSpFMfC';
        const bitgoKey = secp256k1_1.bip32.fromBase58(bitgoKeyXprv);
        if (!bitgoKey.privateKey) {
            throw new Error('no privateKey');
        }
        const bitgoXpub = bitgoKey.neutered().toBase58();
        hopTxBitgoSignature =
            '0xaa' +
                Buffer.from(secp256k1.ecdsaSign(Buffer.from(hopTxid.slice(2), 'hex'), bitgoKey.privateKey).signature).toString('hex');
        const env = 'test';
        bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'test' });
        src_1.Erc20Token.createTokenConstructors().forEach(({ name, coinConstructor }) => {
            bitgo.safeRegister(name, coinConstructor);
        });
        bitgo.safeRegister('teth', src_1.Teth.createInstance);
        bitgo.safeRegister('hteth', src_1.Hteth.createInstance);
        sdk_core_1.common.Environments[env].hsmXpub = bitgoXpub;
        bitgo.initializeTestVars();
        sandbox = sinon_1.default.createSandbox();
    });
    after(function () {
        nock_1.default.cleanAll();
        sandbox.restore();
    });
    describe('EIP1559', function () {
        it('should sign a transaction with EIP1559 fee params', async function () {
            const coin = bitgo.coin('teth');
            const userKeychain = {
                prv: 'xprv9s21ZrQH143K3hekyNj7TciR4XNYe1kMj68W2ipjJGNHETWP7o42AjDnSPgKhdZ4x8NBAvaL72RrXjuXNdmkMqLERZza73oYugGtbLFXG8g',
                pub: 'xpub661MyMwAqRbcGBjE5QG7pkf9cZD33UUD6K46q7ELrbuG7FqXfLNGiXYGHeEnGBb5AWREnk1eA28g8ArZvURbhshXWkTtddHRo54fgyVvLdb',
                rawPub: '023636e68b7b204573abda2616aff6b584910dece2543f1cc6d842caac7d74974b',
                rawPrv: '7438a50010ce7b1dfd86e68046cc78ba1ebd242d6d85d9904d3fcc08734bc172',
            };
            const halfSignedTransaction = await coin.signTransaction({
                txPrebuild: {
                    eip1559: { maxPriorityFeePerGas: 10, maxFeePerGas: 10 },
                    isBatch: false,
                    recipients: [
                        {
                            amount: '42',
                            address: '0xc93b13642d93b4218bb85f67317d6b37286e8028',
                        },
                    ],
                    expireTime: 1627949214,
                    contractSequenceId: 12,
                    gasLimit: undefined,
                    gasPrice: undefined,
                    hopTransaction: undefined,
                    backupKeyNonce: undefined,
                    sequenceId: undefined,
                    nextContractSequenceId: 0,
                },
                prv: userKeychain.prv,
            });
            halfSignedTransaction.halfSigned.eip1559.maxPriorityFeePerGas.should.equal(10);
            halfSignedTransaction.halfSigned.eip1559.maxFeePerGas.should.equal(10);
        });
        it('should sign a transaction with EIP1559 fee params for CCR', async function () {
            const coin = bitgo.coin('hteth');
            const signTransaction = sinon_1.default.spy(src_1.AbstractEthLikeNewCoins.prototype, 'signTransaction');
            const userKeychain = {
                prv: 'xprv9s21ZrQH143K3hekyNj7TciR4XNYe1kMj68W2ipjJGNHETWP7o42AjDnSPgKhdZ4x8NBAvaL72RrXjuXNdmkMqLERZza73oYugGtbLFXG8g',
                pub: 'xpub661MyMwAqRbcGBjE5QG7pkf9cZD33UUD6K46q7ELrbuG7FqXfLNGiXYGHeEnGBb5AWREnk1eA28g8ArZvURbhshXWkTtddHRo54fgyVvLdb',
                rawPub: '023636e68b7b204573abda2616aff6b584910dece2543f1cc6d842caac7d74974b',
                rawPrv: '7438a50010ce7b1dfd86e68046cc78ba1ebd242d6d85d9904d3fcc08734bc172',
            };
            const txBuilder = (0, getBuilder_1.getBuilder)('hteth');
            txBuilder.type(sdk_core_1.TransactionType.Send);
            txBuilder.fee({
                fee: '10',
                gasLimit: '1000',
            });
            const key = testData.KEYPAIR_PRV.getKeys().prv;
            const transferBuilder = txBuilder.transfer();
            transferBuilder
                .amount('0')
                .to('0x19645032c7f1533395d44a629462e751084d3e4c')
                .expirationTime(1590066728)
                .contractSequenceId(5)
                .key(key);
            txBuilder.contract(address1);
            const tx = await txBuilder.build();
            const halfSignedTransaction = await coin.signTransaction({
                txPrebuild: {
                    eip1559: { maxPriorityFeePerGas: 10, maxFeePerGas: 10 },
                    isBatch: false,
                    recipients: [
                        {
                            amount: '42',
                            address: '0xc93b13642d93b4218bb85f67317d6b37286e8028',
                        },
                    ],
                    expireTime: 1627949214,
                    contractSequenceId: 12,
                    gasLimit: undefined,
                    gasPrice: undefined,
                    hopTransaction: undefined,
                    backupKeyNonce: undefined,
                    sequenceId: undefined,
                    nextContractSequenceId: 0,
                    txHex: tx.toBroadcastFormat(),
                },
                prv: userKeychain.prv,
                isEvmBasedCrossChainRecovery: true,
            });
            (0, assert_1.default)(halfSignedTransaction.halfSigned.txHex);
            assert_1.default.strictEqual(halfSignedTransaction.halfSigned.eip1559.maxFeePerGas, 10);
            sandbox.assert.calledOnce(signTransaction);
        });
    });
    describe('Transaction Verification', function () {
        it('should verify a normal txPrebuild from the bitgo server that matches the client txParams', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [{ amount: '1000000000000', address: address1 }],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
            };
            const txPrebuild = {
                recipients: [{ amount: '1000000000000', address: address1 }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: false,
                coin: 'teth',
                wallet: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            const isTransactionVerified = await coin.verifyTransaction({
                txParams,
                txPrebuild: txPrebuild,
                wallet,
                verification,
            });
            isTransactionVerified.should.equal(true);
        });
        it('should verify a batch txPrebuild from the bitgo server that matches the client txParams', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [
                    { amount: '1000000000000', address: address1 },
                    { amount: '2500000000000', address: address2 },
                ],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
            };
            const txPrebuild = {
                recipients: [
                    { amount: '3500000000000', address: (coin?.staticsCoin?.network).batcherContractAddress },
                ],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: true,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            const isTransactionVerified = await coin.verifyTransaction({
                txParams,
                txPrebuild: txPrebuild,
                wallet,
                verification,
            });
            isTransactionVerified.should.equal(true);
        });
        it('should verify ENS address resolution changing recipient address in client txParams', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [{ amount: '1000000000000', address: 'bitgotestwallet.eth' }],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
            };
            const txPrebuild = {
                recipients: [{ amount: '1000000000000', address: '0x40a663963810449d6e72533657a74f112c3b901a' }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: false,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            const isTransactionVerified = await coin.verifyTransaction({
                txParams,
                txPrebuild: txPrebuild,
                wallet,
                verification,
            });
            isTransactionVerified.should.equal(true);
        });
        it('should verify a hop txPrebuild from the bitgo server that matches the client txParams', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [{ amount: 1000000000000000, address: hopDestinationAddress }],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
                hop: true,
            };
            const txPrebuild = {
                recipients: [{ amount: '5000000000000000', address: hopContractAddress }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: false,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
                hopTransaction: {
                    tx: hopTx,
                    id: hopTxid,
                    signature: hopTxBitgoSignature,
                    paymentId: '2773928196',
                    gasPrice: 20000000000,
                    gasLimit: 500000,
                    amount: '1000000000000000',
                    recipient: hopDestinationAddress,
                    nonce: 0,
                    userReqSig: userReqSig,
                    gasPriceMax: 500000000000,
                },
            };
            const verification = {};
            const isTransactionVerified = await coin.verifyTransaction({
                txParams,
                txPrebuild: txPrebuild,
                wallet,
                verification,
            });
            isTransactionVerified.should.equal(true);
        });
        it('should reject when client txParams are missing', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = null;
            const txPrebuild = {
                recipients: [{ amount: '1000000000000', address: address1 }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: false,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            await coin
                .verifyTransaction({ txParams: txParams, txPrebuild: txPrebuild, wallet, verification })
                .should.be.rejectedWith('missing params');
        });
        it('should reject txPrebuild that is both batch and hop', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [
                    { amount: '1000000000000', address: address1 },
                    { amount: '2500000000000', address: address2 },
                ],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
                hop: true,
            };
            const txPrebuild = {
                recipients: [{ amount: '3500000000000', address: address1 }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: true,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
                hopTransaction: {
                    tx: hopTx,
                    id: hopTxid,
                    signature: hopTxBitgoSignature,
                    paymentId: '2773928196',
                    gasPrice: 20000000000,
                    gasLimit: 500000,
                    amount: '1000000000000000',
                    recipient: hopDestinationAddress,
                    nonce: 0,
                    userReqSig: userReqSig,
                    gasPriceMax: 500000000000,
                },
            };
            const verification = {};
            await coin
                .verifyTransaction({ txParams, txPrebuild: txPrebuild, wallet, verification })
                .should.be.rejectedWith('tx cannot be both a batch and hop transaction');
        });
        it('should reject a txPrebuild with more than one recipient', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [
                    { amount: '1000000000000', address: address1 },
                    { amount: '2500000000000', address: address2 },
                ],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
            };
            const txPrebuild = {
                recipients: [
                    { amount: '1000000000000', address: address1 },
                    { amount: '2500000000000', address: address2 },
                ],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: true,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            await coin
                .verifyTransaction({ txParams, txPrebuild: txPrebuild, wallet, verification })
                .should.be.rejectedWith(`teth doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`);
        });
        it('should reject a hop txPrebuild that does not send to its hop address', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [{ amount: '1000000000000000', address: hopDestinationAddress }],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
                hop: true,
            };
            const txPrebuild = {
                recipients: [{ amount: '5000000000000000', address: address1 }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: false,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
                hopTransaction: {
                    tx: hopTx,
                    id: hopTxid,
                    signature: hopTxBitgoSignature,
                    paymentId: '0',
                    gasPrice: 20000000000,
                    gasLimit: 500000,
                    amount: '1000000000000000',
                    recipient: hopDestinationAddress,
                    nonce: 0,
                    userReqSig: userReqSig,
                    gasPriceMax: 500000000000,
                },
            };
            const verification = {};
            await coin
                .verifyTransaction({ txParams, txPrebuild: txPrebuild, wallet, verification })
                .should.be.rejectedWith('recipient address of txPrebuild does not match hop address');
        });
        it('should reject a batch txPrebuild from the bitgo server with the wrong total amount', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [
                    { amount: '1000000000000', address: address1 },
                    { amount: '2500000000000', address: address2 },
                ],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
            };
            const txPrebuild = {
                recipients: [
                    { amount: '5500000000000', address: (coin?.staticsCoin?.network).batcherContractAddress },
                ],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: true,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            await coin
                .verifyTransaction({ txParams, txPrebuild: txPrebuild, wallet, verification })
                .should.be.rejectedWith('batch transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client');
        });
        it('should reject a batch txPrebuild from the bitgo server that does not send to the batcher contract address', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [
                    { amount: '1000000000000', address: address1 },
                    { amount: '2500000000000', address: address2 },
                ],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
            };
            const txPrebuild = {
                recipients: [{ amount: '3500000000000', address: hopContractAddress }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: true,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            await coin
                .verifyTransaction({ txParams, txPrebuild: txPrebuild, wallet, verification })
                .should.be.rejectedWith('recipient address of txPrebuild does not match batcher address');
        });
        it('should reject a normal txPrebuild from the bitgo server with the wrong amount', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [{ amount: '1000000000000', address: address1 }],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
            };
            const txPrebuild = {
                recipients: [{ amount: '2000000000000', address: address1 }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: false,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            await coin
                .verifyTransaction({ txParams, txPrebuild: txPrebuild, wallet, verification })
                .should.be.rejectedWith('normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client');
        });
        it('should reject a normal txPrebuild from the bitgo server with the wrong recipient', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [{ amount: '1000000000000', address: address1 }],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
            };
            const txPrebuild = {
                recipients: [{ amount: '1000000000000', address: address2 }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: false,
                coin: 'teth',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            await coin
                .verifyTransaction({ txParams, txPrebuild: txPrebuild, wallet, verification })
                .should.be.rejectedWith('destination address in normal txPrebuild does not match that in txParams supplied by client');
        });
        it('should verify a token txPrebuild from the bitgo server that matches the client txParams', async function () {
            const coin = bitgo.coin('test');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [{ amount: '1000000000000', address: address1 }],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
            };
            const txPrebuild = {
                recipients: [{ amount: '1000000000000', address: address1 }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: false,
                coin: 'teth',
                token: 'test',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            const isTransactionVerified = await coin.verifyTransaction({
                txParams,
                txPrebuild: txPrebuild,
                wallet,
                verification,
            });
            isTransactionVerified.should.equal(true);
        });
        it('should reject a txPrebuild from the bitgo server with the wrong coin', async function () {
            const coin = bitgo.coin('teth');
            const wallet = new sdk_core_1.Wallet(bitgo, coin, {});
            const txParams = {
                recipients: [{ amount: '1000000000000', address: address1 }],
                wallet: wallet,
                walletPassphrase: 'fakeWalletPassphrase',
            };
            const txPrebuild = {
                recipients: [{ amount: '1000000000000', address: address1 }],
                nextContractSequenceId: 0,
                gasPrice: 20000000000,
                gasLimit: 500000,
                isBatch: false,
                coin: 'btc',
                walletId: 'fakeWalletId',
                walletContractAddress: 'fakeWalletContractAddress',
            };
            const verification = {};
            await coin
                .verifyTransaction({ txParams, txPrebuild: txPrebuild, wallet, verification })
                .should.be.rejectedWith('coin in txPrebuild did not match that in txParams supplied by client');
        });
    });
    describe('Address Verification', function () {
        it('should verify an address generated using forwarder version 0', async function () {
            const coin = bitgo.coin('teth');
            const params = {
                id: '6127bff4ecd84c0006cd9a0e5ccdc36f',
                chain: 0,
                index: 3174,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-08-26T16:23:16.563Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    pendingDeployment: false,
                    forwarderVersion: 0,
                },
            };
            const isAddressVerified = await coin.verifyAddress(params);
            isAddressVerified.should.equal(true);
        });
        it('should verify an address generated using forwarder version 1', async function () {
            const coin = bitgo.coin('teth');
            const params = {
                id: '61250217c8c02b000654b15e7af6f618',
                address: '0xb0b56eeae1b283918caca02a14ada2df17a98e6d',
                chain: 0,
                index: 3162,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-08-24T14:28:39.841Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0xc5a',
                    pendingDeployment: true,
                    forwarderVersion: 1,
                },
            };
            const isAddressVerified = await coin.verifyAddress(params);
            isAddressVerified.should.equal(true);
        });
        it('should reject address verification if coinSpecific field is not an object', async function () {
            const coin = bitgo.coin('teth');
            const params = {
                id: '61250217c8c02b000654b15e7af6f618',
                address: '0xb0b56eeae1b283918caca02a14ada2df17a98e6d',
                chain: 0,
                index: 3162,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
            };
            assert_1.default.rejects(async () => coin.verifyAddress(params), sdk_core_1.InvalidAddressVerificationObjectPropertyError);
        });
        it('should reject address verification when an actual address is different from expected address', async function () {
            const coin = bitgo.coin('teth');
            const params = {
                id: '61250217c8c02b000654b15e7af6f618',
                address: '0x28904591f735994f050804fda3b61b813b16e04c',
                chain: 0,
                index: 3162,
                coin: 'teth',
                lastNonce: 0,
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-08-24T14:28:39.841Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0xc5a',
                    pendingDeployment: true,
                    forwarderVersion: 1,
                },
            };
            assert_1.default.rejects(async () => coin.verifyAddress(params), sdk_core_1.UnexpectedAddressError);
        });
        it('should reject address verification if the derived address is in invalid format', async function () {
            const coin = bitgo.coin('teth');
            const params = {
                id: '61250217c8c02b000654b15e7af6f618',
                address: '0xe0b56eeae1b283918caca02a14ada2df17a98bvf',
                chain: 0,
                index: 3162,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-08-24T14:28:39.841Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0xc5a',
                    pendingDeployment: true,
                    forwarderVersion: 1,
                },
            };
            assert_1.default.rejects(async () => coin.verifyAddress(params), sdk_core_1.InvalidAddressError);
        });
        it('should reject address verification if base address is undefined', async function () {
            const coin = bitgo.coin('teth');
            const params = {
                id: '61250217c8c02b000654b15e7af6f618',
                address: '0xb0b56eeae1b283918caca02a14ada2df17a98e6d',
                chain: 0,
                index: 3162,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-08-24T14:28:39.841Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0xc5a',
                    pendingDeployment: true,
                    forwarderVersion: 1,
                },
            };
            assert_1.default.rejects(async () => coin.verifyAddress(params), sdk_core_1.InvalidAddressError);
        });
        it('should reject address verification if base address is in invalid format', async function () {
            const coin = bitgo.coin('teth');
            const params = {
                id: '61250217c8c02b000654b15e7af6f618',
                address: '0xb0b56eeae1b283918caca02a14ada2df17a98e6d',
                chain: 0,
                index: 3162,
                coin: 'teth',
                lastNonce: 0,
                wallet: '598f606cd8fc24710d2ebadb1d9459bb',
                baseAddress: '0xe0b56eeae1b283918caca02a14ada2df17a98bvf',
                coinSpecific: {
                    nonce: -1,
                    updateTime: '2021-08-24T14:28:39.841Z',
                    txCount: 0,
                    pendingChainInitialization: true,
                    creationFailure: [],
                    salt: '0xc5a',
                    pendingDeployment: true,
                    forwarderVersion: 1,
                },
            };
            assert_1.default.rejects(async () => coin.verifyAddress(params), sdk_core_1.InvalidAddressError);
        });
    });
    describe('EVM Cross Chain Recovery', function () {
        const baseUrl = 'https://api-holesky.etherscan.io';
        it('should build a recovery transaction for hot wallet', async function () {
            const userKey = '{"iv":"VFZ3jvXhxo1Z+Yaf2MtZnA==","v":1,"iter":10000,"ks":256,"ts":64,"mode"\n' +
                ':"ccm","adata":"","cipher":"aes","salt":"p+fkHuLa/8k=","ct":"hYG7pvljLIgCjZ\n' +
                '53PBlCde5KZRmlUKKHLtDMk+HJfuU46hW+x+C9WsIAO4gFPnTCvFVmQ8x7czCtcNFub5AO2otOG\n' +
                'OsX4GE2gXOEmCl1TpWwwNhm7yMUjGJUpgW6ZZgXSXdDitSKi4V/hk78SGSzjFOBSPYRa6I="}\n';
            const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
            const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';
            const destinationAddress = '0xd5ADdE17feD8baed3F32b84AF05B8F2816f7b560';
            const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';
            const walletPassphrase = sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE;
            const basecoin = bitgo.coin('hteth');
            (0, nock_1.default)(baseUrl)
                .get('/api')
                .query(mockData.getTxListRequest(bitgoFeeAddress))
                .reply(200, mockData.getTxListResponse);
            (0, nock_1.default)(baseUrl)
                .get('/api')
                .query(mockData.getBalanceRequest(bitgoFeeAddress))
                .reply(200, mockData.getBalanceResponse);
            (0, nock_1.default)(baseUrl)
                .get('/api')
                .query(mockData.getBalanceRequest(walletContractAddress))
                .reply(200, mockData.getBalanceResponse);
            (0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
            const spy = sinon_1.default.spy(src_1.TransactionBuilder.prototype, 'coinUsesNonPackedEncodingForTxData');
            await basecoin.recover({
                userKey: userKey,
                backupKey: '',
                walletPassphrase: walletPassphrase,
                walletContractAddress: walletContractAddress,
                bitgoFeeAddress: bitgoFeeAddress,
                recoveryDestination: destinationAddress,
                eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
                gasLimit: 500000,
                bitgoDestinationAddress: bitgoDestinationAddress,
                intendedChain: 'tarbeth',
            });
            (0, assert_1.default)(spy.returned(true));
        });
        describe('Non-BitGo Recovery for Hot Wallets (MPCv2)', function () {
            const baseUrl = 'https://api-holesky.etherscan.io';
            let bitgo;
            let basecoin;
            before(function () {
                bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'test' });
                basecoin = bitgo.coin('hteth');
            });
            it('should build a recovery transaction for MPCv2 kind of hot wallets', async function () {
                (0, nock_1.default)(baseUrl)
                    .get('/api')
                    .query(mockData.getTxListRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().bitgoFeeAddress))
                    .reply(200, mockData.getTxListResponse);
                (0, nock_1.default)(baseUrl)
                    .get('/api')
                    .query(mockData.getBalanceRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().bitgoFeeAddress))
                    .reply(200, mockData.getBalanceResponse);
                (0, nock_1.default)(baseUrl)
                    .get('/api')
                    .query(mockData.getBalanceRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().walletContractAddress))
                    .reply(200, mockData.getBalanceResponse);
                (0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
                const params = mockData.getNonBitGoRecoveryForHotWalletsMPCv2();
                const transaction = await basecoin.recover({
                    userKey: params.userKey,
                    backupKey: params.backupKey,
                    walletPassphrase: params.walletPassphrase,
                    walletContractAddress: params.walletContractAddress,
                    bitgoFeeAddress: params.bitgoFeeAddress,
                    recoveryDestination: params.recoveryDestination,
                    eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
                    gasLimit: 500000,
                    bitgoDestinationAddress: params.bitgoDestinationAddress,
                    intendedChain: params.intendedChain,
                });
                should_1.default.exist(transaction);
                transaction.should.have.property('txHex');
            });
            it('should throw an error for invalid user key', async function () {
                const params = mockData.getInvalidNonBitGoRecoveryParams();
                await assert_1.default.rejects(async () => {
                    await basecoin.recover({
                        userKey: params.userKey,
                        backupKey: params.backupKey,
                        walletPassphrase: params.walletPassphrase,
                        walletContractAddress: params.walletContractAddress,
                        bitgoFeeAddress: params.bitgoFeeAddress,
                        recoveryDestination: params.recoveryDestination,
                        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
                        gasLimit: 500000,
                        bitgoDestinationAddress: params.bitgoDestinationAddress,
                        intendedChain: params.intendedChain,
                    });
                }, Error, 'user key is invalid');
            });
        });
        describe('Build Unsigned Sweep for Self-Custody Cold Wallets (MPCv2)', function () {
            const baseUrl = 'https://api-holesky.etherscan.io';
            let bitgo;
            let basecoin;
            before(function () {
                bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'test' });
                basecoin = bitgo.coin('hteth');
            });
            it('should generate an unsigned sweep without derivation seed', async function () {
                (0, nock_1.default)(baseUrl)
                    .get('/api')
                    .query(mockData.getTxListRequest(mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().address))
                    .reply(200, mockData.getTxListResponse);
                (0, nock_1.default)(baseUrl)
                    .get('/api')
                    .query(mockData.getBalanceRequest(mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().address))
                    .reply(200, mockData.getBalanceResponse);
                (0, nock_1.default)(baseUrl)
                    .get('/api')
                    .query(mockData.getBalanceRequest(mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().walletContractAddress))
                    .reply(200, mockData.getBalanceResponse);
                (0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
                const params = mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2();
                const sweepResult = await basecoin.recover({
                    userKey: params.commonKeyChain, // Box A Data
                    backupKey: params.commonKeyChain, // Box B Data
                    derivationSeed: params.derivationSeed, // Key Derivation Seed (optional)
                    recoveryDestination: params.recoveryDestination, // Destination Address
                    gasLimit: 200000, // Gas Limit
                    eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 }, // Max Fee Per Gas and Max Priority Fee Per Gas
                    walletContractAddress: params.walletContractAddress,
                    isTss: true,
                    replayProtectionOptions: {
                        chain: '42',
                        hardfork: 'london',
                    },
                });
                should_1.default.exist(sweepResult);
                const output = sweepResult;
                output.should.have.property('txRequests');
                output.txRequests.should.have.length(1);
                output.txRequests[0].should.have.property('transactions');
                output.txRequests[0].transactions.should.have.length(1);
                output.txRequests[0].should.have.property('walletCoin');
                output.txRequests[0].transactions.should.have.length(1);
                output.txRequests[0].transactions[0].should.have.property('unsignedTx');
                output.txRequests[0].transactions[0].unsignedTx.should.have.property('serializedTxHex');
                output.txRequests[0].transactions[0].unsignedTx.should.have.property('signableHex');
                output.txRequests[0].transactions[0].unsignedTx.should.have.property('derivationPath');
                output.txRequests[0].transactions[0].unsignedTx.should.have.property('feeInfo');
                output.txRequests[0].transactions[0].unsignedTx.should.have.property('parsedTx');
                const parsedTx = output.txRequests[0].transactions[0].unsignedTx.parsedTx;
                parsedTx.should.have.property('spendAmount');
                output.txRequests[0].transactions[0].unsignedTx.parsedTx.should.have.property('outputs');
            });
            it('should throw an error for invalid address', async function () {
                const params = mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2();
                params.recoveryDestination = 'invalidAddress';
                // Ensure userKey and backupKey are the same
                params.userKey =
                    '0234eb39b22fed523ece7c78da29ba1f1de5b64a6e48013e0914de793bc1df0570e779de04758732734d97e54b782c8b336283811af6a2c57bd81438798e1c2446';
                params.backupKey =
                    '0234eb39b22fed523ece7c78da29ba1f1de5b64a6e48013e0914de793bc1df0570e779de04758732734d97e54b782c8b336283811af6a2c57bd81438798e1c2446';
                await assert_1.default.rejects(async () => {
                    await basecoin.recover({
                        recoveryDestination: params.recoveryDestination, // Destination Address
                        gasLimit: 2000, // Gas Limit
                        eip1559: { maxFeePerGas: 200, maxPriorityFeePerGas: 10000 }, // Max Fee Per Gas and Max Priority Fee Per Gas
                        userKey: params.userKey, // Provide the userKey
                        backupKey: params.backupKey, // Provide the backupKey
                        walletContractAddress: params.walletContractAddress, // Provide the walletContractAddress
                        isTss: true,
                        replayProtectionOptions: {
                            chain: '42',
                            hardfork: 'london',
                        },
                    });
                }, Error, 'Error: invalid address');
            });
        });
    });
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"eth.js","sourceRoot":"","sources":["../../../test/unit/eth.ts"],"names":[],"mappings":"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","sourcesContent":["import { TestBitGo, TestBitGoAPI } from '@bitgo/sdk-test';\n\nimport nock from 'nock';\nimport sinon from 'sinon';\nimport { bip32 } from '@bitgo/secp256k1';\nimport * as secp256k1 from 'secp256k1';\nimport {\n  common,\n  InvalidAddressError,\n  InvalidAddressVerificationObjectPropertyError,\n  TransactionType,\n  UnexpectedAddressError,\n  Wallet,\n} from '@bitgo/sdk-core';\nimport { BitGoAPI } from '@bitgo/sdk-api';\nimport {\n  AbstractEthLikeNewCoins,\n  Erc20Token,\n  Hteth,\n  Teth,\n  TransactionBuilder,\n  TransferBuilder,\n  UnsignedSweepTxMPCv2,\n} from '../../src';\nimport { EthereumNetwork } from '@bitgo/statics';\nimport assert from 'assert';\nimport { getBuilder } from './getBuilder';\nimport * as testData from '../resources/eth';\nimport * as mockData from '../fixtures/eth';\nimport should from 'should';\n\nnock.enableNetConnect();\n\ndescribe('ETH:', function () {\n  let bitgo: TestBitGoAPI;\n  let hopTxBitgoSignature;\n  let sandbox: sinon.SinonSandbox;\n\n  const address1 = '0x174cfd823af8ce27ed0afee3fcf3c3ba259116be';\n  const address2 = '0x7e85bdc27c050e3905ebf4b8e634d9ad6edd0de6';\n  const hopContractAddress = '0x47ce7cc86efefef19f8fb516b11735d183da8635';\n  const hopDestinationAddress = '0x9c7e8ce6825bD48278B3Ab59228EE26f8BE7925b';\n  const hopTx =\n    '0xf86b808504a817c8ff8252ff949c7e8ce6825bd48278b3ab59228ee26f8be7925b87038d7ea4c68000801ca011bc22c664570133dfca4f08a0b8d02339cf467046d6a4152f04f368d0eaf99ea01d6dc5cf0c897c8d4c3e1df53d0d042784c424536a4cc5b802552b7d64fee8b5';\n  const hopTxid = '0x4af65143bc77da2b50f35b3d13cacb4db18f026bf84bc0743550bc57b9b53351';\n  const userReqSig =\n    '0x404db307f6147f0d8cd338c34c13906ef46a6faa7e0e119d5194ef05aec16e6f3d710f9b7901460f97e924066b62efd74443bd34402c6d40b49c203a559ff2c8';\n\n  before(function () {\n    const bitgoKeyXprv =\n      'xprv9s21ZrQH143K3tpWBHWe31sLoXNRQ9AvRYJgitkKxQ4ATFQMwvr7hHNqYRUnS7PsjzB7aK1VxqHLuNQjj1sckJ2Jwo2qxmsvejwECSpFMfC';\n    const bitgoKey = bip32.fromBase58(bitgoKeyXprv);\n    if (!bitgoKey.privateKey) {\n      throw new Error('no privateKey');\n    }\n    const bitgoXpub = bitgoKey.neutered().toBase58();\n    hopTxBitgoSignature =\n      '0xaa' +\n      Buffer.from(secp256k1.ecdsaSign(Buffer.from(hopTxid.slice(2), 'hex'), bitgoKey.privateKey).signature).toString(\n        'hex'\n      );\n\n    const env = 'test';\n    bitgo = TestBitGo.decorate(BitGoAPI, { env: 'test' });\n    Erc20Token.createTokenConstructors().forEach(({ name, coinConstructor }) => {\n      bitgo.safeRegister(name, coinConstructor);\n    });\n    bitgo.safeRegister('teth', Teth.createInstance);\n    bitgo.safeRegister('hteth', Hteth.createInstance);\n    common.Environments[env].hsmXpub = bitgoXpub;\n    bitgo.initializeTestVars();\n    sandbox = sinon.createSandbox();\n  });\n\n  after(function () {\n    nock.cleanAll();\n    sandbox.restore();\n  });\n\n  describe('EIP1559', function () {\n    it('should sign a transaction with EIP1559 fee params', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n\n      const userKeychain = {\n        prv: 'xprv9s21ZrQH143K3hekyNj7TciR4XNYe1kMj68W2ipjJGNHETWP7o42AjDnSPgKhdZ4x8NBAvaL72RrXjuXNdmkMqLERZza73oYugGtbLFXG8g',\n        pub: 'xpub661MyMwAqRbcGBjE5QG7pkf9cZD33UUD6K46q7ELrbuG7FqXfLNGiXYGHeEnGBb5AWREnk1eA28g8ArZvURbhshXWkTtddHRo54fgyVvLdb',\n        rawPub: '023636e68b7b204573abda2616aff6b584910dece2543f1cc6d842caac7d74974b',\n        rawPrv: '7438a50010ce7b1dfd86e68046cc78ba1ebd242d6d85d9904d3fcc08734bc172',\n      };\n\n      const halfSignedTransaction = await coin.signTransaction({\n        txPrebuild: {\n          eip1559: { maxPriorityFeePerGas: 10, maxFeePerGas: 10 },\n          isBatch: false,\n          recipients: [\n            {\n              amount: '42',\n              address: '0xc93b13642d93b4218bb85f67317d6b37286e8028',\n            },\n          ],\n          expireTime: 1627949214,\n          contractSequenceId: 12,\n          gasLimit: undefined,\n          gasPrice: undefined,\n          hopTransaction: undefined,\n          backupKeyNonce: undefined,\n          sequenceId: undefined,\n          nextContractSequenceId: 0,\n        },\n        prv: userKeychain.prv,\n      } as any);\n\n      (halfSignedTransaction as any).halfSigned.eip1559.maxPriorityFeePerGas.should.equal(10);\n      (halfSignedTransaction as any).halfSigned.eip1559.maxFeePerGas.should.equal(10);\n    });\n\n    it('should sign a transaction with EIP1559 fee params for CCR', async function () {\n      const coin = bitgo.coin('hteth') as Hteth;\n      const signTransaction = sinon.spy(AbstractEthLikeNewCoins.prototype, 'signTransaction');\n\n      const userKeychain = {\n        prv: 'xprv9s21ZrQH143K3hekyNj7TciR4XNYe1kMj68W2ipjJGNHETWP7o42AjDnSPgKhdZ4x8NBAvaL72RrXjuXNdmkMqLERZza73oYugGtbLFXG8g',\n        pub: 'xpub661MyMwAqRbcGBjE5QG7pkf9cZD33UUD6K46q7ELrbuG7FqXfLNGiXYGHeEnGBb5AWREnk1eA28g8ArZvURbhshXWkTtddHRo54fgyVvLdb',\n        rawPub: '023636e68b7b204573abda2616aff6b584910dece2543f1cc6d842caac7d74974b',\n        rawPrv: '7438a50010ce7b1dfd86e68046cc78ba1ebd242d6d85d9904d3fcc08734bc172',\n      };\n      const txBuilder = getBuilder('hteth') as TransactionBuilder;\n      txBuilder.type(TransactionType.Send);\n      txBuilder.fee({\n        fee: '10',\n        gasLimit: '1000',\n      });\n      const key = testData.KEYPAIR_PRV.getKeys().prv as string;\n      const transferBuilder = txBuilder.transfer() as TransferBuilder;\n      transferBuilder\n        .amount('0')\n        .to('0x19645032c7f1533395d44a629462e751084d3e4c')\n        .expirationTime(1590066728)\n        .contractSequenceId(5)\n        .key(key);\n      txBuilder.contract(address1);\n      const tx = await txBuilder.build();\n\n      const halfSignedTransaction = await coin.signTransaction({\n        txPrebuild: {\n          eip1559: { maxPriorityFeePerGas: 10, maxFeePerGas: 10 },\n          isBatch: false,\n          recipients: [\n            {\n              amount: '42',\n              address: '0xc93b13642d93b4218bb85f67317d6b37286e8028',\n            },\n          ],\n          expireTime: 1627949214,\n          contractSequenceId: 12,\n          gasLimit: undefined,\n          gasPrice: undefined,\n          hopTransaction: undefined,\n          backupKeyNonce: undefined,\n          sequenceId: undefined,\n          nextContractSequenceId: 0,\n          txHex: tx.toBroadcastFormat(),\n        },\n        prv: userKeychain.prv,\n        isEvmBasedCrossChainRecovery: true,\n      } as any);\n\n      assert((halfSignedTransaction as any).halfSigned.txHex);\n      assert.strictEqual((halfSignedTransaction as any).halfSigned.eip1559.maxFeePerGas, 10);\n\n      sandbox.assert.calledOnce(signTransaction);\n    });\n  });\n\n  describe('Transaction Verification', function () {\n    it('should verify a normal txPrebuild from the bitgo server that matches the client txParams', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'teth',\n        wallet: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      const isTransactionVerified = await coin.verifyTransaction({\n        txParams,\n        txPrebuild: txPrebuild as any,\n        wallet,\n        verification,\n      });\n      isTransactionVerified.should.equal(true);\n    });\n\n    it('should verify a batch txPrebuild from the bitgo server that matches the client txParams', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [\n          { amount: '1000000000000', address: address1 },\n          { amount: '2500000000000', address: address2 },\n        ],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [\n          { amount: '3500000000000', address: (coin?.staticsCoin?.network as EthereumNetwork).batcherContractAddress },\n        ],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: true,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      const isTransactionVerified = await coin.verifyTransaction({\n        txParams,\n        txPrebuild: txPrebuild as any,\n        wallet,\n        verification,\n      });\n      isTransactionVerified.should.equal(true);\n    });\n\n    it('should verify ENS address resolution changing recipient address in client txParams', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000', address: 'bitgotestwallet.eth' }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '1000000000000', address: '0x40a663963810449d6e72533657a74f112c3b901a' }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      const isTransactionVerified = await coin.verifyTransaction({\n        txParams,\n        txPrebuild: txPrebuild as any,\n        wallet,\n        verification,\n      });\n      isTransactionVerified.should.equal(true);\n    });\n\n    it('should verify a hop txPrebuild from the bitgo server that matches the client txParams', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [{ amount: 1000000000000000, address: hopDestinationAddress }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n        hop: true,\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '5000000000000000', address: hopContractAddress }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n        hopTransaction: {\n          tx: hopTx,\n          id: hopTxid,\n          signature: hopTxBitgoSignature,\n          paymentId: '2773928196',\n          gasPrice: 20000000000,\n          gasLimit: 500000,\n          amount: '1000000000000000',\n          recipient: hopDestinationAddress,\n          nonce: 0,\n          userReqSig: userReqSig,\n          gasPriceMax: 500000000000,\n        },\n      };\n\n      const verification = {};\n\n      const isTransactionVerified = await coin.verifyTransaction({\n        txParams,\n        txPrebuild: txPrebuild as any,\n        wallet,\n        verification,\n      });\n      isTransactionVerified.should.equal(true);\n    });\n\n    it('should reject when client txParams are missing', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = null;\n\n      const txPrebuild = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await coin\n        .verifyTransaction({ txParams: txParams as any, txPrebuild: txPrebuild as any, wallet, verification })\n        .should.be.rejectedWith('missing params');\n    });\n\n    it('should reject txPrebuild that is both batch and hop', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [\n          { amount: '1000000000000', address: address1 },\n          { amount: '2500000000000', address: address2 },\n        ],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n        hop: true,\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '3500000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: true,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n        hopTransaction: {\n          tx: hopTx,\n          id: hopTxid,\n          signature: hopTxBitgoSignature,\n          paymentId: '2773928196',\n          gasPrice: 20000000000,\n          gasLimit: 500000,\n          amount: '1000000000000000',\n          recipient: hopDestinationAddress,\n          nonce: 0,\n          userReqSig: userReqSig,\n          gasPriceMax: 500000000000,\n        },\n      };\n\n      const verification = {};\n\n      await coin\n        .verifyTransaction({ txParams, txPrebuild: txPrebuild as any, wallet, verification })\n        .should.be.rejectedWith('tx cannot be both a batch and hop transaction');\n    });\n\n    it('should reject a txPrebuild with more than one recipient', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [\n          { amount: '1000000000000', address: address1 },\n          { amount: '2500000000000', address: address2 },\n        ],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [\n          { amount: '1000000000000', address: address1 },\n          { amount: '2500000000000', address: address2 },\n        ],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: true,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await coin\n        .verifyTransaction({ txParams, txPrebuild: txPrebuild as any, wallet, verification })\n        .should.be.rejectedWith(\n          `teth doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`\n        );\n    });\n\n    it('should reject a hop txPrebuild that does not send to its hop address', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000000', address: hopDestinationAddress }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n        hop: true,\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '5000000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n        hopTransaction: {\n          tx: hopTx,\n          id: hopTxid,\n          signature: hopTxBitgoSignature,\n          paymentId: '0',\n          gasPrice: 20000000000,\n          gasLimit: 500000,\n          amount: '1000000000000000',\n          recipient: hopDestinationAddress,\n          nonce: 0,\n          userReqSig: userReqSig,\n          gasPriceMax: 500000000000,\n        },\n      };\n\n      const verification = {};\n\n      await coin\n        .verifyTransaction({ txParams, txPrebuild: txPrebuild as any, wallet, verification })\n        .should.be.rejectedWith('recipient address of txPrebuild does not match hop address');\n    });\n\n    it('should reject a batch txPrebuild from the bitgo server with the wrong total amount', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [\n          { amount: '1000000000000', address: address1 },\n          { amount: '2500000000000', address: address2 },\n        ],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [\n          { amount: '5500000000000', address: (coin?.staticsCoin?.network as EthereumNetwork).batcherContractAddress },\n        ],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: true,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await coin\n        .verifyTransaction({ txParams, txPrebuild: txPrebuild as any, wallet, verification })\n        .should.be.rejectedWith(\n          'batch transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client'\n        );\n    });\n\n    it('should reject a batch txPrebuild from the bitgo server that does not send to the batcher contract address', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [\n          { amount: '1000000000000', address: address1 },\n          { amount: '2500000000000', address: address2 },\n        ],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '3500000000000', address: hopContractAddress }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: true,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await coin\n        .verifyTransaction({ txParams, txPrebuild: txPrebuild as any, wallet, verification })\n        .should.be.rejectedWith('recipient address of txPrebuild does not match batcher address');\n    });\n\n    it('should reject a normal txPrebuild from the bitgo server with the wrong amount', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '2000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await coin\n        .verifyTransaction({ txParams, txPrebuild: txPrebuild as any, wallet, verification })\n        .should.be.rejectedWith(\n          'normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client'\n        );\n    });\n\n    it('should reject a normal txPrebuild from the bitgo server with the wrong recipient', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '1000000000000', address: address2 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'teth',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await coin\n        .verifyTransaction({ txParams, txPrebuild: txPrebuild as any, wallet, verification })\n        .should.be.rejectedWith(\n          'destination address in normal txPrebuild does not match that in txParams supplied by client'\n        );\n    });\n\n    it('should verify a token txPrebuild from the bitgo server that matches the client txParams', async function () {\n      const coin = bitgo.coin('test');\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'teth',\n        token: 'test',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      const isTransactionVerified = await coin.verifyTransaction({\n        txParams,\n        txPrebuild: txPrebuild as any,\n        wallet,\n        verification,\n      });\n      isTransactionVerified.should.equal(true);\n    });\n\n    it('should reject a txPrebuild from the bitgo server with the wrong coin', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n      const wallet = new Wallet(bitgo, coin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'btc',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await coin\n        .verifyTransaction({ txParams, txPrebuild: txPrebuild as any, wallet, verification })\n        .should.be.rejectedWith('coin in txPrebuild did not match that in txParams supplied by client');\n    });\n  });\n\n  describe('Address Verification', function () {\n    it('should verify an address generated using forwarder version 0', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n\n      const params = {\n        id: '6127bff4ecd84c0006cd9a0e5ccdc36f',\n        chain: 0,\n        index: 3174,\n        coin: 'teth',\n        lastNonce: 0,\n        wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n        baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n        coinSpecific: {\n          nonce: -1,\n          updateTime: '2021-08-26T16:23:16.563Z',\n          txCount: 0,\n          pendingChainInitialization: true,\n          creationFailure: [],\n          pendingDeployment: false,\n          forwarderVersion: 0,\n        },\n      };\n\n      const isAddressVerified = await coin.verifyAddress(params as any);\n      isAddressVerified.should.equal(true);\n    });\n\n    it('should verify an address generated using forwarder version 1', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n\n      const params = {\n        id: '61250217c8c02b000654b15e7af6f618',\n        address: '0xb0b56eeae1b283918caca02a14ada2df17a98e6d',\n        chain: 0,\n        index: 3162,\n        coin: 'teth',\n        lastNonce: 0,\n        wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n        baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n        coinSpecific: {\n          nonce: -1,\n          updateTime: '2021-08-24T14:28:39.841Z',\n          txCount: 0,\n          pendingChainInitialization: true,\n          creationFailure: [],\n          salt: '0xc5a',\n          pendingDeployment: true,\n          forwarderVersion: 1,\n        },\n      };\n\n      const isAddressVerified = await coin.verifyAddress(params);\n      isAddressVerified.should.equal(true);\n    });\n\n    it('should reject address verification if coinSpecific field is not an object', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n\n      const params = {\n        id: '61250217c8c02b000654b15e7af6f618',\n        address: '0xb0b56eeae1b283918caca02a14ada2df17a98e6d',\n        chain: 0,\n        index: 3162,\n        coin: 'teth',\n        lastNonce: 0,\n        wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n        baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n      };\n\n      assert.rejects(async () => coin.verifyAddress(params), InvalidAddressVerificationObjectPropertyError);\n    });\n\n    it('should reject address verification when an actual address is different from expected address', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n\n      const params = {\n        id: '61250217c8c02b000654b15e7af6f618',\n        address: '0x28904591f735994f050804fda3b61b813b16e04c',\n        chain: 0,\n        index: 3162,\n        coin: 'teth',\n        lastNonce: 0,\n        baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n        wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n        coinSpecific: {\n          nonce: -1,\n          updateTime: '2021-08-24T14:28:39.841Z',\n          txCount: 0,\n          pendingChainInitialization: true,\n          creationFailure: [],\n          salt: '0xc5a',\n          pendingDeployment: true,\n          forwarderVersion: 1,\n        },\n      };\n\n      assert.rejects(async () => coin.verifyAddress(params), UnexpectedAddressError);\n    });\n\n    it('should reject address verification if the derived address is in invalid format', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n\n      const params = {\n        id: '61250217c8c02b000654b15e7af6f618',\n        address: '0xe0b56eeae1b283918caca02a14ada2df17a98bvf',\n        chain: 0,\n        index: 3162,\n        coin: 'teth',\n        lastNonce: 0,\n        wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n        baseAddress: '0xdf07117705a9f8dc4c2a78de66b7f1797dba9d4e',\n        coinSpecific: {\n          nonce: -1,\n          updateTime: '2021-08-24T14:28:39.841Z',\n          txCount: 0,\n          pendingChainInitialization: true,\n          creationFailure: [],\n          salt: '0xc5a',\n          pendingDeployment: true,\n          forwarderVersion: 1,\n        },\n      };\n\n      assert.rejects(async () => coin.verifyAddress(params), InvalidAddressError);\n    });\n\n    it('should reject address verification if base address is undefined', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n\n      const params = {\n        id: '61250217c8c02b000654b15e7af6f618',\n        address: '0xb0b56eeae1b283918caca02a14ada2df17a98e6d',\n        chain: 0,\n        index: 3162,\n        coin: 'teth',\n        lastNonce: 0,\n        wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n        coinSpecific: {\n          nonce: -1,\n          updateTime: '2021-08-24T14:28:39.841Z',\n          txCount: 0,\n          pendingChainInitialization: true,\n          creationFailure: [],\n          salt: '0xc5a',\n          pendingDeployment: true,\n          forwarderVersion: 1,\n        },\n      };\n\n      assert.rejects(async () => coin.verifyAddress(params), InvalidAddressError);\n    });\n\n    it('should reject address verification if base address is in invalid format', async function () {\n      const coin = bitgo.coin('teth') as Teth;\n\n      const params = {\n        id: '61250217c8c02b000654b15e7af6f618',\n        address: '0xb0b56eeae1b283918caca02a14ada2df17a98e6d',\n        chain: 0,\n        index: 3162,\n        coin: 'teth',\n        lastNonce: 0,\n        wallet: '598f606cd8fc24710d2ebadb1d9459bb',\n        baseAddress: '0xe0b56eeae1b283918caca02a14ada2df17a98bvf',\n        coinSpecific: {\n          nonce: -1,\n          updateTime: '2021-08-24T14:28:39.841Z',\n          txCount: 0,\n          pendingChainInitialization: true,\n          creationFailure: [],\n          salt: '0xc5a',\n          pendingDeployment: true,\n          forwarderVersion: 1,\n        },\n      };\n\n      assert.rejects(async () => coin.verifyAddress(params), InvalidAddressError);\n    });\n  });\n\n  describe('EVM Cross Chain Recovery', function () {\n    const baseUrl = 'https://api-holesky.etherscan.io';\n    it('should build a recovery transaction for hot wallet', async function () {\n      const userKey =\n        '{\"iv\":\"VFZ3jvXhxo1Z+Yaf2MtZnA==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\"\\n' +\n        ':\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"p+fkHuLa/8k=\",\"ct\":\"hYG7pvljLIgCjZ\\n' +\n        '53PBlCde5KZRmlUKKHLtDMk+HJfuU46hW+x+C9WsIAO4gFPnTCvFVmQ8x7czCtcNFub5AO2otOG\\n' +\n        'OsX4GE2gXOEmCl1TpWwwNhm7yMUjGJUpgW6ZZgXSXdDitSKi4V/hk78SGSzjFOBSPYRa6I=\"}\\n';\n      const walletContractAddress = TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS as string;\n      const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';\n      const destinationAddress = '0xd5ADdE17feD8baed3F32b84AF05B8F2816f7b560';\n      const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';\n      const walletPassphrase = TestBitGo.V2.TEST_RECOVERY_PASSCODE as string;\n\n      const basecoin = bitgo.coin('hteth') as Hteth;\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTxListRequest(bitgoFeeAddress))\n        .reply(200, mockData.getTxListResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(bitgoFeeAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(walletContractAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n\n      const spy = sinon.spy(TransactionBuilder.prototype, 'coinUsesNonPackedEncodingForTxData');\n      await basecoin.recover({\n        userKey: userKey,\n        backupKey: '',\n        walletPassphrase: walletPassphrase,\n        walletContractAddress: walletContractAddress,\n        bitgoFeeAddress: bitgoFeeAddress,\n        recoveryDestination: destinationAddress,\n        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n        gasLimit: 500000,\n        bitgoDestinationAddress: bitgoDestinationAddress,\n        intendedChain: 'tarbeth',\n      });\n      assert(spy.returned(true));\n    });\n\n    describe('Non-BitGo Recovery for Hot Wallets (MPCv2)', function () {\n      const baseUrl = 'https://api-holesky.etherscan.io';\n      let bitgo: TestBitGoAPI;\n      let basecoin: Hteth;\n\n      before(function () {\n        bitgo = TestBitGo.decorate(BitGoAPI, { env: 'test' });\n        basecoin = bitgo.coin('hteth') as Hteth;\n      });\n\n      it('should build a recovery transaction for MPCv2 kind of hot wallets', async function () {\n        nock(baseUrl)\n          .get('/api')\n          .query(mockData.getTxListRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().bitgoFeeAddress))\n          .reply(200, mockData.getTxListResponse);\n\n        nock(baseUrl)\n          .get('/api')\n          .query(mockData.getBalanceRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().bitgoFeeAddress))\n          .reply(200, mockData.getBalanceResponse);\n\n        nock(baseUrl)\n          .get('/api')\n          .query(mockData.getBalanceRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().walletContractAddress))\n          .reply(200, mockData.getBalanceResponse);\n\n        nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n\n        const params = mockData.getNonBitGoRecoveryForHotWalletsMPCv2();\n\n        const transaction = await (basecoin as AbstractEthLikeNewCoins).recover({\n          userKey: params.userKey,\n          backupKey: params.backupKey,\n          walletPassphrase: params.walletPassphrase,\n          walletContractAddress: params.walletContractAddress,\n          bitgoFeeAddress: params.bitgoFeeAddress,\n          recoveryDestination: params.recoveryDestination,\n          eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n          gasLimit: 500000,\n          bitgoDestinationAddress: params.bitgoDestinationAddress,\n          intendedChain: params.intendedChain,\n        });\n\n        should.exist(transaction);\n        transaction.should.have.property('txHex');\n      });\n\n      it('should throw an error for invalid user key', async function () {\n        const params = mockData.getInvalidNonBitGoRecoveryParams();\n\n        await assert.rejects(\n          async () => {\n            await (basecoin as AbstractEthLikeNewCoins).recover({\n              userKey: params.userKey,\n              backupKey: params.backupKey,\n              walletPassphrase: params.walletPassphrase,\n              walletContractAddress: params.walletContractAddress,\n              bitgoFeeAddress: params.bitgoFeeAddress,\n              recoveryDestination: params.recoveryDestination,\n              eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n              gasLimit: 500000,\n              bitgoDestinationAddress: params.bitgoDestinationAddress,\n              intendedChain: params.intendedChain,\n            });\n          },\n          Error,\n          'user key is invalid'\n        );\n      });\n    });\n\n    describe('Build Unsigned Sweep for Self-Custody Cold Wallets (MPCv2)', function () {\n      const baseUrl = 'https://api-holesky.etherscan.io';\n      let bitgo: TestBitGoAPI;\n      let basecoin: Hteth;\n\n      before(function () {\n        bitgo = TestBitGo.decorate(BitGoAPI, { env: 'test' });\n        basecoin = bitgo.coin('hteth') as Hteth;\n      });\n\n      it('should generate an unsigned sweep without derivation seed', async function () {\n        nock(baseUrl)\n          .get('/api')\n          .query(mockData.getTxListRequest(mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().address))\n          .reply(200, mockData.getTxListResponse);\n\n        nock(baseUrl)\n          .get('/api')\n          .query(mockData.getBalanceRequest(mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().address))\n          .reply(200, mockData.getBalanceResponse);\n        nock(baseUrl)\n          .get('/api')\n          .query(\n            mockData.getBalanceRequest(\n              mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().walletContractAddress\n            )\n          )\n          .reply(200, mockData.getBalanceResponse);\n\n        nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n\n        const params = mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2();\n        const sweepResult = await (basecoin as AbstractEthLikeNewCoins).recover({\n          userKey: params.commonKeyChain, // Box A Data\n          backupKey: params.commonKeyChain, // Box B Data\n          derivationSeed: params.derivationSeed, // Key Derivation Seed (optional)\n          recoveryDestination: params.recoveryDestination, // Destination Address\n          gasLimit: 200000, // Gas Limit\n          eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 }, // Max Fee Per Gas and Max Priority Fee Per Gas\n          walletContractAddress: params.walletContractAddress,\n          isTss: true,\n          replayProtectionOptions: {\n            chain: '42',\n            hardfork: 'london',\n          },\n        });\n        should.exist(sweepResult);\n        const output = sweepResult as UnsignedSweepTxMPCv2;\n        output.should.have.property('txRequests');\n        output.txRequests.should.have.length(1);\n        output.txRequests[0].should.have.property('transactions');\n        output.txRequests[0].transactions.should.have.length(1);\n        output.txRequests[0].should.have.property('walletCoin');\n        output.txRequests[0].transactions.should.have.length(1);\n        output.txRequests[0].transactions[0].should.have.property('unsignedTx');\n        output.txRequests[0].transactions[0].unsignedTx.should.have.property('serializedTxHex');\n        output.txRequests[0].transactions[0].unsignedTx.should.have.property('signableHex');\n        output.txRequests[0].transactions[0].unsignedTx.should.have.property('derivationPath');\n        output.txRequests[0].transactions[0].unsignedTx.should.have.property('feeInfo');\n        output.txRequests[0].transactions[0].unsignedTx.should.have.property('parsedTx');\n        const parsedTx = output.txRequests[0].transactions[0].unsignedTx.parsedTx as { spendAmount: string };\n        parsedTx.should.have.property('spendAmount');\n        (output.txRequests[0].transactions[0].unsignedTx.parsedTx as { outputs: any[] }).should.have.property(\n          'outputs'\n        );\n      });\n\n      it('should throw an error for invalid address', async function () {\n        const params = mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2();\n        params.recoveryDestination = 'invalidAddress';\n\n        // Ensure userKey and backupKey are the same\n        params.userKey =\n          '0234eb39b22fed523ece7c78da29ba1f1de5b64a6e48013e0914de793bc1df0570e779de04758732734d97e54b782c8b336283811af6a2c57bd81438798e1c2446';\n        params.backupKey =\n          '0234eb39b22fed523ece7c78da29ba1f1de5b64a6e48013e0914de793bc1df0570e779de04758732734d97e54b782c8b336283811af6a2c57bd81438798e1c2446';\n\n        await assert.rejects(\n          async () => {\n            await (basecoin as AbstractEthLikeNewCoins).recover({\n              recoveryDestination: params.recoveryDestination, // Destination Address\n              gasLimit: 2000, // Gas Limit\n              eip1559: { maxFeePerGas: 200, maxPriorityFeePerGas: 10000 }, // Max Fee Per Gas and Max Priority Fee Per Gas\n              userKey: params.userKey, // Provide the userKey\n              backupKey: params.backupKey, // Provide the backupKey\n              walletContractAddress: params.walletContractAddress, // Provide the walletContractAddress\n              isTss: true,\n              replayProtectionOptions: {\n                chain: '42',\n                hardfork: 'london',\n              },\n            });\n          },\n          Error,\n          'Error: invalid address'\n        );\n      });\n    });\n  });\n});\n"]}

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


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