PHP WebShell
Текущая директория: /opt/BitGoJS/modules/bitgo/dist/test/v2/unit/coins/utxo/recovery
Просмотр файла: backupKeyRecovery.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
/**
* @prettier
*/
require("should");
const sinon = require("sinon");
const nock = require("nock");
const utxolib = require("@bitgo/utxo-lib");
const { toOutput, outputScripts } = utxolib.bitgo;
const abstract_utxo_1 = require("@bitgo/abstract-utxo");
const util_1 = require("../util");
const mock_1 = require("./mock");
const sdk_core_1 = require("@bitgo/sdk-core");
const config = { krsProviders: sdk_core_1.krsProviders };
nock.disableNetConnect();
function configOverride(f) {
const backup = { ...sdk_core_1.krsProviders };
before(function () {
f(config);
});
after(function () {
Object.entries(backup).forEach(([k, v]) => {
config[k] = v;
});
});
}
const walletPassphrase = 'lol';
function getNamedKeys([userKey, backupKey, bitgoKey], password) {
function encode(k) {
return k.isNeutered() ? k.toBase58() : (0, util_1.encryptKeychain)(password, (0, util_1.toKeychainBase58)(k));
}
return {
userKey: encode(userKey),
backupKey: encode(backupKey),
bitgoKey: encode(bitgoKey),
};
}
function getKeysForUnsignedSweep([userKey, backupKey, bitgoKey], password) {
return getNamedKeys([userKey.neutered(), backupKey.neutered(), bitgoKey.neutered()], password);
}
function getKeysForKeyRecoveryService([userKey, backupKey, bitgoKey], password) {
return getNamedKeys([userKey, backupKey.neutered(), bitgoKey.neutered()], password);
}
function getKeysForFullSignedRecovery([userKey, backupKey, bitgoKey], password) {
return getNamedKeys([userKey, backupKey, bitgoKey.neutered()], password);
}
function getScriptTypes2Of3() {
return outputScripts.scriptTypes2Of3;
}
function run(coin, scriptType, walletKeys, params, tags = []) {
if (!coin.supportsAddressType(scriptType)) {
return;
}
describe(`Backup Key Recovery [${[coin.getChain(), ...tags, params.krsProvider].join(',')}]`, function () {
const externalWallet = (0, util_1.getWalletKeys)('external');
const recoveryDestination = (0, util_1.getWalletAddress)(coin.network, externalWallet);
let keyRecoveryServiceAddress;
let recovery;
let recoveryTx;
// 1e8 * 9e7 < 9.007e15 but 2e8 * 9e7 > 9.007e15 to test both code paths in queryBlockchainUnspentsPath
const valueMul = coin.amountType === 'bigint' ? BigInt(9e7) : BigInt(1);
const allUnspents = [
utxolib.testutil.toUnspent({ scriptType, value: BigInt(1e8) * valueMul }, 0, coin.network, walletKeys),
utxolib.testutil.toUnspent({ scriptType, value: BigInt(2e8) * valueMul }, 2, coin.network, walletKeys),
utxolib.testutil.toUnspent({ scriptType, value: BigInt(3e8) * valueMul }, 3, coin.network, walletKeys),
// this unspent will not be picked up due to the index gap
utxolib.testutil.toUnspent({ scriptType, value: BigInt(23e8) }, 23, coin.network, walletKeys),
];
const recoverUnspents = allUnspents.slice(0, -1);
// If the coin is bch, convert the mocked unspent address to cashaddr format since that is the format that blockchair
// returns on the /dashboards/addresses response
const mockedApiUnspents = coin.getChain() === 'bch' || coin.getChain() === 'bcha'
? recoverUnspents.map((u) => ({ ...u, address: coin.canonicalAddress(u.address, 'cashaddr').split(':')[1] }))
: recoverUnspents;
before('mock', function () {
sinon.stub(abstract_utxo_1.CoingeckoApi.prototype, 'getUSDPrice').resolves(69420);
});
configOverride(function (config) {
const configKrsProviders = { ...config.krsProviders };
configKrsProviders.dai.supportedCoins = [coin.getFamily()];
configKrsProviders.keyternal.supportedCoins = [coin.getFamily()];
keyRecoveryServiceAddress = (0, util_1.getWalletAddress)(coin.network, externalWallet, 0, 100);
configKrsProviders.keyternal.feeAddresses = { [coin.getChain()]: keyRecoveryServiceAddress };
config.krsProviders = configKrsProviders;
});
after(function () {
sinon.restore();
});
before('create recovery data', async function () {
recovery = await (0, abstract_utxo_1.backupKeyRecovery)(coin, util_1.defaultBitGo, {
walletPassphrase,
recoveryDestination,
scan: 5,
ignoreAddressTypes: [],
userKeyPath: params.userKeyPath,
krsProvider: params.krsProvider,
...params.keys,
recoveryProvider: new mock_1.MockRecoveryProvider(mockedApiUnspents),
});
const txHex = recovery.transactionHex ?? recovery.txHex;
const isPsbt = utxolib.bitgo.isPsbt(txHex);
recoveryTx = isPsbt
? utxolib.bitgo.createPsbtFromHex(txHex, coin.network)
: utxolib.bitgo.createTransactionFromHex(txHex, coin.network, coin.amountType);
recovery.txid =
recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().getId() : recoveryTx.getId();
});
it('matches fixture', async function () {
(0, util_1.shouldEqualJSON)(recovery, await (0, util_1.getFixture)(coin, `recovery/backupKeyRecovery-${(params.krsProvider ? tags.concat([params.krsProvider]) : tags).join('-')}`, recovery));
});
it('has expected input count', function () {
(recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.data.inputs : recoveryTx.ins).length.should.eql(recoverUnspents.length);
});
function checkInputsSignedBy(tx, rootKey, expectCount) {
if (tx instanceof utxolib.bitgo.UtxoPsbt) {
function validate(tx, inputIndex) {
try {
return tx.validateSignaturesOfInputHD(inputIndex, rootKey);
}
catch (e) {
if (e.message === 'No signatures to validate') {
return false;
}
throw e;
}
}
tx.data.inputs.forEach((input, inputIndex) => {
validate(tx, inputIndex).should.eql(!!expectCount);
});
}
else {
const prevOutputs = recoverUnspents
.map((u) => toOutput(u, coin.network))
.map((v) => ({ ...v, value: utxolib.bitgo.toTNumber(v.value, coin.amountType) }));
tx.ins.forEach((input, inputIndex) => {
const unspent = recoverUnspents[inputIndex];
const { publicKey } = rootKey.derivePath(walletKeys.getDerivationPath(rootKey, unspent.chain, unspent.index));
const signatures = utxolib.bitgo
.getSignatureVerifications(tx, inputIndex, utxolib.bitgo.toTNumber(unspent.value, coin.amountType), { publicKey }, prevOutputs)
.filter((s) => s.signedBy !== undefined);
signatures.length.should.eql(expectCount);
});
}
}
it((params.hasUserSignature ? 'has' : 'has no') + ' user signature', function () {
checkInputsSignedBy(recoveryTx, walletKeys.user, params.hasUserSignature ? 1 : 0);
});
it((params.hasBackupSignature ? 'has' : 'has no') + ' backup signature', function () {
checkInputsSignedBy(recoveryTx, walletKeys.backup, params.hasBackupSignature ? 1 : 0);
});
if (params.hasUserSignature && params.hasBackupSignature) {
it('has no placeholder signatures', function () {
if (recoveryTx instanceof utxolib.bitgo.UtxoTransaction) {
recoveryTx.ins.forEach((input) => {
const parsed = utxolib.bitgo.parseSignatureScript(input);
switch (parsed.scriptType) {
case 'p2sh':
case 'p2shP2wsh':
case 'p2wsh':
case 'taprootScriptPathSpend':
parsed.signatures.forEach((signature, i) => {
if (utxolib.bitgo.isPlaceholderSignature(signature)) {
throw new Error(`placeholder signature at index ${i}`);
}
});
break;
default:
throw new Error(`unexpected scriptType ${scriptType}`);
}
});
}
else {
this.skip();
}
});
}
it((params.hasKrsOutput ? 'has' : 'has no') + ' key recovery service output', function () {
const outs = recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().outs : recoveryTx.outs;
outs.length.should.eql(1);
const outputAddresses = outs.map((o) => utxolib.address.fromOutputScript(o.script, recoveryTx.network));
outputAddresses
.includes(keyRecoveryServiceAddress)
.should.eql(!!params.hasKrsOutput && params.krsProvider === 'keyternal');
outputAddresses.includes(recoveryDestination).should.eql(true);
});
});
}
util_1.utxoCoins.forEach((coin) => {
const walletKeys = (0, util_1.getDefaultWalletKeys)();
getScriptTypes2Of3().forEach((scriptType) => {
run(coin, scriptType, walletKeys, {
keys: getKeysForUnsignedSweep(walletKeys.triple, walletPassphrase),
hasUserSignature: false,
hasBackupSignature: false,
}, [scriptType, 'unsignedRecovery']);
['dai', 'keyternal'].forEach((krsProvider) => {
if (krsProvider === 'keyternal' && !['p2sh', 'p2wsh', 'p2shP2wsh'].includes(scriptType)) {
return;
}
run(coin, scriptType, walletKeys, {
keys: getKeysForKeyRecoveryService(walletKeys.triple, walletPassphrase),
krsProvider: krsProvider,
hasUserSignature: true,
hasBackupSignature: false,
hasKrsOutput: false,
}, [scriptType, 'keyRecoveryService']);
});
run(coin, scriptType, walletKeys, {
keys: getKeysForFullSignedRecovery(walletKeys.triple, walletPassphrase),
hasUserSignature: true,
hasBackupSignature: true,
}, [scriptType, 'fullSignedRecovery']);
run(coin, scriptType, walletKeys, {
keys: getKeysForFullSignedRecovery(walletKeys.triple, walletPassphrase),
hasUserSignature: true,
hasBackupSignature: true,
feeRate: 2,
}, [scriptType, 'fullSignedRecovery', 'fixedFeeRate']);
{
const userKeyPath = '99/99';
const exoticWalletKeys = new utxolib.bitgo.RootWalletKeys(util_1.keychains, [
userKeyPath,
utxolib.bitgo.RootWalletKeys.defaultPrefix,
utxolib.bitgo.RootWalletKeys.defaultPrefix,
]);
run(coin, scriptType, exoticWalletKeys, {
keys: getKeysForFullSignedRecovery(exoticWalletKeys.triple, walletPassphrase),
userKeyPath,
hasUserSignature: true,
hasBackupSignature: true,
}, [scriptType, 'fullSignedRecovery', 'customUserKeyPath']);
}
});
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"backupKeyRecovery.js","sourceRoot":"","sources":["../../../../../../../test/v2/unit/coins/utxo/recovery/backupKeyRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport 'should';\nimport * as mocha from 'mocha';\nimport * as sinon from 'sinon';\nimport * as nock from 'nock';\nimport { BIP32Interface } from '@bitgo/utxo-lib';\n\nimport * as utxolib from '@bitgo/utxo-lib';\nconst { toOutput, outputScripts } = utxolib.bitgo;\ntype WalletUnspent = utxolib.bitgo.WalletUnspent<bigint>;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\ntype ScriptType2Of3 = utxolib.bitgo.outputScripts.ScriptType2Of3;\n\nimport { Config } from '../../../../../../src/config';\nimport {\n  AbstractUtxoCoin,\n  backupKeyRecovery,\n  BackupKeyRecoveryTransansaction,\n  CoingeckoApi,\n  FormattedOfflineVaultTxInfo,\n} from '@bitgo/abstract-utxo';\n\nimport {\n  defaultBitGo,\n  encryptKeychain,\n  getDefaultWalletKeys,\n  getFixture,\n  getWalletAddress,\n  getWalletKeys,\n  keychains,\n  shouldEqualJSON,\n  toKeychainBase58,\n  utxoCoins,\n} from '../util';\n\nimport { MockRecoveryProvider } from './mock';\nimport { krsProviders, Triple } from '@bitgo/sdk-core';\n\nconst config = { krsProviders };\n\nnock.disableNetConnect();\n\nfunction configOverride(f: (config: Config) => void) {\n  const backup = { ...krsProviders };\n  before(function () {\n    f(config);\n  });\n  after(function () {\n    Object.entries(backup).forEach(([k, v]) => {\n      config[k] = v;\n    });\n  });\n}\n\nconst walletPassphrase = 'lol';\n\ntype NamedKeys = {\n  userKey: string;\n  backupKey: string;\n  bitgoKey: string;\n};\n\nfunction getNamedKeys([userKey, backupKey, bitgoKey]: Triple<BIP32Interface>, password: string): NamedKeys {\n  function encode(k: BIP32Interface): string {\n    return k.isNeutered() ? k.toBase58() : encryptKeychain(password, toKeychainBase58(k));\n  }\n  return {\n    userKey: encode(userKey),\n    backupKey: encode(backupKey),\n    bitgoKey: encode(bitgoKey),\n  };\n}\n\nfunction getKeysForUnsignedSweep([userKey, backupKey, bitgoKey]: Triple<BIP32Interface>, password: string): NamedKeys {\n  return getNamedKeys([userKey.neutered(), backupKey.neutered(), bitgoKey.neutered()], password);\n}\n\nfunction getKeysForKeyRecoveryService(\n  [userKey, backupKey, bitgoKey]: Triple<BIP32Interface>,\n  password: string\n): NamedKeys {\n  return getNamedKeys([userKey, backupKey.neutered(), bitgoKey.neutered()], password);\n}\n\nfunction getKeysForFullSignedRecovery(\n  [userKey, backupKey, bitgoKey]: Triple<BIP32Interface>,\n  password: string\n): NamedKeys {\n  return getNamedKeys([userKey, backupKey, bitgoKey.neutered()], password);\n}\n\nfunction getScriptTypes2Of3() {\n  return outputScripts.scriptTypes2Of3;\n}\n\nfunction run(\n  coin: AbstractUtxoCoin,\n  scriptType: ScriptType2Of3,\n  walletKeys: RootWalletKeys,\n  params: {\n    keys: NamedKeys;\n    userKeyPath?: string;\n    krsProvider?: string;\n    hasUserSignature: boolean;\n    hasBackupSignature: boolean;\n    hasKrsOutput?: boolean;\n    feeRate?: number;\n  },\n  tags: string[] = []\n) {\n  if (!coin.supportsAddressType(scriptType)) {\n    return;\n  }\n\n  describe(`Backup Key Recovery [${[coin.getChain(), ...tags, params.krsProvider].join(',')}]`, function () {\n    const externalWallet = getWalletKeys('external');\n    const recoveryDestination = getWalletAddress(coin.network, externalWallet);\n\n    let keyRecoveryServiceAddress: string;\n    let recovery: (BackupKeyRecoveryTransansaction | FormattedOfflineVaultTxInfo) & { txid?: string };\n    let recoveryTx: utxolib.bitgo.UtxoTransaction<number | bigint> | utxolib.bitgo.UtxoPsbt;\n\n    // 1e8 * 9e7 < 9.007e15 but 2e8 * 9e7 > 9.007e15 to test both code paths in queryBlockchainUnspentsPath\n    const valueMul = coin.amountType === 'bigint' ? BigInt(9e7) : BigInt(1);\n    const allUnspents = [\n      utxolib.testutil.toUnspent({ scriptType, value: BigInt(1e8) * valueMul }, 0, coin.network, walletKeys),\n      utxolib.testutil.toUnspent({ scriptType, value: BigInt(2e8) * valueMul }, 2, coin.network, walletKeys),\n      utxolib.testutil.toUnspent({ scriptType, value: BigInt(3e8) * valueMul }, 3, coin.network, walletKeys),\n      // this unspent will not be picked up due to the index gap\n      utxolib.testutil.toUnspent({ scriptType, value: BigInt(23e8) }, 23, coin.network, walletKeys),\n    ];\n\n    const recoverUnspents = allUnspents.slice(0, -1);\n\n    // If the coin is bch, convert the mocked unspent address to cashaddr format since that is the format that blockchair\n    // returns on the /dashboards/addresses response\n    const mockedApiUnspents =\n      coin.getChain() === 'bch' || coin.getChain() === 'bcha'\n        ? recoverUnspents.map((u) => ({ ...u, address: coin.canonicalAddress(u.address, 'cashaddr').split(':')[1] }))\n        : recoverUnspents;\n\n    before('mock', function () {\n      sinon.stub(CoingeckoApi.prototype, 'getUSDPrice').resolves(69_420);\n    });\n\n    configOverride(function (config: Config) {\n      const configKrsProviders = { ...config.krsProviders };\n      configKrsProviders.dai.supportedCoins = [coin.getFamily()];\n      configKrsProviders.keyternal.supportedCoins = [coin.getFamily()];\n      keyRecoveryServiceAddress = getWalletAddress(coin.network, externalWallet, 0, 100);\n      configKrsProviders.keyternal.feeAddresses = { [coin.getChain()]: keyRecoveryServiceAddress };\n      config.krsProviders = configKrsProviders;\n    });\n\n    after(function () {\n      sinon.restore();\n    });\n\n    before('create recovery data', async function () {\n      recovery = await backupKeyRecovery(coin, defaultBitGo, {\n        walletPassphrase,\n        recoveryDestination,\n        scan: 5,\n        ignoreAddressTypes: [],\n        userKeyPath: params.userKeyPath,\n        krsProvider: params.krsProvider,\n        ...params.keys,\n        recoveryProvider: new MockRecoveryProvider(mockedApiUnspents),\n      });\n      const txHex =\n        (recovery as BackupKeyRecoveryTransansaction).transactionHex ?? (recovery as FormattedOfflineVaultTxInfo).txHex;\n      const isPsbt = utxolib.bitgo.isPsbt(txHex);\n      recoveryTx = isPsbt\n        ? utxolib.bitgo.createPsbtFromHex(txHex, coin.network)\n        : utxolib.bitgo.createTransactionFromHex(txHex as string, coin.network, coin.amountType);\n      recovery.txid =\n        recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().getId() : recoveryTx.getId();\n    });\n\n    it('matches fixture', async function () {\n      shouldEqualJSON(\n        recovery,\n        await getFixture(\n          coin,\n          `recovery/backupKeyRecovery-${(params.krsProvider ? tags.concat([params.krsProvider]) : tags).join('-')}`,\n          recovery\n        )\n      );\n    });\n\n    it('has expected input count', function () {\n      (recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.data.inputs : recoveryTx.ins).length.should.eql(\n        recoverUnspents.length\n      );\n    });\n\n    function checkInputsSignedBy(\n      tx: utxolib.bitgo.UtxoTransaction<number | bigint> | utxolib.bitgo.UtxoPsbt,\n      rootKey: BIP32Interface,\n      expectCount: number\n    ) {\n      if (tx instanceof utxolib.bitgo.UtxoPsbt) {\n        function validate(tx: utxolib.bitgo.UtxoPsbt, inputIndex: number) {\n          try {\n            return tx.validateSignaturesOfInputHD(inputIndex, rootKey);\n          } catch (e) {\n            if (e.message === 'No signatures to validate') {\n              return false;\n            }\n            throw e;\n          }\n        }\n        tx.data.inputs.forEach((input, inputIndex) => {\n          validate(tx, inputIndex).should.eql(!!expectCount);\n        });\n      } else {\n        const prevOutputs = recoverUnspents\n          .map((u) => toOutput(u, coin.network))\n          .map((v) => ({ ...v, value: utxolib.bitgo.toTNumber(v.value, coin.amountType) }));\n        tx.ins.forEach((input, inputIndex) => {\n          const unspent = recoverUnspents[inputIndex] as WalletUnspent;\n          const { publicKey } = rootKey.derivePath(walletKeys.getDerivationPath(rootKey, unspent.chain, unspent.index));\n          const signatures = utxolib.bitgo\n            .getSignatureVerifications(\n              tx,\n              inputIndex,\n              utxolib.bitgo.toTNumber(unspent.value, coin.amountType),\n              { publicKey },\n              prevOutputs\n            )\n            .filter((s) => s.signedBy !== undefined);\n          signatures.length.should.eql(expectCount);\n        });\n      }\n    }\n\n    it((params.hasUserSignature ? 'has' : 'has no') + ' user signature', function () {\n      checkInputsSignedBy(recoveryTx, walletKeys.user, params.hasUserSignature ? 1 : 0);\n    });\n\n    it((params.hasBackupSignature ? 'has' : 'has no') + ' backup signature', function () {\n      checkInputsSignedBy(recoveryTx, walletKeys.backup, params.hasBackupSignature ? 1 : 0);\n    });\n\n    if (params.hasUserSignature && params.hasBackupSignature) {\n      it('has no placeholder signatures', function (this: mocha.Context) {\n        if (recoveryTx instanceof utxolib.bitgo.UtxoTransaction) {\n          recoveryTx.ins.forEach((input) => {\n            const parsed = utxolib.bitgo.parseSignatureScript(input);\n            switch (parsed.scriptType) {\n              case 'p2sh':\n              case 'p2shP2wsh':\n              case 'p2wsh':\n              case 'taprootScriptPathSpend':\n                parsed.signatures.forEach((signature, i) => {\n                  if (utxolib.bitgo.isPlaceholderSignature(signature)) {\n                    throw new Error(`placeholder signature at index ${i}`);\n                  }\n                });\n                break;\n              default:\n                throw new Error(`unexpected scriptType ${scriptType}`);\n            }\n          });\n        } else {\n          this.skip();\n        }\n      });\n    }\n\n    it((params.hasKrsOutput ? 'has' : 'has no') + ' key recovery service output', function () {\n      const outs = recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().outs : recoveryTx.outs;\n      outs.length.should.eql(1);\n      const outputAddresses = outs.map((o) => utxolib.address.fromOutputScript(o.script, recoveryTx.network));\n      outputAddresses\n        .includes(keyRecoveryServiceAddress)\n        .should.eql(!!params.hasKrsOutput && params.krsProvider === 'keyternal');\n      outputAddresses.includes(recoveryDestination).should.eql(true);\n    });\n  });\n}\n\nutxoCoins.forEach((coin) => {\n  const walletKeys = getDefaultWalletKeys();\n\n  getScriptTypes2Of3().forEach((scriptType) => {\n    run(\n      coin,\n      scriptType,\n      walletKeys,\n      {\n        keys: getKeysForUnsignedSweep(walletKeys.triple, walletPassphrase),\n        hasUserSignature: false,\n        hasBackupSignature: false,\n      },\n      [scriptType, 'unsignedRecovery']\n    );\n\n    ['dai', 'keyternal'].forEach((krsProvider) => {\n      if (krsProvider === 'keyternal' && !['p2sh', 'p2wsh', 'p2shP2wsh'].includes(scriptType)) {\n        return;\n      }\n      run(\n        coin,\n        scriptType,\n        walletKeys,\n        {\n          keys: getKeysForKeyRecoveryService(walletKeys.triple, walletPassphrase),\n          krsProvider: krsProvider,\n          hasUserSignature: true,\n          hasBackupSignature: false,\n          hasKrsOutput: false,\n        },\n        [scriptType, 'keyRecoveryService']\n      );\n    });\n\n    run(\n      coin,\n      scriptType,\n      walletKeys,\n      {\n        keys: getKeysForFullSignedRecovery(walletKeys.triple, walletPassphrase),\n        hasUserSignature: true,\n        hasBackupSignature: true,\n      },\n      [scriptType, 'fullSignedRecovery']\n    );\n\n    run(\n      coin,\n      scriptType,\n      walletKeys,\n      {\n        keys: getKeysForFullSignedRecovery(walletKeys.triple, walletPassphrase),\n        hasUserSignature: true,\n        hasBackupSignature: true,\n        feeRate: 2,\n      },\n      [scriptType, 'fullSignedRecovery', 'fixedFeeRate']\n    );\n\n    {\n      const userKeyPath = '99/99';\n      const exoticWalletKeys = new utxolib.bitgo.RootWalletKeys(keychains, [\n        userKeyPath,\n        utxolib.bitgo.RootWalletKeys.defaultPrefix,\n        utxolib.bitgo.RootWalletKeys.defaultPrefix,\n      ]);\n\n      run(\n        coin,\n        scriptType,\n        exoticWalletKeys,\n        {\n          keys: getKeysForFullSignedRecovery(exoticWalletKeys.triple, walletPassphrase),\n          userKeyPath,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n        },\n        [scriptType, 'fullSignedRecovery', 'customUserKeyPath']\n      );\n    }\n  });\n});\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!