PHP WebShell
Текущая директория: /opt/BitGoJS/modules/bitgo/dist/test/v2/unit/coins
Просмотр файла: abstractUtxoCoin.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const should = require("should");
const sinon = require("sinon");
const sdk_core_1 = require("@bitgo/sdk-core");
const sdk_test_1 = require("@bitgo/sdk-test");
const bitgo_1 = require("../../../../src/bitgo");
describe('Abstract UTXO Coin:', () => {
describe('Parse Transaction:', () => {
const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
const coin = bitgo.coin('tbtc');
/*
* mock objects which get passed into parse transaction.
* These objects are structured to force parse transaction into a
* particular execution path for these tests.
*/
const verification = {
disableNetworking: true,
keychains: {
user: { id: '0', pub: 'aaa', type: 'independent' },
backup: { id: '1', pub: 'bbb', type: 'independent' },
bitgo: { id: '2', pub: 'ccc', type: 'independent' },
},
};
const wallet = sinon.createStubInstance(sdk_core_1.Wallet, {
migratedFrom: '2MzJxAENaesCFu3orrCdj22c69tLEsKXQoR',
});
const outputAmount = (0.01 * 1e8).toString();
async function runClassifyOutputsTest(outputAddress, verification, expectExternal, txParams = {}) {
sinon.stub(coin, 'explainTransaction').resolves({
outputs: [],
changeOutputs: [
{
address: outputAddress,
amount: outputAmount,
},
],
});
if (!txParams.changeAddress) {
sinon.stub(coin, 'verifyAddress').throws(new sdk_core_1.UnexpectedAddressError('test error'));
}
const parsedTransaction = await coin.parseTransaction({
txParams,
txPrebuild: { txHex: '' },
wallet: wallet,
verification,
});
should.exist(parsedTransaction.outputs[0]);
parsedTransaction.outputs[0].should.deepEqual({
address: outputAddress,
amount: outputAmount,
external: expectExternal,
});
const isExplicit = txParams.recipients !== undefined &&
txParams.recipients.some((recipient) => recipient.address === outputAddress);
should.equal(parsedTransaction.explicitExternalSpendAmount, isExplicit && expectExternal ? outputAmount : '0');
should.equal(parsedTransaction.implicitExternalSpendAmount, !isExplicit && expectExternal ? outputAmount : '0');
coin.explainTransaction.restore();
if (!txParams.changeAddress) {
coin.verifyAddress.restore();
}
}
it('should classify outputs which spend change back to a v1 wallet base address as internal', async function () {
return runClassifyOutputsTest(wallet.migratedFrom(), verification, false);
});
it('should classify outputs which spend change back to a v1 wallet base address as external ' +
'if considerMigratedFromAddressInternal is set and false', async function () {
return runClassifyOutputsTest(wallet.migratedFrom(), { ...verification, considerMigratedFromAddressInternal: false }, true);
});
it('should classify outputs which spend to addresses not on the wallet as external', async function () {
return runClassifyOutputsTest('2Mxjx4E2EEe4yJuLvdEuAdMUd4id1emPCZs', verification, true);
});
it('should accept a custom change address', async function () {
const changeAddress = '2NAuziD75WnPPHJVwnd4ckgY4SuJaDVVbMD';
return runClassifyOutputsTest(changeAddress, verification, false, { changeAddress, recipients: [] });
});
it('should classify outputs with external address in recipients as explicit', async function () {
const externalAddress = '2NAuziD75WnPPHJVwnd4ckgY4SuJaDVVbMD';
return runClassifyOutputsTest(externalAddress, verification, true, {
recipients: [{ address: externalAddress, amount: outputAmount }],
});
});
});
describe('Custom Change Wallets', () => {
const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
const coin = bitgo.coin('tbtc');
const keys = {
send: {
user: { id: '0', key: coin.keychains().create() },
backup: { id: '1', key: coin.keychains().create() },
bitgo: { id: '2', key: coin.keychains().create() },
},
change: {
user: { id: '3', key: coin.keychains().create() },
backup: { id: '4', key: coin.keychains().create() },
bitgo: { id: '5', key: coin.keychains().create() },
},
};
const customChangeKeySignatures = {
user: '',
backup: '',
bitgo: '',
};
const addressData = {
chain: 11,
index: 1,
addressType: 'p2shP2wsh',
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
keychains: [
{ pub: keys.change.user.key.pub },
{ pub: keys.change.backup.key.pub },
{ pub: keys.change.bitgo.key.pub },
],
threshold: 2,
};
const { address: changeAddress, coinSpecific } = coin.generateAddress(addressData);
const changeWalletId = 'changeWalletId';
const stubData = {
signedSendingWallet: {
keyIds: sinon.stub().returns([keys.send.user.id, keys.send.backup.id, keys.send.bitgo.id]),
coinSpecific: sinon.stub().returns({ customChangeWalletId: changeWalletId }),
},
changeWallet: {
keyIds: sinon.stub().returns([keys.change.user.id, keys.change.backup.id, keys.change.bitgo.id]),
createAddress: sinon.stub().resolves(changeAddress),
},
};
before(async () => {
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
const sign = async ({ key }) => (await coin.signMessage({ prv: keys.send.user.key.prv }, key.pub)).toString('hex');
customChangeKeySignatures.user = await sign(keys.change.user);
customChangeKeySignatures.backup = await sign(keys.change.backup);
customChangeKeySignatures.bitgo = await sign(keys.change.bitgo);
});
it('should consider addresses derived from the custom change keys as internal spends', async () => {
const signedSendingWallet = sinon.createStubInstance(sdk_core_1.Wallet, stubData.signedSendingWallet);
const changeWallet = sinon.createStubInstance(sdk_core_1.Wallet, stubData.changeWallet);
sinon.stub(coin, 'keychains').returns({
get: sinon.stub().callsFake(({ id }) => {
switch (id) {
case keys.send.user.id:
return Promise.resolve({ id, ...keys.send.user.key });
case keys.send.backup.id:
return Promise.resolve({ id, ...keys.send.backup.key });
case keys.send.bitgo.id:
return Promise.resolve({ id, ...keys.send.bitgo.key });
case keys.change.user.id:
return Promise.resolve({ id, ...keys.change.user.key });
case keys.change.backup.id:
return Promise.resolve({ id, ...keys.change.backup.key });
case keys.change.bitgo.id:
return Promise.resolve({ id, ...keys.change.bitgo.key });
}
}),
});
sinon.stub(coin, 'wallets').returns({
get: sinon.stub().callsFake(() => Promise.resolve(changeWallet)),
});
const outputAmount = 10000;
const recipients = [];
sinon.stub(coin, 'explainTransaction').resolves({
outputs: [],
changeOutputs: [
{
address: changeAddress,
amount: outputAmount,
},
],
});
signedSendingWallet._wallet = signedSendingWallet._wallet || { customChangeKeySignatures };
const parsedTransaction = await coin.parseTransaction({
txParams: { changeAddress, recipients },
txPrebuild: { txHex: '' },
wallet: signedSendingWallet,
verification: {
addresses: {
[changeAddress]: {
coinSpecific,
chain: addressData.chain,
index: addressData.index,
},
},
},
});
should.exist(parsedTransaction.outputs[0]);
parsedTransaction.outputs[0].should.deepEqual({
address: changeAddress,
amount: outputAmount,
external: false,
needsCustomChangeKeySignatureVerification: true,
});
coin.explainTransaction.restore();
coin.wallets.restore();
coin.keychains.restore();
});
});
describe('Verify Transaction', () => {
const bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'mock' });
const coin = bitgo.coin('tbtc');
const userKeychain = coin.keychains().create();
const otherKeychain = coin.keychains().create();
const changeKeys = {
user: coin.keychains().create(),
backup: coin.keychains().create(),
bitgo: coin.keychains().create(),
};
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
const sign = async (key, keychain) => (await coin.signMessage(keychain, key.pub)).toString('hex');
const signUser = (key) => sign(key, userKeychain);
const signOther = (key) => sign(key, otherKeychain);
const passphrase = 'test_passphrase';
const stubData = {
unsignedSendingWallet: {
keyIds: sinon.stub().returns(['0', '1', '2']),
},
parseTransactionData: {
badKey: {
keychains: {
// user public key swapped out
user: {
pub: otherKeychain.pub,
encryptedPrv: bitgo.encrypt({ input: userKeychain.prv, password: passphrase }),
},
},
needsCustomChangeKeySignatureVerification: true,
},
noCustomChange: {
keychains: { user: userKeychain },
needsCustomChangeKeySignatureVerification: true,
},
emptyCustomChange: {
keychains: { user: userKeychain },
needsCustomChangeKeySignatureVerification: true,
customChange: {},
},
// needs to be async function to create signatures
badSigs: async () => ({
keychains: { user: userKeychain },
needsCustomChangeKeySignatureVerification: true,
customChange: {
keys: [changeKeys.user, changeKeys.backup, changeKeys.bitgo],
signatures: [
await signOther(changeKeys.user),
await signOther(changeKeys.backup),
await signOther(changeKeys.bitgo),
],
},
}),
goodSigs: async () => ({
keychains: { user: userKeychain },
needsCustomChangeKeySignatureVerification: true,
customChange: {
keys: [changeKeys.user, changeKeys.backup, changeKeys.bitgo],
signatures: [
await signUser(changeKeys.user),
await signUser(changeKeys.backup),
await signUser(changeKeys.bitgo),
],
},
missingOutputs: 1,
}),
},
};
const unsignedSendingWallet = sinon.createStubInstance(sdk_core_1.Wallet, stubData.unsignedSendingWallet);
it('should fail if the user private key cannot be verified to match the user public key', async () => {
sinon.stub(coin, 'parseTransaction').resolves(stubData.parseTransactionData.badKey);
const verifyWallet = sinon.createStubInstance(sdk_core_1.Wallet, {});
await coin
.verifyTransaction({
txParams: {
walletPassphrase: passphrase,
},
txPrebuild: {},
wallet: verifyWallet,
verification: {},
})
.should.be.rejectedWith(/transaction requires verification of user public key, but it was unable to be verified/);
coin.parseTransaction.restore();
});
it('should fail if the custom change verification data is required but missing', async () => {
sinon.stub(coin, 'parseTransaction').resolves(stubData.parseTransactionData.noCustomChange);
await coin
.verifyTransaction({
txParams: {
walletPassphrase: passphrase,
},
txPrebuild: {},
wallet: unsignedSendingWallet,
verification: {},
})
.should.be.rejectedWith(/parsed transaction is missing required custom change verification data/);
coin.parseTransaction.restore();
});
it('should fail if the custom change keys or key signatures are missing', async () => {
sinon.stub(coin, 'parseTransaction').resolves(stubData.parseTransactionData.emptyCustomChange);
await coin
.verifyTransaction({
txParams: {
walletPassphrase: passphrase,
},
txPrebuild: {},
wallet: unsignedSendingWallet,
verification: {},
})
.should.be.rejectedWith(/customChange property is missing keys or signatures/);
coin.parseTransaction.restore();
});
it('should fail if the custom change key signatures cannot be verified', async () => {
sinon.stub(coin, 'parseTransaction').resolves((await stubData.parseTransactionData.badSigs()));
await coin
.verifyTransaction({
txParams: {
walletPassphrase: passphrase,
},
txPrebuild: {},
wallet: unsignedSendingWallet,
verification: {},
})
.should.be.rejectedWith(/transaction requires verification of custom change key signatures, but they were unable to be verified/);
coin.parseTransaction.restore();
});
it('should successfully verify a custom change transaction when change keys and signatures are valid', async () => {
sinon.stub(coin, 'parseTransaction').resolves((await stubData.parseTransactionData.goodSigs()));
// if verify transaction gets rejected with the outputs missing error message,
// then we know that the verification of the custom change key signatures was successful
await coin
.verifyTransaction({
txParams: {
walletPassphrase: passphrase,
},
txPrebuild: {},
wallet: unsignedSendingWallet,
verification: {},
})
.should.be.rejectedWith(/expected outputs missing in transaction prebuild/);
coin.parseTransaction.restore();
});
it('should not allow more than 150 basis points of implicit external outputs (for paygo outputs)', async () => {
const coinMock = sinon.stub(coin, 'parseTransaction').resolves({
keychains: {},
keySignatures: {},
outputs: [],
missingOutputs: [],
explicitExternalOutputs: [],
implicitExternalOutputs: [],
changeOutputs: [],
explicitExternalSpendAmount: 10000,
implicitExternalSpendAmount: 151,
needsCustomChangeKeySignatureVerification: false,
});
await coin
.verifyTransaction({
txParams: {
walletPassphrase: passphrase,
},
txPrebuild: {},
wallet: unsignedSendingWallet,
})
.should.be.rejectedWith('prebuild attempts to spend to unintended external recipients');
coinMock.restore();
});
it('should allow 150 basis points of implicit external outputs (for paygo outputs)', async () => {
const coinMock = sinon.stub(coin, 'parseTransaction').resolves({
keychains: {},
keySignatures: {},
outputs: [],
missingOutputs: [],
explicitExternalOutputs: [],
implicitExternalOutputs: [],
changeOutputs: [],
explicitExternalSpendAmount: 1000,
implicitExternalSpendAmount: 15,
needsCustomChangeKeySignatureVerification: false,
});
const bitcoinMock = sinon
.stub(coin, 'createTransactionFromHex')
.returns({ ins: [] });
await coin
.verifyTransaction({
txParams: {
walletPassphrase: passphrase,
},
txPrebuild: {
txHex: '00',
},
wallet: unsignedSendingWallet,
})
.should.eventually.be.true();
coinMock.restore();
bitcoinMock.restore();
});
it('should not allow any implicit external outputs if paygo outputs are disallowed', async () => {
const coinMock = sinon.stub(coin, 'parseTransaction').resolves({
keychains: {},
keySignatures: {},
outputs: [],
missingOutputs: [],
explicitExternalOutputs: [],
implicitExternalOutputs: [],
changeOutputs: [],
explicitExternalSpendAmount: 0,
implicitExternalSpendAmount: 10,
needsCustomChangeKeySignatureVerification: false,
});
await coin
.verifyTransaction({
txParams: {
walletPassphrase: passphrase,
},
txPrebuild: {
txHex: '00',
},
wallet: unsignedSendingWallet,
verification: {
allowPaygoOutput: false,
},
})
.should.be.rejectedWith('prebuild attempts to spend to unintended external recipients');
coinMock.restore();
});
it('should allow paygo outputs if empty verification object is passed', async () => {
const coinMock = sinon.stub(coin, 'parseTransaction').resolves({
keychains: {},
keySignatures: {},
outputs: [],
missingOutputs: [],
explicitExternalOutputs: [],
implicitExternalOutputs: [],
changeOutputs: [],
explicitExternalSpendAmount: 1000,
implicitExternalSpendAmount: 15,
needsCustomChangeKeySignatureVerification: false,
});
const bitcoinMock = sinon
.stub(coin, 'createTransactionFromHex')
.returns({ ins: [] });
await coin
.verifyTransaction({
txParams: {
walletPassphrase: passphrase,
},
txPrebuild: {
txHex: '00',
},
wallet: unsignedSendingWallet,
verification: {},
})
.should.eventually.be.true();
coinMock.restore();
bitcoinMock.restore();
});
it('should work with bigint amounts', async () => {
// need a coin that uses bigint
const bigintCoin = bitgo.coin('tdoge');
const coinMock = sinon.stub(bigintCoin, 'parseTransaction').resolves({
keychains: {},
keySignatures: {},
outputs: [],
missingOutputs: [],
explicitExternalOutputs: [
{
address: 'external_address',
amount: '10000',
},
],
implicitExternalOutputs: [
{
address: 'external_address_2',
amount: '15',
},
],
changeOutputs: [],
explicitExternalSpendAmount: BigInt(10000),
implicitExternalSpendAmount: BigInt(15),
needsCustomChangeKeySignatureVerification: false,
});
const bitcoinMock = sinon
.stub(bigintCoin, 'createTransactionFromHex')
.returns({ ins: [] });
await bigintCoin
.verifyTransaction({
txParams: {
walletPassphrase: passphrase,
},
txPrebuild: {
txHex: '00',
},
wallet: unsignedSendingWallet,
verification: {},
})
.should.eventually.be.true();
coinMock.restore();
bitcoinMock.restore();
});
});
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstractUtxoCoin.js","sourceRoot":"","sources":["../../../../../test/v2/unit/coins/abstractUtxoCoin.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo/utxo-lib';\nimport * as should from 'should';\nimport * as sinon from 'sinon';\nimport { Wallet, UnexpectedAddressError, VerificationOptions } from '@bitgo/sdk-core';\nimport { TestBitGo } from '@bitgo/sdk-test';\nimport { BitGo } from '../../../../src/bitgo';\nimport { AbstractUtxoCoin, UtxoWallet, Output, TransactionExplanation, TransactionParams } from '@bitgo/abstract-utxo';\n\ndescribe('Abstract UTXO Coin:', () => {\n  describe('Parse Transaction:', () => {\n    const bitgo: BitGo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    const coin = bitgo.coin('tbtc') as AbstractUtxoCoin;\n\n    /*\n     * mock objects which get passed into parse transaction.\n     * These objects are structured to force parse transaction into a\n     * particular execution path for these tests.\n     */\n    const verification: VerificationOptions = {\n      disableNetworking: true,\n      keychains: {\n        user: { id: '0', pub: 'aaa', type: 'independent' },\n        backup: { id: '1', pub: 'bbb', type: 'independent' },\n        bitgo: { id: '2', pub: 'ccc', type: 'independent' },\n      },\n    };\n\n    const wallet = sinon.createStubInstance(Wallet, {\n      migratedFrom: '2MzJxAENaesCFu3orrCdj22c69tLEsKXQoR',\n    });\n\n    const outputAmount = (0.01 * 1e8).toString();\n\n    async function runClassifyOutputsTest(\n      outputAddress,\n      verification,\n      expectExternal,\n      txParams: TransactionParams = {}\n    ) {\n      sinon.stub(coin, 'explainTransaction').resolves({\n        outputs: [] as Output[],\n        changeOutputs: [\n          {\n            address: outputAddress,\n            amount: outputAmount,\n          },\n        ],\n      } as TransactionExplanation);\n\n      if (!txParams.changeAddress) {\n        sinon.stub(coin, 'verifyAddress').throws(new UnexpectedAddressError('test error'));\n      }\n\n      const parsedTransaction = await coin.parseTransaction({\n        txParams,\n        txPrebuild: { txHex: '' },\n        wallet: wallet as unknown as UtxoWallet,\n        verification,\n      });\n\n      should.exist(parsedTransaction.outputs[0]);\n      parsedTransaction.outputs[0].should.deepEqual({\n        address: outputAddress,\n        amount: outputAmount,\n        external: expectExternal,\n      });\n\n      const isExplicit =\n        txParams.recipients !== undefined &&\n        txParams.recipients.some((recipient) => recipient.address === outputAddress);\n      should.equal(parsedTransaction.explicitExternalSpendAmount, isExplicit && expectExternal ? outputAmount : '0');\n      should.equal(parsedTransaction.implicitExternalSpendAmount, !isExplicit && expectExternal ? outputAmount : '0');\n\n      (coin.explainTransaction as any).restore();\n\n      if (!txParams.changeAddress) {\n        (coin.verifyAddress as any).restore();\n      }\n    }\n\n    it('should classify outputs which spend change back to a v1 wallet base address as internal', async function () {\n      return runClassifyOutputsTest(wallet.migratedFrom(), verification, false);\n    });\n\n    it(\n      'should classify outputs which spend change back to a v1 wallet base address as external ' +\n        'if considerMigratedFromAddressInternal is set and false',\n      async function () {\n        return runClassifyOutputsTest(\n          wallet.migratedFrom(),\n          { ...verification, considerMigratedFromAddressInternal: false },\n          true\n        );\n      }\n    );\n\n    it('should classify outputs which spend to addresses not on the wallet as external', async function () {\n      return runClassifyOutputsTest('2Mxjx4E2EEe4yJuLvdEuAdMUd4id1emPCZs', verification, true);\n    });\n\n    it('should accept a custom change address', async function () {\n      const changeAddress = '2NAuziD75WnPPHJVwnd4ckgY4SuJaDVVbMD';\n      return runClassifyOutputsTest(changeAddress, verification, false, { changeAddress, recipients: [] });\n    });\n\n    it('should classify outputs with external address in recipients as explicit', async function () {\n      const externalAddress = '2NAuziD75WnPPHJVwnd4ckgY4SuJaDVVbMD';\n      return runClassifyOutputsTest(externalAddress, verification, true, {\n        recipients: [{ address: externalAddress, amount: outputAmount }],\n      });\n    });\n  });\n\n  describe('Custom Change Wallets', () => {\n    const bitgo: BitGo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    const coin = bitgo.coin('tbtc') as AbstractUtxoCoin;\n\n    const keys = {\n      send: {\n        user: { id: '0', key: coin.keychains().create() },\n        backup: { id: '1', key: coin.keychains().create() },\n        bitgo: { id: '2', key: coin.keychains().create() },\n      },\n      change: {\n        user: { id: '3', key: coin.keychains().create() },\n        backup: { id: '4', key: coin.keychains().create() },\n        bitgo: { id: '5', key: coin.keychains().create() },\n      },\n    };\n\n    const customChangeKeySignatures = {\n      user: '',\n      backup: '',\n      bitgo: '',\n    };\n\n    const addressData = {\n      chain: 11,\n      index: 1,\n      addressType: 'p2shP2wsh' as const,\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      keychains: [\n        { pub: keys.change.user.key.pub! },\n        { pub: keys.change.backup.key.pub! },\n        { pub: keys.change.bitgo.key.pub! },\n      ],\n      threshold: 2,\n    };\n\n    const { address: changeAddress, coinSpecific } = coin.generateAddress(addressData);\n\n    const changeWalletId = 'changeWalletId';\n    const stubData = {\n      signedSendingWallet: {\n        keyIds: sinon.stub().returns([keys.send.user.id, keys.send.backup.id, keys.send.bitgo.id]),\n        coinSpecific: sinon.stub().returns({ customChangeWalletId: changeWalletId }),\n      },\n      changeWallet: {\n        keyIds: sinon.stub().returns([keys.change.user.id, keys.change.backup.id, keys.change.bitgo.id]),\n        createAddress: sinon.stub().resolves(changeAddress),\n      },\n    };\n\n    before(async () => {\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      const sign = async ({ key }) =>\n        (await coin.signMessage({ prv: keys.send.user.key.prv }, key.pub!)).toString('hex');\n      customChangeKeySignatures.user = await sign(keys.change.user);\n      customChangeKeySignatures.backup = await sign(keys.change.backup);\n      customChangeKeySignatures.bitgo = await sign(keys.change.bitgo);\n    });\n\n    it('should consider addresses derived from the custom change keys as internal spends', async () => {\n      const signedSendingWallet = sinon.createStubInstance(Wallet, stubData.signedSendingWallet as any);\n      const changeWallet = sinon.createStubInstance(Wallet, stubData.changeWallet as any);\n\n      sinon.stub(coin, 'keychains').returns({\n        get: sinon.stub().callsFake(({ id }) => {\n          switch (id) {\n            case keys.send.user.id:\n              return Promise.resolve({ id, ...keys.send.user.key });\n            case keys.send.backup.id:\n              return Promise.resolve({ id, ...keys.send.backup.key });\n            case keys.send.bitgo.id:\n              return Promise.resolve({ id, ...keys.send.bitgo.key });\n            case keys.change.user.id:\n              return Promise.resolve({ id, ...keys.change.user.key });\n            case keys.change.backup.id:\n              return Promise.resolve({ id, ...keys.change.backup.key });\n            case keys.change.bitgo.id:\n              return Promise.resolve({ id, ...keys.change.bitgo.key });\n          }\n        }),\n      } as any);\n\n      sinon.stub(coin, 'wallets').returns({\n        get: sinon.stub().callsFake(() => Promise.resolve(changeWallet)),\n      } as any);\n\n      const outputAmount = 10000;\n      const recipients = [];\n\n      sinon.stub(coin, 'explainTransaction').resolves({\n        outputs: [],\n        changeOutputs: [\n          {\n            address: changeAddress,\n            amount: outputAmount,\n          },\n        ],\n      } as any);\n\n      signedSendingWallet._wallet = signedSendingWallet._wallet || { customChangeKeySignatures };\n\n      const parsedTransaction = await coin.parseTransaction({\n        txParams: { changeAddress, recipients },\n        txPrebuild: { txHex: '' },\n        wallet: signedSendingWallet as any,\n        verification: {\n          addresses: {\n            [changeAddress]: {\n              coinSpecific,\n              chain: addressData.chain,\n              index: addressData.index,\n            },\n          },\n        },\n      });\n\n      should.exist(parsedTransaction.outputs[0]);\n      parsedTransaction.outputs[0].should.deepEqual({\n        address: changeAddress,\n        amount: outputAmount,\n        external: false,\n        needsCustomChangeKeySignatureVerification: true,\n      });\n\n      (coin.explainTransaction as any).restore();\n      (coin.wallets as any).restore();\n      (coin.keychains as any).restore();\n    });\n  });\n\n  describe('Verify Transaction', () => {\n    const bitgo: BitGo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    const coin = bitgo.coin('tbtc') as AbstractUtxoCoin;\n\n    const userKeychain = coin.keychains().create();\n    const otherKeychain = coin.keychains().create();\n\n    const changeKeys = {\n      user: coin.keychains().create(),\n      backup: coin.keychains().create(),\n      bitgo: coin.keychains().create(),\n    };\n\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n    const sign = async (key, keychain) => (await coin.signMessage(keychain, key.pub!)).toString('hex');\n    const signUser = (key) => sign(key, userKeychain);\n    const signOther = (key) => sign(key, otherKeychain);\n    const passphrase = 'test_passphrase';\n\n    const stubData = {\n      unsignedSendingWallet: {\n        keyIds: sinon.stub().returns(['0', '1', '2']),\n      },\n      parseTransactionData: {\n        badKey: {\n          keychains: {\n            // user public key swapped out\n            user: {\n              pub: otherKeychain.pub,\n              encryptedPrv: bitgo.encrypt({ input: userKeychain.prv, password: passphrase }),\n            },\n          },\n          needsCustomChangeKeySignatureVerification: true,\n        },\n        noCustomChange: {\n          keychains: { user: userKeychain },\n          needsCustomChangeKeySignatureVerification: true,\n        },\n        emptyCustomChange: {\n          keychains: { user: userKeychain },\n          needsCustomChangeKeySignatureVerification: true,\n          customChange: {},\n        },\n        // needs to be async function to create signatures\n        badSigs: async () => ({\n          keychains: { user: userKeychain },\n          needsCustomChangeKeySignatureVerification: true,\n          customChange: {\n            keys: [changeKeys.user, changeKeys.backup, changeKeys.bitgo],\n            signatures: [\n              await signOther(changeKeys.user),\n              await signOther(changeKeys.backup),\n              await signOther(changeKeys.bitgo),\n            ],\n          },\n        }),\n        goodSigs: async () => ({\n          keychains: { user: userKeychain },\n          needsCustomChangeKeySignatureVerification: true,\n          customChange: {\n            keys: [changeKeys.user, changeKeys.backup, changeKeys.bitgo],\n            signatures: [\n              await signUser(changeKeys.user),\n              await signUser(changeKeys.backup),\n              await signUser(changeKeys.bitgo),\n            ],\n          },\n          missingOutputs: 1,\n        }),\n      },\n    };\n\n    const unsignedSendingWallet = sinon.createStubInstance(Wallet, stubData.unsignedSendingWallet as any);\n\n    it('should fail if the user private key cannot be verified to match the user public key', async () => {\n      sinon.stub(coin, 'parseTransaction').resolves(stubData.parseTransactionData.badKey as any);\n      const verifyWallet = sinon.createStubInstance(Wallet, {});\n\n      await coin\n        .verifyTransaction({\n          txParams: {\n            walletPassphrase: passphrase,\n          },\n          txPrebuild: {},\n          wallet: verifyWallet as any,\n          verification: {},\n        })\n        .should.be.rejectedWith(\n          /transaction requires verification of user public key, but it was unable to be verified/\n        );\n\n      (coin.parseTransaction as any).restore();\n    });\n\n    it('should fail if the custom change verification data is required but missing', async () => {\n      sinon.stub(coin, 'parseTransaction').resolves(stubData.parseTransactionData.noCustomChange as any);\n\n      await coin\n        .verifyTransaction({\n          txParams: {\n            walletPassphrase: passphrase,\n          },\n          txPrebuild: {},\n          wallet: unsignedSendingWallet as any,\n          verification: {},\n        })\n        .should.be.rejectedWith(/parsed transaction is missing required custom change verification data/);\n\n      (coin.parseTransaction as any).restore();\n    });\n\n    it('should fail if the custom change keys or key signatures are missing', async () => {\n      sinon.stub(coin, 'parseTransaction').resolves(stubData.parseTransactionData.emptyCustomChange as any);\n\n      await coin\n        .verifyTransaction({\n          txParams: {\n            walletPassphrase: passphrase,\n          },\n          txPrebuild: {},\n          wallet: unsignedSendingWallet as any,\n          verification: {},\n        })\n        .should.be.rejectedWith(/customChange property is missing keys or signatures/);\n\n      (coin.parseTransaction as any).restore();\n    });\n\n    it('should fail if the custom change key signatures cannot be verified', async () => {\n      sinon.stub(coin, 'parseTransaction').resolves((await stubData.parseTransactionData.badSigs()) as any);\n\n      await coin\n        .verifyTransaction({\n          txParams: {\n            walletPassphrase: passphrase,\n          },\n          txPrebuild: {},\n          wallet: unsignedSendingWallet as any,\n          verification: {},\n        })\n        .should.be.rejectedWith(\n          /transaction requires verification of custom change key signatures, but they were unable to be verified/\n        );\n\n      (coin.parseTransaction as any).restore();\n    });\n\n    it('should successfully verify a custom change transaction when change keys and signatures are valid', async () => {\n      sinon.stub(coin, 'parseTransaction').resolves((await stubData.parseTransactionData.goodSigs()) as any);\n\n      // if verify transaction gets rejected with the outputs missing error message,\n      // then we know that the verification of the custom change key signatures was successful\n      await coin\n        .verifyTransaction({\n          txParams: {\n            walletPassphrase: passphrase,\n          },\n          txPrebuild: {},\n          wallet: unsignedSendingWallet as any,\n          verification: {},\n        })\n        .should.be.rejectedWith(/expected outputs missing in transaction prebuild/);\n\n      (coin.parseTransaction as any).restore();\n    });\n\n    it('should not allow more than 150 basis points of implicit external outputs (for paygo outputs)', async () => {\n      const coinMock = sinon.stub(coin, 'parseTransaction').resolves({\n        keychains: {} as any,\n        keySignatures: {},\n        outputs: [],\n        missingOutputs: [],\n        explicitExternalOutputs: [],\n        implicitExternalOutputs: [],\n        changeOutputs: [],\n        explicitExternalSpendAmount: 10000,\n        implicitExternalSpendAmount: 151,\n        needsCustomChangeKeySignatureVerification: false,\n      });\n\n      await coin\n        .verifyTransaction({\n          txParams: {\n            walletPassphrase: passphrase,\n          },\n          txPrebuild: {},\n          wallet: unsignedSendingWallet as any,\n        })\n        .should.be.rejectedWith('prebuild attempts to spend to unintended external recipients');\n\n      coinMock.restore();\n    });\n\n    it('should allow 150 basis points of implicit external outputs (for paygo outputs)', async () => {\n      const coinMock = sinon.stub(coin, 'parseTransaction').resolves({\n        keychains: {} as any,\n        keySignatures: {},\n        outputs: [],\n        missingOutputs: [],\n        explicitExternalOutputs: [],\n        implicitExternalOutputs: [],\n        changeOutputs: [],\n        explicitExternalSpendAmount: 1000,\n        implicitExternalSpendAmount: 15,\n        needsCustomChangeKeySignatureVerification: false,\n      });\n\n      const bitcoinMock = sinon\n        .stub(coin, 'createTransactionFromHex')\n        .returns({ ins: [] } as unknown as utxolib.bitgo.UtxoTransaction);\n\n      await coin\n        .verifyTransaction({\n          txParams: {\n            walletPassphrase: passphrase,\n          },\n          txPrebuild: {\n            txHex: '00',\n          },\n          wallet: unsignedSendingWallet as any,\n        })\n        .should.eventually.be.true();\n\n      coinMock.restore();\n      bitcoinMock.restore();\n    });\n\n    it('should not allow any implicit external outputs if paygo outputs are disallowed', async () => {\n      const coinMock = sinon.stub(coin, 'parseTransaction').resolves({\n        keychains: {} as any,\n        keySignatures: {},\n        outputs: [],\n        missingOutputs: [],\n        explicitExternalOutputs: [],\n        implicitExternalOutputs: [],\n        changeOutputs: [],\n        explicitExternalSpendAmount: 0,\n        implicitExternalSpendAmount: 10,\n        needsCustomChangeKeySignatureVerification: false,\n      });\n\n      await coin\n        .verifyTransaction({\n          txParams: {\n            walletPassphrase: passphrase,\n          },\n          txPrebuild: {\n            txHex: '00',\n          },\n          wallet: unsignedSendingWallet as any,\n          verification: {\n            allowPaygoOutput: false,\n          },\n        })\n        .should.be.rejectedWith('prebuild attempts to spend to unintended external recipients');\n\n      coinMock.restore();\n    });\n\n    it('should allow paygo outputs if empty verification object is passed', async () => {\n      const coinMock = sinon.stub(coin, 'parseTransaction').resolves({\n        keychains: {} as any,\n        keySignatures: {},\n        outputs: [],\n        missingOutputs: [],\n        explicitExternalOutputs: [],\n        implicitExternalOutputs: [],\n        changeOutputs: [],\n        explicitExternalSpendAmount: 1000,\n        implicitExternalSpendAmount: 15,\n        needsCustomChangeKeySignatureVerification: false,\n      });\n\n      const bitcoinMock = sinon\n        .stub(coin, 'createTransactionFromHex')\n        .returns({ ins: [] } as unknown as utxolib.bitgo.UtxoTransaction);\n\n      await coin\n        .verifyTransaction({\n          txParams: {\n            walletPassphrase: passphrase,\n          },\n          txPrebuild: {\n            txHex: '00',\n          },\n          wallet: unsignedSendingWallet as any,\n          verification: {},\n        })\n        .should.eventually.be.true();\n\n      coinMock.restore();\n      bitcoinMock.restore();\n    });\n\n    it('should work with bigint amounts', async () => {\n      // need a coin that uses bigint\n      const bigintCoin = bitgo.coin('tdoge') as AbstractUtxoCoin;\n\n      const coinMock = sinon.stub(bigintCoin, 'parseTransaction').resolves({\n        keychains: {} as any,\n        keySignatures: {},\n        outputs: [],\n        missingOutputs: [],\n        explicitExternalOutputs: [\n          {\n            address: 'external_address',\n            amount: '10000',\n          },\n        ],\n        implicitExternalOutputs: [\n          {\n            address: 'external_address_2',\n            amount: '15',\n          },\n        ],\n        changeOutputs: [],\n        explicitExternalSpendAmount: BigInt(10000),\n        implicitExternalSpendAmount: BigInt(15),\n        needsCustomChangeKeySignatureVerification: false,\n      });\n\n      const bitcoinMock = sinon\n        .stub(bigintCoin, 'createTransactionFromHex')\n        .returns({ ins: [] } as unknown as utxolib.bitgo.UtxoTransaction);\n\n      await bigintCoin\n        .verifyTransaction({\n          txParams: {\n            walletPassphrase: passphrase,\n          },\n          txPrebuild: {\n            txHex: '00',\n          },\n          wallet: unsignedSendingWallet as any,\n          verification: {},\n        })\n        .should.eventually.be.true();\n\n      coinMock.restore();\n      bitcoinMock.restore();\n    });\n  });\n});\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!