PHP WebShell

Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/abstract-utxo/dist/src/transaction/fixedScript

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

"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.parseOutput = parseOutput;
const debug_1 = __importDefault(require("debug"));
const lodash_1 = __importDefault(require("lodash"));
const sdk_core_1 = require("@bitgo/sdk-core");
const abstractUtxoCoin_1 = require("../../abstractUtxoCoin");
const debug = (0, debug_1.default)('bitgo:v2:parseoutput');
/**
 * Check an address which failed initial validation to see if it's the base address of a migrated v1 bch wallet.
 *
 * The wallet in question could be a migrated SafeHD BCH wallet, and the transaction we
 * are currently parsing is trying to spend change back to the v1 wallet base address.
 *
 * It does this since we don't allow new address creation for these wallets,
 * and instead return the base address from the v1 wallet when a new address is requested.
 * If this new address is requested for the purposes of spending change back to the wallet,
 * the change will go to the v1 wallet base address. This address *is* on the wallet,
 * but it will still cause an error to be thrown by verifyAddress, since the derivation path
 * used for this address is non-standard. (I have seen these addresses derived using paths m/0/0 and m/101,
 * whereas the v2 addresses are derived using path  m/0/0/${chain}/${index}).
 *
 * This means we need to check for this case explicitly in this catch block, and classify
 * these types of outputs as internal instead of external. Failing to do so would cause the
 * transaction's implicit external outputs (ie, outputs which go to addresses not specified in
 * the recipients array) to add up to more than the 150 basis point limit which we enforce on
 * pay-as-you-go outputs (which should be the only implicit external outputs on our transactions).
 *
 * The 150 basis point limit for implicit external sends is enforced in verifyTransaction,
 * which calls this function to get information on the total external/internal spend amounts
 * for a transaction. The idea here is to protect from the transaction being maliciously modified
 * to add more implicit external spends (eg, to an attacker-controlled wallet).
 *
 * See verifyTransaction for more information on how transaction prebuilds are verified before signing.
 *
 * @param wallet {Wallet} wallet which is making the transaction
 * @param currentAddress {string} address to check for externality relative to v1 wallet base address
 */
function isMigratedAddress(wallet, currentAddress) {
    if (lodash_1.default.isString(wallet.migratedFrom()) && wallet.migratedFrom() === currentAddress) {
        debug('found address %s which was migrated from v1 wallet, address is not external', currentAddress);
        return true;
    }
    return false;
}
/**
 * Check to see if an address is derived from the given custom change keys
 * @param {VerifyCustomChangeAddressOptions} params
 * @return {boolean}
 */
