PHP WebShell
Текущая директория: /opt/BitGoJS/modules/bitgo/dist/test/v2/unit/coins/utxo/recovery
Просмотр файла: crossChainRecovery.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
/**
* @prettier
*/
const assert = require("assert");
const should = require("should");
const nock = require("nock");
const utxolib = require("@bitgo/utxo-lib");
const util_1 = require("../util");
const sdk_test_1 = require("@bitgo/sdk-test");
const nockBitGo_1 = require("../util/nockBitGo");
const transaction_1 = require("../util/transaction");
const keychains_1 = require("../util/keychains");
const mock_1 = require("./mock");
const abstract_utxo_1 = require("@bitgo/abstract-utxo");
const sinon = require("sinon");
function getKeyId(k) {
return (0, sdk_test_1.getSeed)(k.pub).toString('hex');
}
function nockWallet(coin, walletId, walletKeys) {
return [
(0, nockBitGo_1.nockBitGo)()
.get(`/api/v2/${coin.getChain()}/wallet/${walletId}`)
.reply(200, {
id: walletId,
coin: coin.getChain(),
label: 'crossChainRecovery',
keys: walletKeys.map((k) => getKeyId(k)),
})
.persist(),
...walletKeys.map((k) => (0, nockBitGo_1.nockBitGo)()
.get(`/api/v2/${coin.getChain()}/key/${getKeyId(k)}`)
.reply(200, k)
.persist()),
];
}
function nockWalletAddress(coin, walletId, address) {
return (0, nockBitGo_1.nockBitGo)()
.get(`/api/v2/${coin.getChain()}/wallet/${walletId}/address/${address.address}`)
.reply(200, {
address: address.address,
chain: address.chain,
index: address.index,
coin: coin.getChain(),
wallet: walletId,
coinSpecific: address.coinSpecific,
})
.persist();
}
/**
* Setup test for cross-chain recovery.
*
* Users can receive deposits on wallet addresses that are on a different chain.
*
* For instance, a user can receive litecoin on a bitcoin wallet.
* This means that the litecoin blockchain has a transaction with outputs that are spendable
* with keys that were originally created for a BitGo BTC wallet.
* In this example, LTC is the "source coin" and BTC is the "recovery coin"
* In cases like these we must use construct a transaction for litecoin network using keys of the
* bitcoin wallet.
*
* @param sourceCoin - the coin to construct the transaction for
* @param recoveryCoin - the coin the receiving wallet was set up for
*/
function run(sourceCoin, recoveryCoin) {
describe(`Cross-Chain Recovery [sourceCoin=${sourceCoin.getChain()} recoveryCoin=${recoveryCoin.getChain()}]`, function () {
const walletKeys = (0, util_1.getDefaultWalletKeys)();
const recoveryWalletId = '5abacebe28d72fbd07e0b8cbba0ff39e';
// the address the accidental deposit went to, in both sourceCoin and addressCoin formats
const [depositAddressSourceCoin, depositAddressRecoveryCoin] = [sourceCoin, recoveryCoin].map((coin) => coin.generateAddress({ keychains: util_1.keychainsBase58, index: 0 }));
// the address where we want to recover our funds to
const recoveryAddress = sourceCoin.generateAddress({ keychains: util_1.keychainsBase58, index: 1 }).address;
const nocks = [];
let depositTx;
function getDepositUnspents() {
return [
(0, util_1.mockUnspent)(sourceCoin.network, walletKeys, 'p2sh', 0, (sourceCoin.amountType === 'bigint' ? BigInt('10999999800000001') : 1e8)),
];
}
function getDepositTransaction() {
return (0, transaction_1.createFullSignedTransaction)(sourceCoin.network, getDepositUnspents(), depositAddressSourceCoin.address, (0, keychains_1.getDefaultWalletUnspentSigner)());
}
before('prepare deposit tx', function () {
depositTx = getDepositTransaction();
});
function getRecoveryUnspents() {
return [
{
id: depositTx.getId(),
address: depositAddressSourceCoin.address,
chain: depositAddressSourceCoin.chain,
index: depositAddressSourceCoin.index,
value: depositTx.outs[0].value,
},
];
}
before('setup nocks', function () {
nocks.push(...nockWallet(recoveryCoin, recoveryWalletId, util_1.keychainsBase58));
nocks.push(nockWalletAddress(recoveryCoin, recoveryWalletId, depositAddressRecoveryCoin));
});
after(function () {
nocks.forEach((n) => n.done());
});
after(function () {
nock.cleanAll();
});
afterEach(function () {
sinon.restore();
});
function testMatchFixture(name, getRecoveryResult) {
it(`should match fixture (${name})`, async function () {
const recovery = getRecoveryResult();
let recoveryObj = {
...recovery,
tx: (0, util_1.transactionHexToObj)(recovery.txHex, sourceCoin.network, sourceCoin.amountType),
};
if (sourceCoin.amountType === 'bigint') {
recoveryObj = JSON.parse(JSON.stringify(recoveryObj, (k, v) => {
if (typeof v === 'bigint') {
return v.toString();
}
else {
return v;
}
}));
}
(0, util_1.shouldEqualJSON)(recoveryObj, await (0, util_1.getFixture)(sourceCoin, `recovery/crossChainRecovery-${recoveryCoin.getChain()}-${name}`, recoveryObj));
});
}
function checkRecoveryTransactionSignature(tx) {
if (typeof tx === 'string') {
tx = utxolib.bitgo.createTransactionFromBuffer(Buffer.from(tx, 'hex'), sourceCoin.network, {
amountType: sourceCoin.amountType,
});
}
const unspents = getRecoveryUnspents();
should.equal(tx.ins.length, unspents.length);
tx.ins.forEach((input, i) => {
assert(typeof tx !== 'string');
utxolib.bitgo
.verifySignatureWithUnspent(tx, i, getRecoveryUnspents(), walletKeys)
.should.eql([true, false, false]);
});
}
it('should test signed cross chain recovery', async () => {
const getRecoveryProviderStub = sinon
.stub(abstract_utxo_1.AbstractUtxoCoin.prototype, 'getRecoveryProvider')
.returns(new mock_1.MockCrossChainRecoveryProvider(sourceCoin, getDepositUnspents(), depositTx));
const params = {
recoveryCoin,
txid: depositTx.getId(),
recoveryAddress,
wallet: recoveryWalletId,
};
const signedRecovery = (await sourceCoin.recoverFromWrongChain({
...params,
xprv: util_1.keychainsBase58[0].prv,
}));
should.equal(getRecoveryProviderStub.callCount, 1);
testMatchFixture('signed', () => signedRecovery);
it('should have valid signatures for signed recovery', function () {
checkRecoveryTransactionSignature(signedRecovery.txHex);
});
});
it('should test unsigned cross chain recovery', async () => {
const getRecoveryProviderStub = sinon
.stub(abstract_utxo_1.AbstractUtxoCoin.prototype, 'getRecoveryProvider')
.returns(new mock_1.MockCrossChainRecoveryProvider(sourceCoin, getDepositUnspents(), depositTx));
const params = {
recoveryCoin,
txid: depositTx.getId(),
recoveryAddress,
wallet: recoveryWalletId,
};
const unsignedRecovery = (await sourceCoin.recoverFromWrongChain({
...params,
signed: false,
}));
should.equal(getRecoveryProviderStub.callCount, 1);
testMatchFixture('unsigned', () => unsignedRecovery);
it('should be signable for unsigned recovery', async function () {
const signedTx = await sourceCoin.signTransaction({
txPrebuild: unsignedRecovery,
prv: util_1.keychainsBase58[0].prv,
pubs: util_1.keychainsBase58.map((k) => k.pub),
});
checkRecoveryTransactionSignature(signedTx.txHex);
});
});
});
}
function isSupportedCrossChainRecovery(sourceCoin, recoveryCoin) {
return abstract_utxo_1.supportedCrossChainRecoveries[sourceCoin.getFamily()]?.includes(recoveryCoin.getFamily());
}
util_1.utxoCoins.forEach((coin) => {
util_1.utxoCoins
.filter((otherCoin) => coin !== otherCoin &&
isSupportedCrossChainRecovery(coin, otherCoin) &&
((utxolib.isMainnet(coin.network) && utxolib.isMainnet(otherCoin.network)) ||
(utxolib.isTestnet(coin.network) && utxolib.isTestnet(otherCoin.network))))
.forEach((otherCoin) => {
if (coin.amountType === 'bigint') {
run(coin, otherCoin);
}
else {
run(coin, otherCoin);
}
});
});
describe(`Cross-Chain Recovery getWallet`, async function () {
const bitgo = util_1.defaultBitGo;
const recoveryCoin = (0, util_1.getUtxoCoin)('btc');
const recoveryWalletId = '5abacebe28d72fbd07e0b8cbba0ff39e';
it('should search v1 wallets if the v2 endpoint responds with a 4xx error', async function () {
const errorResponses = [400, 404];
for (const error of errorResponses) {
const nockV2Wallet = (0, nockBitGo_1.nockBitGo)(bitgo)
.get(`/api/v2/${recoveryCoin.getChain()}/wallet/${recoveryWalletId}`)
.reply(error);
const nockV1Wallet = (0, nockBitGo_1.nockBitGo)(bitgo).get(`/api/v1/wallet/${recoveryWalletId}`).reply(error);
await assert.rejects(() => (0, abstract_utxo_1.getWallet)(bitgo, recoveryCoin, recoveryWalletId), Error(`could not get wallet ${recoveryWalletId} from v1 or v2: ApiResponseError: ${error}`));
nockV2Wallet.done();
nockV1Wallet.done();
}
});
it('should throw an error if the v2 endpoint responds with a 5xx error', async function () {
const errorResponses = [500];
for (const error of errorResponses) {
const nockV2Wallet = (0, nockBitGo_1.nockBitGo)(bitgo)
.get(`/api/v2/${recoveryCoin.getChain()}/wallet/${recoveryWalletId}`)
.reply(error);
await assert.rejects(() => (0, abstract_utxo_1.getWallet)(bitgo, recoveryCoin, recoveryWalletId), {
name: 'ApiResponseError',
status: 500,
result: {},
invalidToken: false,
needsOTP: false,
});
nockV2Wallet.done();
}
});
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crossChainRecovery.js","sourceRoot":"","sources":["../../../../../../../test/v2/unit/coins/utxo/recovery/crossChainRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as assert from 'assert';\nimport * as should from 'should';\nimport * as nock from 'nock';\nimport * as utxolib from '@bitgo/utxo-lib';\nimport { Triple } from '@bitgo/sdk-core';\nimport {\n  getFixture,\n  keychainsBase58,\n  KeychainBase58,\n  mockUnspent,\n  shouldEqualJSON,\n  utxoCoins,\n  transactionHexToObj,\n  getDefaultWalletKeys,\n  defaultBitGo,\n  getUtxoCoin,\n} from '../util';\nimport { getSeed } from '@bitgo/sdk-test';\nimport { nockBitGo } from '../util/nockBitGo';\nimport { createFullSignedTransaction } from '../util/transaction';\nimport { getDefaultWalletUnspentSigner } from '../util/keychains';\nimport { MockCrossChainRecoveryProvider } from './mock';\nimport {\n  AbstractUtxoCoin,\n  CrossChainRecoverySigned,\n  CrossChainRecoveryUnsigned,\n  getWallet,\n  supportedCrossChainRecoveries,\n} from '@bitgo/abstract-utxo';\nimport * as sinon from 'sinon';\n\ntype WalletUnspent<TNumber extends number | bigint = number> = utxolib.bitgo.WalletUnspent<TNumber>;\n\nfunction getKeyId(k: KeychainBase58): string {\n  return getSeed(k.pub).toString('hex');\n}\n\nfunction nockWallet(coin: AbstractUtxoCoin, walletId: string, walletKeys: Triple<KeychainBase58>): nock.Scope[] {\n  return [\n    nockBitGo()\n      .get(`/api/v2/${coin.getChain()}/wallet/${walletId}`)\n      .reply(200, {\n        id: walletId,\n        coin: coin.getChain(),\n        label: 'crossChainRecovery',\n        keys: walletKeys.map((k) => getKeyId(k)),\n      })\n      .persist(),\n    ...walletKeys.map((k) =>\n      nockBitGo()\n        .get(`/api/v2/${coin.getChain()}/key/${getKeyId(k)}`)\n        .reply(200, k)\n        .persist()\n    ),\n  ];\n}\n\ntype Address = {\n  address: string;\n  chain: number;\n  index: number;\n  coinSpecific: unknown;\n};\n\nfunction nockWalletAddress(coin: AbstractUtxoCoin, walletId: string, address: Address): nock.Scope {\n  return nockBitGo()\n    .get(`/api/v2/${coin.getChain()}/wallet/${walletId}/address/${address.address}`)\n    .reply(200, {\n      address: address.address,\n      chain: address.chain,\n      index: address.index,\n      coin: coin.getChain(),\n      wallet: walletId,\n      coinSpecific: address.coinSpecific,\n    })\n    .persist();\n}\n\n/**\n * Setup test for cross-chain recovery.\n *\n * Users can receive deposits on wallet addresses that are on a different chain.\n *\n * For instance, a user can receive litecoin on a bitcoin wallet.\n * This means that the litecoin blockchain has a transaction with outputs that are spendable\n * with keys that were originally created for a BitGo BTC wallet.\n * In this example, LTC is the \"source coin\" and BTC is the \"recovery coin\"\n * In cases like these we must use construct a transaction for litecoin network using keys of the\n * bitcoin wallet.\n *\n * @param sourceCoin - the coin to construct the transaction for\n * @param recoveryCoin - the coin the receiving wallet was set up for\n */\nfunction run<TNumber extends number | bigint = number>(sourceCoin: AbstractUtxoCoin, recoveryCoin: AbstractUtxoCoin) {\n  describe(`Cross-Chain Recovery [sourceCoin=${sourceCoin.getChain()} recoveryCoin=${recoveryCoin.getChain()}]`, function () {\n    const walletKeys = getDefaultWalletKeys();\n    const recoveryWalletId = '5abacebe28d72fbd07e0b8cbba0ff39e';\n    // the address the accidental deposit went to, in both sourceCoin and addressCoin formats\n    const [depositAddressSourceCoin, depositAddressRecoveryCoin] = [sourceCoin, recoveryCoin].map((coin) =>\n      coin.generateAddress({ keychains: keychainsBase58, index: 0 })\n    );\n    // the address where we want to recover our funds to\n    const recoveryAddress = sourceCoin.generateAddress({ keychains: keychainsBase58, index: 1 }).address;\n    const nocks: nock.Scope[] = [];\n\n    let depositTx: utxolib.bitgo.UtxoTransaction<TNumber>;\n\n    function getDepositUnspents(): utxolib.bitgo.Unspent<TNumber>[] {\n      return [\n        mockUnspent<TNumber>(\n          sourceCoin.network,\n          walletKeys,\n          'p2sh',\n          0,\n          (sourceCoin.amountType === 'bigint' ? BigInt('10999999800000001') : 1e8) as TNumber\n        ),\n      ];\n    }\n\n    function getDepositTransaction(): utxolib.bitgo.UtxoTransaction<TNumber> {\n      return createFullSignedTransaction<TNumber>(\n        sourceCoin.network,\n        getDepositUnspents(),\n        depositAddressSourceCoin.address,\n        getDefaultWalletUnspentSigner()\n      );\n    }\n\n    before('prepare deposit tx', function () {\n      depositTx = getDepositTransaction();\n    });\n\n    function getRecoveryUnspents(): WalletUnspent<TNumber>[] {\n      return [\n        {\n          id: depositTx.getId(),\n          address: depositAddressSourceCoin.address,\n          chain: depositAddressSourceCoin.chain as utxolib.bitgo.ChainCode,\n          index: depositAddressSourceCoin.index,\n          value: depositTx.outs[0].value,\n        },\n      ];\n    }\n\n    before('setup nocks', function () {\n      nocks.push(...nockWallet(recoveryCoin, recoveryWalletId, keychainsBase58));\n      nocks.push(nockWalletAddress(recoveryCoin, recoveryWalletId, depositAddressRecoveryCoin));\n    });\n\n    after(function () {\n      nocks.forEach((n) => n.done());\n    });\n\n    after(function () {\n      nock.cleanAll();\n    });\n\n    afterEach(function () {\n      sinon.restore();\n    });\n\n    function testMatchFixture(\n      name: string,\n      getRecoveryResult: () => CrossChainRecoverySigned<TNumber> | CrossChainRecoveryUnsigned<TNumber>\n    ) {\n      it(`should match fixture (${name})`, async function () {\n        const recovery = getRecoveryResult();\n        let recoveryObj = {\n          ...recovery,\n          tx: transactionHexToObj(recovery.txHex as string, sourceCoin.network, sourceCoin.amountType),\n        };\n        if (sourceCoin.amountType === 'bigint') {\n          recoveryObj = JSON.parse(\n            JSON.stringify(recoveryObj, (k, v) => {\n              if (typeof v === 'bigint') {\n                return v.toString();\n              } else {\n                return v;\n              }\n            })\n          );\n        }\n        shouldEqualJSON(\n          recoveryObj,\n          await getFixture(sourceCoin, `recovery/crossChainRecovery-${recoveryCoin.getChain()}-${name}`, recoveryObj)\n        );\n      });\n    }\n\n    function checkRecoveryTransactionSignature(tx: string | utxolib.bitgo.UtxoTransaction<TNumber>) {\n      if (typeof tx === 'string') {\n        tx = utxolib.bitgo.createTransactionFromBuffer<TNumber>(Buffer.from(tx, 'hex'), sourceCoin.network, {\n          amountType: sourceCoin.amountType,\n        });\n      }\n      const unspents = getRecoveryUnspents();\n      should.equal(tx.ins.length, unspents.length);\n      tx.ins.forEach((input, i) => {\n        assert(typeof tx !== 'string');\n        utxolib.bitgo\n          .verifySignatureWithUnspent<TNumber>(tx, i, getRecoveryUnspents(), walletKeys)\n          .should.eql([true, false, false]);\n      });\n    }\n\n    it('should test signed cross chain recovery', async () => {\n      const getRecoveryProviderStub = sinon\n        .stub(AbstractUtxoCoin.prototype, 'getRecoveryProvider')\n        .returns(new MockCrossChainRecoveryProvider<TNumber>(sourceCoin, getDepositUnspents(), depositTx));\n      const params = {\n        recoveryCoin,\n        txid: depositTx.getId(),\n        recoveryAddress,\n        wallet: recoveryWalletId,\n      };\n      const signedRecovery = (await sourceCoin.recoverFromWrongChain<TNumber>({\n        ...params,\n        xprv: keychainsBase58[0].prv,\n      })) as CrossChainRecoverySigned<TNumber>;\n      should.equal(getRecoveryProviderStub.callCount, 1);\n\n      testMatchFixture('signed', () => signedRecovery);\n\n      it('should have valid signatures for signed recovery', function () {\n        checkRecoveryTransactionSignature(signedRecovery.txHex as string);\n      });\n    });\n\n    it('should test unsigned cross chain recovery', async () => {\n      const getRecoveryProviderStub = sinon\n        .stub(AbstractUtxoCoin.prototype, 'getRecoveryProvider')\n        .returns(new MockCrossChainRecoveryProvider<TNumber>(sourceCoin, getDepositUnspents(), depositTx));\n      const params = {\n        recoveryCoin,\n        txid: depositTx.getId(),\n        recoveryAddress,\n        wallet: recoveryWalletId,\n      };\n      const unsignedRecovery = (await sourceCoin.recoverFromWrongChain<TNumber>({\n        ...params,\n        signed: false,\n      })) as CrossChainRecoveryUnsigned<TNumber>;\n      should.equal(getRecoveryProviderStub.callCount, 1);\n\n      testMatchFixture('unsigned', () => unsignedRecovery);\n\n      it('should be signable for unsigned recovery', async function () {\n        const signedTx = await sourceCoin.signTransaction<TNumber>({\n          txPrebuild: unsignedRecovery,\n          prv: keychainsBase58[0].prv,\n          pubs: keychainsBase58.map((k) => k.pub) as Triple<string>,\n        });\n        checkRecoveryTransactionSignature((signedTx as { txHex: string }).txHex);\n      });\n    });\n  });\n}\n\nfunction isSupportedCrossChainRecovery(sourceCoin: AbstractUtxoCoin, recoveryCoin: AbstractUtxoCoin): boolean {\n  return supportedCrossChainRecoveries[sourceCoin.getFamily()]?.includes(recoveryCoin.getFamily());\n}\n\nutxoCoins.forEach((coin) => {\n  utxoCoins\n    .filter(\n      (otherCoin) =>\n        coin !== otherCoin &&\n        isSupportedCrossChainRecovery(coin, otherCoin) &&\n        ((utxolib.isMainnet(coin.network) && utxolib.isMainnet(otherCoin.network)) ||\n          (utxolib.isTestnet(coin.network) && utxolib.isTestnet(otherCoin.network)))\n    )\n    .forEach((otherCoin) => {\n      if (coin.amountType === 'bigint') {\n        run<bigint>(coin, otherCoin);\n      } else {\n        run(coin, otherCoin);\n      }\n    });\n});\n\ndescribe(`Cross-Chain Recovery getWallet`, async function () {\n  const bitgo = defaultBitGo;\n  const recoveryCoin = getUtxoCoin('btc');\n  const recoveryWalletId = '5abacebe28d72fbd07e0b8cbba0ff39e';\n\n  it('should search v1 wallets if the v2 endpoint responds with a 4xx error', async function () {\n    const errorResponses = [400, 404];\n\n    for (const error of errorResponses) {\n      const nockV2Wallet = nockBitGo(bitgo)\n        .get(`/api/v2/${recoveryCoin.getChain()}/wallet/${recoveryWalletId}`)\n        .reply(error);\n      const nockV1Wallet = nockBitGo(bitgo).get(`/api/v1/wallet/${recoveryWalletId}`).reply(error);\n      await assert.rejects(\n        () => getWallet(bitgo, recoveryCoin, recoveryWalletId),\n        Error(`could not get wallet ${recoveryWalletId} from v1 or v2: ApiResponseError: ${error}`)\n      );\n      nockV2Wallet.done();\n      nockV1Wallet.done();\n    }\n  });\n\n  it('should throw an error if the v2 endpoint responds with a 5xx error', async function () {\n    const errorResponses = [500];\n    for (const error of errorResponses) {\n      const nockV2Wallet = nockBitGo(bitgo)\n        .get(`/api/v2/${recoveryCoin.getChain()}/wallet/${recoveryWalletId}`)\n        .reply(error);\n      await assert.rejects(() => getWallet(bitgo, recoveryCoin, recoveryWalletId), {\n        name: 'ApiResponseError',\n        status: 500,\n        result: {},\n        invalidToken: false,\n        needsOTP: false,\n      });\n      nockV2Wallet.done();\n    }\n  });\n});\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!