PHP WebShell
Текущая директория: /opt/BitGoJS/modules/bitgo/dist/test/v2/unit/coins/utxo
Просмотр файла: prebuildAndSign.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
/**
* @prettier
*/
const assert = require("assert");
const abstract_utxo_1 = require("@bitgo/abstract-utxo");
const utxolib = require("@bitgo/utxo-lib");
const nock = require("nock");
const util_1 = require("./util");
const sdk_core_1 = require("@bitgo/sdk-core");
const sdk_test_1 = require("@bitgo/sdk-test");
const src_1 = require("../../../../../src");
const txFormats = ['legacy', 'psbt'];
const walletPassphrase = 'gabagool';
const webauthnWalletPassPhrase = 'just the gabagool';
const scriptTypes = [...utxolib.bitgo.outputScripts.scriptTypes2Of3, 'taprootKeyPathSpend', 'p2shP2pk'];
function assertSignable(psbtHex, inputScripts, network) {
const psbt = utxolib.bitgo.createPsbtFromHex(psbtHex, network);
// Make sure that you can sign with bitgo key and extract the transaction
// No signatures should be present if it's a p2shP2pk input
if (!inputScripts.includes('p2shP2pk')) {
const key = inputScripts.includes('p2trMusig2') ? rootWalletKeys.backup : rootWalletKeys.bitgo;
psbt.signAllInputsHD(key, { deterministic: true });
psbt.validateSignaturesOfAllInputs();
psbt.finalizeAllInputs();
const tx = psbt.extractTransaction();
assert.ok(tx);
}
}
// Build the key objects
const rootWalletKeys = (0, util_1.getDefaultWalletKeys)();
const keyDocumentObjects = rootWalletKeys.triple.map((bip32, keyIdx) => {
return {
id: (0, sdk_test_1.getSeed)(util_1.keychainsBase58[keyIdx].pub).toString('hex'),
pub: bip32.neutered().toBase58(),
source: ['user', 'backup', 'bitgo'][keyIdx],
encryptedPrv: (0, util_1.encryptKeychain)(walletPassphrase, util_1.keychainsBase58[keyIdx]),
webauthnDevices: [
{
otpDeviceId: '123',
authenticatorInfo: {
credID: 'credID',
fmt: 'packed',
publicKey: 'some value',
},
prfSalt: '456',
encryptedPrv: (0, util_1.encryptKeychain)(webauthnWalletPassPhrase, util_1.keychainsBase58[keyIdx]),
},
],
coinSpecific: {},
};
});
function run(coin, inputScripts, txFormat) {
function createPrebuildPsbt(inputs, outputs) {
const psbt = utxolib.testutil.constructPsbt(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');
utxolib.bitgo.addXpubsToPsbt(psbt, rootWalletKeys);
return psbt;
}
function createNocks(params) {
const nocks = [];
// Nock the prebuild route (/tx/build, blockheight)
const expected_params = {
recipients: [params.recipient],
rbfTxIds: params.rbfTxIds,
feeMultiplier: params.feeMultiplier,
changeAddressType: ['p2trMusig2', 'p2wsh', 'p2shP2wsh', 'p2sh', 'p2tr'],
};
if (params.txFormat) {
expected_params['txFormat'] = params.txFormat;
}
nocks.push(nock(params.bgUrl)
.post(`/api/v2/${coin.getChain()}/wallet/${params.wallet.id()}/tx/build`, expected_params)
.reply(200, { txHex: params.prebuild.toHex(), txInfo: {} }));
nocks.push(nock(params.bgUrl).get(`/api/v2/${coin.getChain()}/public/block/latest`).reply(200, { height: 1000 }));
// nock the keychain fetch - 3 times (prebuildAndSign, verifyTransaction, and signTransaction)
params.keyDocuments.forEach((keyDocument) => {
nocks.push(nock(params.bgUrl).get(`/api/v2/${coin.getChain()}/key/${keyDocument.id}`).times(3).reply(200, keyDocument));
});
// nock the address info fetch
if (params.nockOutputAddresses) {
nocks.push(nock(params.bgUrl)
.get(`/api/v2/${coin.getChain()}/wallet/${params.wallet.id()}/address/${params.addressInfo.address}`)
.reply(200, params.addressInfo));
}
if (params.rbfTxIds) {
nocks.push(nock(params.bgUrl)
.get(`/api/v2/${coin.getChain()}/wallet/${params.wallet.id()}/tx/${params.rbfTxIds[0]}?includeRbf=true`)
.reply(200, {
outputs: [
{
address: params.recipient.address,
value: Number(params.recipient.amount),
valueString: params.recipient.amount,
wallet: params.selfSend ? params.wallet.id() : 'some-other-wallet-id', // external output if not a self send
},
// Dummy change output to test transfer entries filtering
{
address: params.recipient.address,
value: Number(params.recipient.amount),
valueString: params.recipient.amount,
wallet: params.wallet.id(), // internal output
},
],
}));
}
// nock the deterministic nonce response
if (inputScripts.includes('taprootKeyPathSpend')) {
const psbt = params.prebuild.clone();
psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);
psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);
nocks.push(nock(params.bgUrl)
.post(`/api/v2/${coin.getChain()}/wallet/${params.wallet.id()}/tx/signpsbt`, (body) => body.psbt)
.reply(200, { psbt: psbt.toHex() }));
}
return nocks;
}
describe(`${coin.getFullName()}-prebuildAndSign-txFormat=${txFormat}-inputScripts=${inputScripts.join(',')}`, function () {
const wallet = (0, util_1.getUtxoWallet)(coin, {
coinSpecific: { addressVersion: 'base58' },
keys: keyDocumentObjects.map((k) => k.id),
id: 'walletId',
});
const bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
const bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
let prebuild;
let recipient;
let addressInfo;
const fee = BigInt(10000);
before(async function () {
// Make output address information
const outputAmount = BigInt(inputScripts.length) * BigInt(1e8) - fee;
const outputScriptType = 'p2sh';
const outputChain = utxolib.bitgo.getExternalChainCode(outputScriptType);
const outputAddress = utxolib.bitgo.getWalletAddress(rootWalletKeys, outputChain, 0, coin.network);
recipient = {
address: outputAddress,
amount: outputAmount.toString(),
};
addressInfo = {
address: outputAddress,
chain: outputChain,
index: 0,
coin: coin.getChain(),
wallet: wallet.id(),
coinSpecific: {},
};
prebuild = createPrebuildPsbt(inputScripts.map((s) => ({ scriptType: s, value: BigInt(1e8) })), [{ scriptType: outputScriptType, value: outputAmount }]);
});
afterEach(nock.cleanAll);
[true, false].forEach((useWebauthn) => {
it(`should succeed with ${useWebauthn ? 'webauthn encryptedPrv' : 'encryptedPrv'}`, async function () {
const txCoins = ['tzec', 'zec', 'ltc', 'bcha', 'doge', 'dash', 'btg', 'bch'];
const nocks = createNocks({
bgUrl,
wallet,
keyDocuments: keyDocumentObjects,
prebuild,
recipient,
addressInfo,
nockOutputAddresses: txFormat !== 'psbt',
txFormat: !txCoins.includes(coin.getChain()) ? 'psbt' : undefined,
});
// call prebuild and sign, nocks should be consumed
const res = (await wallet.prebuildAndSignTransaction({
recipients: [recipient],
walletPassphrase: useWebauthn ? webauthnWalletPassPhrase : walletPassphrase,
}));
nocks.forEach((nock) => assert.ok(nock.isDone()));
assertSignable(res.txHex, inputScripts, coin.network);
});
it('should fail if the wallet passphrase is incorrect', async function () {
createNocks({
bgUrl,
wallet,
keyDocuments: keyDocumentObjects,
prebuild,
recipient,
addressInfo,
nockOutputAddresses: txFormat !== 'psbt',
});
await wallet
.prebuildAndSignTransaction({
recipients: [recipient],
walletPassphrase: Math.random().toString(),
})
.should.be.rejectedWith('unable to decrypt keychain with the given wallet passphrase');
});
});
[true, false].forEach((selfSend) => {
it(`should be able to build, sign, & verify a replacement transaction with selfSend: ${selfSend}`, async function () {
const rbfTxIds = ['tx-to-be-replaced'], feeMultiplier = 1.5;
const txCoins = ['tzec', 'zec', 'ltc', 'bcha', 'doge', 'dash', 'btg', 'bch'];
const nocks = createNocks({
bgUrl,
wallet,
keyDocuments: keyDocumentObjects,
prebuild,
recipient,
addressInfo,
rbfTxIds,
feeMultiplier,
selfSend,
nockOutputAddresses: txFormat !== 'psbt',
txFormat: !txCoins.includes(coin.getChain()) ? 'psbt' : undefined,
});
// call prebuild and sign, nocks should be consumed
const res = (await wallet.prebuildAndSignTransaction({
recipients: [recipient],
walletPassphrase,
rbfTxIds,
feeMultiplier,
}));
nocks.forEach((nock) => assert.ok(nock.isDone()));
assertSignable(res.txHex, inputScripts, coin.network);
});
});
});
}
util_1.utxoCoins
.filter((coin) => utxolib.getMainnet(coin.network) !== utxolib.networks.bitcoinsv)
.forEach((coin) => {
scriptTypes
// Don't iterate over p2shP2pk - in no scenario would a wallet spend two p2shP2pk inputs as these
// are single signature inputs that are used for replay protection and are added to the transaction
// by our system from a separate wallet. We do run tests below where one of the inputs is a p2shP2pk and
// the other is an input spent by the user.
.filter((scriptType) => scriptType !== 'p2shP2pk')
.forEach((inputScript) => {
const inputScriptCleaned = (inputScript === 'taprootKeyPathSpend' ? 'p2trMusig2' : inputScript);
if (!coin.supportsAddressType(inputScriptCleaned)) {
return;
}
run(coin, [inputScript, inputScript], 'psbt');
if ((0, abstract_utxo_1.getReplayProtectionAddresses)(coin.network).length) {
run(coin, ['p2shP2pk', inputScript], 'psbt');
}
});
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prebuildAndSign.js","sourceRoot":"","sources":["../../../../../../test/v2/unit/coins/utxo/prebuildAndSign.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as assert from 'assert';\n\nimport { AbstractUtxoCoin, getReplayProtectionAddresses } from '@bitgo/abstract-utxo';\nimport * as utxolib from '@bitgo/utxo-lib';\nimport * as nock from 'nock';\n\nimport { encryptKeychain, getDefaultWalletKeys, getUtxoWallet, keychainsBase58, utxoCoins } from './util';\nimport { common, HalfSignedUtxoTransaction, Wallet } from '@bitgo/sdk-core';\nimport { getSeed, TestBitGo } from '@bitgo/sdk-test';\nimport { BitGo } from '../../../../../src';\n\nconst txFormats = ['legacy', 'psbt'] as const;\nexport type TxFormat = (typeof txFormats)[number];\n\ntype KeyDoc = {\n  id: string;\n  pub: string;\n  source: string;\n  encryptedPrv: string;\n  coinSpecific: any;\n};\n\nconst walletPassphrase = 'gabagool';\nconst webauthnWalletPassPhrase = 'just the gabagool';\n\nconst scriptTypes = [...utxolib.bitgo.outputScripts.scriptTypes2Of3, 'taprootKeyPathSpend', 'p2shP2pk'] as const;\nexport type ScriptType = (typeof scriptTypes)[number];\n\ntype Input = {\n  scriptType: ScriptType;\n  value: bigint;\n};\n\nfunction assertSignable(psbtHex: string, inputScripts: ScriptType[], network: utxolib.Network): void {\n  const psbt = utxolib.bitgo.createPsbtFromHex(psbtHex, network);\n  // Make sure that you can sign with bitgo key and extract the transaction\n  // No signatures should be present if it's a p2shP2pk input\n  if (!inputScripts.includes('p2shP2pk')) {\n    const key = inputScripts.includes('p2trMusig2') ? rootWalletKeys.backup : rootWalletKeys.bitgo;\n    psbt.signAllInputsHD(key, { deterministic: true });\n    psbt.validateSignaturesOfAllInputs();\n    psbt.finalizeAllInputs();\n    const tx = psbt.extractTransaction();\n    assert.ok(tx);\n  }\n}\n\n// Build the key objects\nconst rootWalletKeys = getDefaultWalletKeys();\nconst keyDocumentObjects = rootWalletKeys.triple.map((bip32, keyIdx) => {\n  return {\n    id: getSeed(keychainsBase58[keyIdx].pub).toString('hex'),\n    pub: bip32.neutered().toBase58(),\n    source: ['user', 'backup', 'bitgo'][keyIdx],\n    encryptedPrv: encryptKeychain(walletPassphrase, keychainsBase58[keyIdx]),\n    webauthnDevices: [\n      {\n        otpDeviceId: '123',\n        authenticatorInfo: {\n          credID: 'credID',\n          fmt: 'packed',\n          publicKey: 'some value',\n        },\n        prfSalt: '456',\n        encryptedPrv: encryptKeychain(webauthnWalletPassPhrase, keychainsBase58[keyIdx]),\n      },\n    ],\n    coinSpecific: {},\n  };\n});\n\nfunction run(coin: AbstractUtxoCoin, inputScripts: ScriptType[], txFormat: TxFormat): void {\n  function createPrebuildPsbt(inputs: Input[], outputs: { scriptType: 'p2sh'; value: bigint }[]) {\n    const psbt = utxolib.testutil.constructPsbt(\n      inputs as utxolib.testutil.Input[],\n      outputs,\n      coin.network,\n      rootWalletKeys,\n      'unsigned'\n    );\n    utxolib.bitgo.addXpubsToPsbt(psbt, rootWalletKeys);\n    return psbt;\n  }\n\n  function createNocks(params: {\n    bgUrl: string;\n    wallet: Wallet;\n    keyDocuments: KeyDoc[];\n    prebuild: utxolib.bitgo.UtxoPsbt;\n    recipient: { address: string; amount: string };\n    addressInfo: Record<string, any>;\n    rbfTxIds?: string[];\n    feeMultiplier?: number;\n    selfSend?: boolean;\n    nockOutputAddresses?: boolean;\n    txFormat?: string;\n  }): nock.Scope[] {\n    const nocks: nock.Scope[] = [];\n\n    // Nock the prebuild route (/tx/build, blockheight)\n    const expected_params = {\n      recipients: [params.recipient],\n      rbfTxIds: params.rbfTxIds,\n      feeMultiplier: params.feeMultiplier,\n      changeAddressType: ['p2trMusig2', 'p2wsh', 'p2shP2wsh', 'p2sh', 'p2tr'],\n    };\n    if (params.txFormat) {\n      expected_params['txFormat'] = params.txFormat;\n    }\n    nocks.push(\n      nock(params.bgUrl)\n        .post(`/api/v2/${coin.getChain()}/wallet/${params.wallet.id()}/tx/build`, expected_params)\n        .reply(200, { txHex: params.prebuild.toHex(), txInfo: {} })\n    );\n    nocks.push(nock(params.bgUrl).get(`/api/v2/${coin.getChain()}/public/block/latest`).reply(200, { height: 1000 }));\n\n    // nock the keychain fetch - 3 times (prebuildAndSign, verifyTransaction, and signTransaction)\n    params.keyDocuments.forEach((keyDocument) => {\n      nocks.push(\n        nock(params.bgUrl).get(`/api/v2/${coin.getChain()}/key/${keyDocument.id}`).times(3).reply(200, keyDocument)\n      );\n    });\n\n    // nock the address info fetch\n    if (params.nockOutputAddresses) {\n      nocks.push(\n        nock(params.bgUrl)\n          .get(`/api/v2/${coin.getChain()}/wallet/${params.wallet.id()}/address/${params.addressInfo.address}`)\n          .reply(200, params.addressInfo)\n      );\n    }\n\n    if (params.rbfTxIds) {\n      nocks.push(\n        nock(params.bgUrl)\n          .get(`/api/v2/${coin.getChain()}/wallet/${params.wallet.id()}/tx/${params.rbfTxIds[0]}?includeRbf=true`)\n          .reply(200, {\n            outputs: [\n              {\n                address: params.recipient.address,\n                value: Number(params.recipient.amount),\n                valueString: params.recipient.amount,\n                wallet: params.selfSend ? params.wallet.id() : 'some-other-wallet-id', // external output if not a self send\n              },\n              // Dummy change output to test transfer entries filtering\n              {\n                address: params.recipient.address,\n                value: Number(params.recipient.amount),\n                valueString: params.recipient.amount,\n                wallet: params.wallet.id(), // internal output\n              },\n            ],\n          })\n      );\n    }\n\n    // nock the deterministic nonce response\n    if (inputScripts.includes('taprootKeyPathSpend')) {\n      const psbt = params.prebuild.clone();\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.user);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo);\n      nocks.push(\n        nock(params.bgUrl)\n          .post(`/api/v2/${coin.getChain()}/wallet/${params.wallet.id()}/tx/signpsbt`, (body) => body.psbt)\n          .reply(200, { psbt: psbt.toHex() })\n      );\n    }\n\n    return nocks;\n  }\n\n  describe(`${coin.getFullName()}-prebuildAndSign-txFormat=${txFormat}-inputScripts=${inputScripts.join(\n    ','\n  )}`, function () {\n    const wallet = getUtxoWallet(coin, {\n      coinSpecific: { addressVersion: 'base58' },\n      keys: keyDocumentObjects.map((k) => k.id),\n      id: 'walletId',\n    });\n\n    const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    const bgUrl = common.Environments[bitgo.getEnv()].uri;\n    let prebuild: utxolib.bitgo.UtxoPsbt;\n    let recipient: { address: string; amount: string };\n    let addressInfo: Record<string, any>;\n    const fee = BigInt(10000);\n\n    before(async function () {\n      // Make output address information\n      const outputAmount = BigInt(inputScripts.length) * BigInt(1e8) - fee;\n      const outputScriptType: utxolib.bitgo.outputScripts.ScriptType = 'p2sh';\n      const outputChain = utxolib.bitgo.getExternalChainCode(outputScriptType);\n      const outputAddress = utxolib.bitgo.getWalletAddress(rootWalletKeys, outputChain, 0, coin.network);\n\n      recipient = {\n        address: outputAddress,\n        amount: outputAmount.toString(),\n      };\n      addressInfo = {\n        address: outputAddress,\n        chain: outputChain,\n        index: 0,\n        coin: coin.getChain(),\n        wallet: wallet.id(),\n        coinSpecific: {},\n      };\n\n      prebuild = createPrebuildPsbt(\n        inputScripts.map((s) => ({ scriptType: s, value: BigInt(1e8) })),\n        [{ scriptType: outputScriptType, value: outputAmount }]\n      );\n    });\n\n    afterEach(nock.cleanAll);\n\n    [true, false].forEach((useWebauthn) => {\n      it(`should succeed with ${useWebauthn ? 'webauthn encryptedPrv' : 'encryptedPrv'}`, async function () {\n        const txCoins = ['tzec', 'zec', 'ltc', 'bcha', 'doge', 'dash', 'btg', 'bch'];\n        const nocks = createNocks({\n          bgUrl,\n          wallet,\n          keyDocuments: keyDocumentObjects,\n          prebuild,\n          recipient,\n          addressInfo,\n          nockOutputAddresses: txFormat !== 'psbt',\n          txFormat: !txCoins.includes(coin.getChain()) ? 'psbt' : undefined,\n        });\n\n        // call prebuild and sign, nocks should be consumed\n        const res = (await wallet.prebuildAndSignTransaction({\n          recipients: [recipient],\n          walletPassphrase: useWebauthn ? webauthnWalletPassPhrase : walletPassphrase,\n        })) as HalfSignedUtxoTransaction;\n\n        nocks.forEach((nock) => assert.ok(nock.isDone()));\n\n        assertSignable(res.txHex, inputScripts, coin.network);\n      });\n\n      it('should fail if the wallet passphrase is incorrect', async function () {\n        createNocks({\n          bgUrl,\n          wallet,\n          keyDocuments: keyDocumentObjects,\n          prebuild,\n          recipient,\n          addressInfo,\n          nockOutputAddresses: txFormat !== 'psbt',\n        });\n\n        await wallet\n          .prebuildAndSignTransaction({\n            recipients: [recipient],\n            walletPassphrase: Math.random().toString(),\n          })\n          .should.be.rejectedWith('unable to decrypt keychain with the given wallet passphrase');\n      });\n    });\n\n    [true, false].forEach((selfSend) => {\n      it(`should be able to build, sign, & verify a replacement transaction with selfSend: ${selfSend}`, async function () {\n        const rbfTxIds = ['tx-to-be-replaced'],\n          feeMultiplier = 1.5;\n        const txCoins = ['tzec', 'zec', 'ltc', 'bcha', 'doge', 'dash', 'btg', 'bch'];\n        const nocks = createNocks({\n          bgUrl,\n          wallet,\n          keyDocuments: keyDocumentObjects,\n          prebuild,\n          recipient,\n          addressInfo,\n          rbfTxIds,\n          feeMultiplier,\n          selfSend,\n          nockOutputAddresses: txFormat !== 'psbt',\n          txFormat: !txCoins.includes(coin.getChain()) ? 'psbt' : undefined,\n        });\n\n        // call prebuild and sign, nocks should be consumed\n        const res = (await wallet.prebuildAndSignTransaction({\n          recipients: [recipient],\n          walletPassphrase,\n          rbfTxIds,\n          feeMultiplier,\n        })) as HalfSignedUtxoTransaction;\n\n        nocks.forEach((nock) => assert.ok(nock.isDone()));\n\n        assertSignable(res.txHex, inputScripts, coin.network);\n      });\n    });\n  });\n}\n\nutxoCoins\n  .filter((coin) => utxolib.getMainnet(coin.network) !== utxolib.networks.bitcoinsv)\n  .forEach((coin) => {\n    scriptTypes\n      // Don't iterate over p2shP2pk - in no scenario would a wallet spend two p2shP2pk inputs as these\n      // are single signature inputs that are used for replay protection and are added to the transaction\n      // by our system from a separate wallet. We do run tests below where one of the inputs is a p2shP2pk and\n      // the other is an input spent by the user.\n      .filter((scriptType) => scriptType !== 'p2shP2pk')\n      .forEach((inputScript) => {\n        const inputScriptCleaned = (\n          inputScript === 'taprootKeyPathSpend' ? 'p2trMusig2' : inputScript\n        ) as utxolib.bitgo.outputScripts.ScriptType2Of3;\n\n        if (!coin.supportsAddressType(inputScriptCleaned)) {\n          return;\n        }\n\n        run(coin, [inputScript, inputScript], 'psbt');\n        if (getReplayProtectionAddresses(coin.network).length) {\n          run(coin, ['p2shP2pk', inputScript], 'psbt');\n        }\n      });\n  });\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!