async function verifyCustomChangeAddress(params) {
    const { coin, customChangeKeys, addressType, addressDetails, currentAddress } = params;
    try {
        return await coin.verifyAddress(lodash_1.default.extend({ addressType }, addressDetails, {
            keychains: customChangeKeys,
            address: currentAddress,
        }));
    }
    catch (e) {
        debug('failed to verify custom change address %s', currentAddress);
        return false;
    }
}
async function handleVerifyAddressError({ e, currentAddress, wallet, txParams, customChangeKeys, coin, addressDetails, addressType, considerMigratedFromAddressInternal, }) {
    // Todo: name server-side errors to avoid message-based checking [BG-5124]
    const walletAddressNotFound = e.message.includes('wallet address not found');
    const unexpectedAddress = e instanceof sdk_core_1.UnexpectedAddressError;
    if (walletAddressNotFound || unexpectedAddress) {
        if (unexpectedAddress && !walletAddressNotFound) {
            // check to see if this is a migrated v1 bch address - it could be internal
            const isMigrated = isMigratedAddress(wallet, currentAddress);
            if (isMigrated) {
                return { external: considerMigratedFromAddressInternal === false };
            }
            debug('Address %s was found on wallet but could not be reconstructed', currentAddress);
            // attempt to verify address using custom change address keys if the wallet has that feature enabled
            if (customChangeKeys &&
                (await verifyCustomChangeAddress({ coin, addressDetails, addressType, currentAddress, customChangeKeys }))) {
                // address is valid against the custom change keys. Mark address as not external
                // and request signature verification for the custom change keys
                debug('Address %s verified as derived from the custom change keys', currentAddress);
                return { external: false, needsCustomChangeKeySignatureVerification: true };
            }
        }
        // the address was found, but not on the wallet, which simply means it's external
        debug('Address %s presumed external', currentAddress);
        return { external: true };
    }
    else if (e instanceof sdk_core_1.InvalidAddressDerivationPropertyError && currentAddress === txParams.changeAddress) {
        // expect to see this error when passing in a custom changeAddress with no chain or index
        return { external: false };
    }
    console.error('Address classification failed for address', currentAddress);
    console.trace(e);
    /**
     * It might be a completely invalid address or a bad validation attempt or something else completely, in
     * which case we do not proceed and rather rethrow the error, which is safer than assuming that the address
     * validation failed simply because it's external to the wallet.
     */
    throw e;
}
async function fetchAddressDetails({ reqId, disableNetworking, addressDetailsPrebuild, addressDetailsVerification, currentAddress, wallet, }) {
    let addressDetails = lodash_1.default.extend({}, addressDetailsPrebuild, addressDetailsVerification);
    debug('Locally available address %s details: %O', currentAddress, addressDetails);
    if (lodash_1.default.isEmpty(addressDetails) && !disableNetworking) {
        addressDetails = await wallet.getAddress({ address: currentAddress, reqId });
        debug('Downloaded address %s details: %O', currentAddress, addressDetails);
    }
    return addressDetails;
}
async function parseOutput({ currentOutput, coin, txPrebuild, verification, keychainArray, wallet, txParams, customChange, reqId, }) {
    const disableNetworking = !!verification.disableNetworking;
    const currentAddress = currentOutput.address;
    if (currentAddress === undefined) {
        // In the case that the address is undefined, it means that the output has a non-encodeable scriptPubkey
        // If this is the case, then we need to check that the amount is 0 and we can skip the rest.
        if (currentOutput.amount.toString() !== '0') {
            throw new Error('output with undefined address must have amount of 0');
        }
        return currentOutput;
    }
    // attempt to grab the address details from either the prebuilt tx, or the verification params.
    // If both of these are empty, then we will try to get the address details from bitgo instead
    const addressDetailsPrebuild = lodash_1.default.get(txPrebuild, `txInfo.walletAddressDetails.${currentAddress}`, {});
    const addressDetailsVerification = verification?.addresses?.[currentAddress] ?? {};
    debug('Parsing address details for %s', currentAddress);
    let currentAddressDetails = undefined;
    let currentAddressType = undefined;
    const RECIPIENT_THRESHOLD = 1000;
    try {
        // In the case of PSBTs, we can already determine the internal/external status of the output addresses
        // based on the derivation information being included in the PSBT. We can short circuit GET v2.wallet.address
        // and save on network requests. Since we have the derivation information already, we can still verify the address
        if (currentOutput.external !== undefined) {
            // In the case that we have a custom change wallet, we need to verify the address against the custom change keys
            // and not the wallet keys. This check is done in the handleVerifyAddressError function if this error is thrown.
            if (customChange !== undefined) {
                throw new sdk_core_1.UnexpectedAddressError('`address validation failure');
            }
            // If it is an internal address, we can skip the network request and just verify the address locally with the
            // derivation information we have. Otherwise, if the address is external, which is the only remaining case, we
            // can just return the current output as is without contacting the server.
            if ((0, abstractUtxoCoin_1.isWalletOutput)(currentOutput)) {
                const res = await coin.isWalletAddress({
                    addressType: abstractUtxoCoin_1.AbstractUtxoCoin.inferAddressType({ chain: currentOutput.chain }) || undefined,
                    keychains: keychainArray,
                    address: currentAddress,
                    chain: currentOutput.chain,
                    index: currentOutput.index,
                });
                if (!res) {
                    throw new sdk_core_1.UnexpectedAddressError();
                }
            }
            return currentOutput;
        }
        /**
         * The only way to determine whether an address is known on the wallet is to initiate a network request and
         * fetch it. Should the request fail and return a 404, it will throw and therefore has to be caught. For that
         * reason, address wallet ownership detection is wrapped in a try/catch. Additionally, once the address
         * details are fetched on the wallet, a local address validation is run, whose errors however are generated
         * client-side and can therefore be analyzed with more granularity and type checking.
         */
        /**
         * In order to minimize API requests, we assume that explicit recipients are always external when the
         * recipient list is > 1000 This is not always a valid assumption and could lead greater apparent spend (but never lower)
         */
        if (txParams.recipients !== undefined && txParams.recipients.length > RECIPIENT_THRESHOLD) {
            const isCurrentAddressInRecipients = txParams.recipients.some((recipient) => recipient.address.includes(currentAddress));
            if (isCurrentAddressInRecipients) {
                return { ...currentOutput };
            }
        }
        const addressDetails = await fetchAddressDetails({
            reqId,
            addressDetailsVerification,
            addressDetailsPrebuild,
            currentAddress,
            disableNetworking,
            wallet,
        });
        // verify that the address is on the wallet. verifyAddress throws if
        // it fails to correctly rederive the address, meaning it's external
        currentAddressType = abstractUtxoCoin_1.AbstractUtxoCoin.inferAddressType(addressDetails) || undefined;
        currentAddressDetails = addressDetails;
        await coin.verifyAddress(lodash_1.default.extend({ addressType: currentAddressType }, addressDetails, {
            keychains: keychainArray,
            address: currentAddress,
        }));
        debug('Address %s verification passed', currentAddress);
        // verify address succeeded without throwing, so the address was
        // correctly rederived from the wallet keychains, making it not external
        return lodash_1.default.extend({}, currentOutput, addressDetails, { external: false });
    }
    catch (e) {
        debug('Address %s verification threw an error:', currentAddress, e);
        return lodash_1.default.extend({}, currentOutput, await handleVerifyAddressError({
            e,
            coin,
            currentAddress,
            wallet,
            txParams,
            customChangeKeys: customChange && customChange.keys,
            addressDetails: currentAddressDetails,
            addressType: currentAddressType,
            considerMigratedFromAddressInternal: verification.considerMigratedFromAddressInternal,
        }));
    }
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parseOutput.js","sourceRoot":"","sources":["../../../../src/transaction/fixedScript/parseOutput.ts"],"names":[],"mappings":"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","sourcesContent":["import debugLib from 'debug';\nimport _ from 'lodash';\nimport {\n  AddressVerificationData,\n  IRequestTracer,\n  InvalidAddressDerivationPropertyError,\n  IWallet,\n  TransactionPrebuild,\n  UnexpectedAddressError,\n  VerificationOptions,\n  ITransactionRecipient,\n  Triple,\n} from '@bitgo/sdk-core';\n\nimport { AbstractUtxoCoin, Output, isWalletOutput } from '../../abstractUtxoCoin';\n\nconst debug = debugLib('bitgo:v2:parseoutput');\n\ninterface HandleVerifyAddressErrorResponse {\n  external: boolean;\n  needsCustomChangeKeySignatureVerification?: boolean;\n}\n\n/**\n * Check an address which failed initial validation to see if it's the base address of a migrated v1 bch wallet.\n *\n * The wallet in question could be a migrated SafeHD BCH wallet, and the transaction we\n * are currently parsing is trying to spend change back to the v1 wallet base address.\n *\n * It does this since we don't allow new address creation for these wallets,\n * and instead return the base address from the v1 wallet when a new address is requested.\n * If this new address is requested for the purposes of spending change back to the wallet,\n * the change will go to the v1 wallet base address. This address *is* on the wallet,\n * but it will still cause an error to be thrown by verifyAddress, since the derivation path\n * used for this address is non-standard. (I have seen these addresses derived using paths m/0/0 and m/101,\n * whereas the v2 addresses are derived using path  m/0/0/${chain}/${index}).\n *\n * This means we need to check for this case explicitly in this catch block, and classify\n * these types of outputs as internal instead of external. Failing to do so would cause the\n * transaction's implicit external outputs (ie, outputs which go to addresses not specified in\n * the recipients array) to add up to more than the 150 basis point limit which we enforce on\n * pay-as-you-go outputs (which should be the only implicit external outputs on our transactions).\n *\n * The 150 basis point limit for implicit external sends is enforced in verifyTransaction,\n * which calls this function to get information on the total external/internal spend amounts\n * for a transaction. The idea here is to protect from the transaction being maliciously modified\n * to add more implicit external spends (eg, to an attacker-controlled wallet).\n *\n * See verifyTransaction for more information on how transaction prebuilds are verified before signing.\n *\n * @param wallet {Wallet} wallet which is making the transaction\n * @param currentAddress {string} address to check for externality relative to v1 wallet base address\n */\nfunction isMigratedAddress(wallet: IWallet, currentAddress: string): boolean {\n  if (_.isString(wallet.migratedFrom()) && wallet.migratedFrom() === currentAddress) {\n    debug('found address %s which was migrated from v1 wallet, address is not external', currentAddress);\n    return true;\n  }\n\n  return false;\n}\n\ninterface VerifyCustomChangeAddressOptions {\n  coin: AbstractUtxoCoin;\n  customChangeKeys: HandleVerifyAddressErrorOptions['customChangeKeys'];\n  addressType: HandleVerifyAddressErrorOptions['addressType'];\n  addressDetails: HandleVerifyAddressErrorOptions['addressDetails'];\n  currentAddress: HandleVerifyAddressErrorOptions['currentAddress'];\n}\n\n/**\n * Check to see if an address is derived from the given custom change keys\n * @param {VerifyCustomChangeAddressOptions} params\n * @return {boolean}\n */\nasync function verifyCustomChangeAddress(params: VerifyCustomChangeAddressOptions): Promise<boolean> {\n  const { coin, customChangeKeys, addressType, addressDetails, currentAddress } = params;\n  try {\n    return await coin.verifyAddress(\n      _.extend({ addressType }, addressDetails, {\n        keychains: customChangeKeys,\n        address: currentAddress,\n      })\n    );\n  } catch (e) {\n    debug('failed to verify custom change address %s', currentAddress);\n    return false;\n  }\n}\n\ninterface HandleVerifyAddressErrorOptions {\n  e: Error;\n  currentAddress: string;\n  wallet: IWallet;\n  txParams: {\n    changeAddress?: string;\n  };\n  customChangeKeys?: CustomChangeOptions['keys'];\n  coin: AbstractUtxoCoin;\n  addressDetails?: any;\n  addressType?: string;\n  considerMigratedFromAddressInternal?: boolean;\n}\n\nasync function handleVerifyAddressError({\n  e,\n  currentAddress,\n  wallet,\n  txParams,\n  customChangeKeys,\n  coin,\n  addressDetails,\n  addressType,\n  considerMigratedFromAddressInternal,\n}: HandleVerifyAddressErrorOptions): Promise<HandleVerifyAddressErrorResponse> {\n  // Todo: name server-side errors to avoid message-based checking [BG-5124]\n  const walletAddressNotFound = e.message.includes('wallet address not found');\n  const unexpectedAddress = e instanceof UnexpectedAddressError;\n  if (walletAddressNotFound || unexpectedAddress) {\n    if (unexpectedAddress && !walletAddressNotFound) {\n      // check to see if this is a migrated v1 bch address - it could be internal\n      const isMigrated = isMigratedAddress(wallet, currentAddress);\n      if (isMigrated) {\n        return { external: considerMigratedFromAddressInternal === false };\n      }\n\n      debug('Address %s was found on wallet but could not be reconstructed', currentAddress);\n\n      // attempt to verify address using custom change address keys if the wallet has that feature enabled\n      if (\n        customChangeKeys &&\n        (await verifyCustomChangeAddress({ coin, addressDetails, addressType, currentAddress, customChangeKeys }))\n      ) {\n        // address is valid against the custom change keys. Mark address as not external\n        // and request signature verification for the custom change keys\n        debug('Address %s verified as derived from the custom change keys', currentAddress);\n        return { external: false, needsCustomChangeKeySignatureVerification: true };\n      }\n    }\n\n    // the address was found, but not on the wallet, which simply means it's external\n    debug('Address %s presumed external', currentAddress);\n    return { external: true };\n  } else if (e instanceof InvalidAddressDerivationPropertyError && currentAddress === txParams.changeAddress) {\n    // expect to see this error when passing in a custom changeAddress with no chain or index\n    return { external: false };\n  }\n\n  console.error('Address classification failed for address', currentAddress);\n  console.trace(e);\n  /**\n   * It might be a completely invalid address or a bad validation attempt or something else completely, in\n   * which case we do not proceed and rather rethrow the error, which is safer than assuming that the address\n   * validation failed simply because it's external to the wallet.\n   */\n  throw e;\n}\n\ninterface FetchAddressDetailsOptions {\n  reqId?: IRequestTracer;\n  disableNetworking: boolean;\n  addressDetailsPrebuild: any;\n  addressDetailsVerification: any;\n  currentAddress: string;\n  wallet: IWallet;\n}\n\nasync function fetchAddressDetails({\n  reqId,\n  disableNetworking,\n  addressDetailsPrebuild,\n  addressDetailsVerification,\n  currentAddress,\n  wallet,\n}: FetchAddressDetailsOptions) {\n  let addressDetails = _.extend({}, addressDetailsPrebuild, addressDetailsVerification);\n  debug('Locally available address %s details: %O', currentAddress, addressDetails);\n  if (_.isEmpty(addressDetails) && !disableNetworking) {\n    addressDetails = await wallet.getAddress({ address: currentAddress, reqId });\n    debug('Downloaded address %s details: %O', currentAddress, addressDetails);\n  }\n  return addressDetails;\n}\n\nexport interface CustomChangeOptions {\n  keys: Triple<{ pub: string }>;\n  signatures: Triple<string>;\n}\n\nexport interface ParseOutputOptions {\n  currentOutput: Output;\n  coin: AbstractUtxoCoin;\n  txPrebuild: TransactionPrebuild;\n  verification: VerificationOptions;\n  keychainArray: Triple<{ pub: string }>;\n  wallet: IWallet;\n  txParams: {\n    recipients: ITransactionRecipient[];\n    changeAddress?: string;\n  };\n  customChange?: CustomChangeOptions;\n  reqId?: IRequestTracer;\n}\n\nexport async function parseOutput({\n  currentOutput,\n  coin,\n  txPrebuild,\n  verification,\n  keychainArray,\n  wallet,\n  txParams,\n  customChange,\n  reqId,\n}: ParseOutputOptions): Promise<Output> {\n  const disableNetworking = !!verification.disableNetworking;\n  const currentAddress = currentOutput.address;\n\n  if (currentAddress === undefined) {\n    // In the case that the address is undefined, it means that the output has a non-encodeable scriptPubkey\n    // If this is the case, then we need to check that the amount is 0 and we can skip the rest.\n    if (currentOutput.amount.toString() !== '0') {\n      throw new Error('output with undefined address must have amount of 0');\n    }\n    return currentOutput;\n  }\n\n  // attempt to grab the address details from either the prebuilt tx, or the verification params.\n  // If both of these are empty, then we will try to get the address details from bitgo instead\n  const addressDetailsPrebuild = _.get(txPrebuild, `txInfo.walletAddressDetails.${currentAddress}`, {});\n  const addressDetailsVerification: AddressVerificationData = verification?.addresses?.[currentAddress] ?? {};\n  debug('Parsing address details for %s', currentAddress);\n  let currentAddressDetails = undefined;\n  let currentAddressType: string | undefined = undefined;\n  const RECIPIENT_THRESHOLD = 1000;\n  try {\n    // In the case of PSBTs, we can already determine the internal/external status of the output addresses\n    // based on the derivation information being included in the PSBT. We can short circuit GET v2.wallet.address\n    // and save on network requests. Since we have the derivation information already, we can still verify the address\n    if (currentOutput.external !== undefined) {\n      // In the case that we have a custom change wallet, we need to verify the address against the custom change keys\n      // and not the wallet keys. This check is done in the handleVerifyAddressError function if this error is thrown.\n      if (customChange !== undefined) {\n        throw new UnexpectedAddressError('`address validation failure');\n      }\n      // If it is an internal address, we can skip the network request and just verify the address locally with the\n      // derivation information we have. Otherwise, if the address is external, which is the only remaining case, we\n      // can just return the current output as is without contacting the server.\n      if (isWalletOutput(currentOutput)) {\n        const res = await coin.isWalletAddress({\n          addressType: AbstractUtxoCoin.inferAddressType({ chain: currentOutput.chain }) || undefined,\n          keychains: keychainArray,\n          address: currentAddress,\n          chain: currentOutput.chain,\n          index: currentOutput.index,\n        });\n        if (!res) {\n          throw new UnexpectedAddressError();\n        }\n      }\n      return currentOutput;\n    }\n    /**\n     * The only way to determine whether an address is known on the wallet is to initiate a network request and\n     * fetch it. Should the request fail and return a 404, it will throw and therefore has to be caught. For that\n     * reason, address wallet ownership detection is wrapped in a try/catch. Additionally, once the address\n     * details are fetched on the wallet, a local address validation is run, whose errors however are generated\n     * client-side and can therefore be analyzed with more granularity and type checking.\n     */\n\n    /**\n     * In order to minimize API requests, we assume that explicit recipients are always external when the\n     * recipient list is > 1000 This is not always a valid assumption and could lead greater apparent spend (but never lower)\n     */\n    if (txParams.recipients !== undefined && txParams.recipients.length > RECIPIENT_THRESHOLD) {\n      const isCurrentAddressInRecipients = txParams.recipients.some((recipient) =>\n        recipient.address.includes(currentAddress)\n      );\n\n      if (isCurrentAddressInRecipients) {\n        return { ...currentOutput };\n      }\n    }\n\n    const addressDetails = await fetchAddressDetails({\n      reqId,\n      addressDetailsVerification,\n      addressDetailsPrebuild,\n      currentAddress,\n      disableNetworking,\n      wallet,\n    });\n    // verify that the address is on the wallet. verifyAddress throws if\n    // it fails to correctly rederive the address, meaning it's external\n    currentAddressType = AbstractUtxoCoin.inferAddressType(addressDetails) || undefined;\n    currentAddressDetails = addressDetails;\n    await coin.verifyAddress(\n      _.extend({ addressType: currentAddressType }, addressDetails, {\n        keychains: keychainArray,\n        address: currentAddress,\n      })\n    );\n    debug('Address %s verification passed', currentAddress);\n\n    // verify address succeeded without throwing, so the address was\n    // correctly rederived from the wallet keychains, making it not external\n    return _.extend({}, currentOutput, addressDetails, { external: false });\n  } catch (e) {\n    debug('Address %s verification threw an error:', currentAddress, e);\n    return _.extend(\n      {},\n      currentOutput,\n      await handleVerifyAddressError({\n        e,\n        coin,\n        currentAddress,\n        wallet,\n        txParams,\n        customChangeKeys: customChange && customChange.keys,\n        addressDetails: currentAddressDetails,\n        addressType: currentAddressType,\n        considerMigratedFromAddressInternal: verification.considerMigratedFromAddressInternal,\n      })\n    );\n  }\n}\n"]}

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


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