PHP WebShell

Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/sdk-api/dist/src/v1

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

"use strict";
/**
 * @hidden
 */
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    var desc = Object.getOwnPropertyDescriptor(m, k);
    if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
      desc = { enumerable: true, get: function() { return m[k]; } };
    }
    Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
    Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
    o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
    var ownKeys = function(o) {
        ownKeys = Object.getOwnPropertyNames || function (o) {
            var ar = [];
            for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
            return ar;
        };
        return ownKeys(o);
    };
    return function (mod) {
        if (mod && mod.__esModule) return mod;
        var result = {};
        if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
        __setModuleDefault(result, mod);
        return result;
    };
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
/**
 */
//
// TravelRule Object
// BitGo accessor for a specific enterprise
//
// Copyright 2014, BitGo, Inc.  All Rights Reserved.
//
const sdk_core_1 = require("@bitgo/sdk-core");
const utxo_lib_1 = require("@bitgo/utxo-lib");
const utxolib = __importStar(require("@bitgo/utxo-lib"));
const lodash_1 = __importDefault(require("lodash"));
//
// Constructor
//
const TravelRule = function (bitgo) {
    // @ts-expect-error - no implicit this
    this.bitgo = bitgo;
};
TravelRule.prototype.url = function (extra) {
    extra = extra || '';
    return this.bitgo.url('/travel/' + extra);
};
/**
 * Get available travel-rule info recipients for a transaction
 * @param params
 *  txid: transaction id
 * @param callback
 * @returns {*}
 */
TravelRule.prototype.getRecipients = function (params, callback) {
    params = params || {};
    params.txid = params.txid || params.hash;
    sdk_core_1.common.validateParams(params, ['txid'], [], callback);
    const url = this.url(params.txid + '/recipients');
    return Promise.resolve(this.bitgo.get(url).result('recipients')).then(callback).catch(callback);
};
TravelRule.prototype.validateTravelInfo = function (info) {
    const fields = {
        amount: { type: 'number' },
        toAddress: { type: 'string' },
        toEnterprise: { type: 'string' },
        fromUserName: { type: 'string' },
        fromUserAccount: { type: 'string' },
        fromUserAddress: { type: 'string' },
        toUserName: { type: 'string' },
        toUserAccount: { type: 'string' },
        toUserAddress: { type: 'string' },
        extra: { type: 'object' },
    };
    lodash_1.default.forEach(fields, function (field, fieldName) {
        // No required fields yet -- should there be?
        if (field.required) {
            if (info[fieldName] === undefined) {
                throw new Error('missing required field ' + fieldName + ' in travel info');
            }
        }
        if (info[fieldName] && typeof info[fieldName] !== field.type) {
            throw new Error('incorrect type for field ' + fieldName + ' in travel info, expected ' + field.type);
        }
    });
    // Strip out any other fields we don't know about
    const result = lodash_1.default.pick(info, lodash_1.default.keys(fields));
    if (lodash_1.default.isEmpty(result)) {
        throw new Error('empty travel data');
    }
    return result;
};
/**
 * Takes a transaction object as returned by getTransaction or listTransactions, along
 * with a keychain (or hdnode object), and attempts to decrypt any encrypted travel
 * info included in the transaction's receivedTravelInfo field.
 * Parameters:
 *   tx: a transaction object
 *   keychain: keychain object (with xprv)
 * Returns:
 *   the tx object, augmented with decrypted travelInfo fields
 */
TravelRule.prototype.decryptReceivedTravelInfo = function (params = {}) {
    const tx = params.tx;
    if (!lodash_1.default.isObject(tx)) {
        throw new Error('expecting tx param to be object');
    }
    if (!tx.receivedTravelInfo || !tx.receivedTravelInfo.length) {
        return tx;
    }
    const keychain = params.keychain;
    if (!lodash_1.default.isObject(keychain) || !lodash_1.default.isString(keychain.xprv)) {
        throw new Error('expecting keychain param with xprv');
    }
    const hdNode = utxo_lib_1.bip32.fromBase58(keychain.xprv);
    tx.receivedTravelInfo.forEach((info) => {
        const key = hdNode.derivePath((0, sdk_core_1.sanitizeLegacyPath)(info.toPubKeyPath));
        const secret = (0, sdk_core_1.getSharedSecret)(key, Buffer.from(info.fromPubKey, 'hex')).toString('hex');
        try {
            const decrypted = this.bitgo.decrypt({
                input: info.encryptedTravelInfo,
                password: secret,
            });
            info.travelInfo = JSON.parse(decrypted);
        }
        catch (err) {
            console.error('failed to decrypt or parse travel info for ', info.transactionId + ':' + info.outputIndex);
        }
    });
    return tx;
};
TravelRule.prototype.prepareParams = function (params) {
    params = params || {};
    params.txid = params.txid || params.hash;
    sdk_core_1.common.validateParams(params, ['txid'], ['fromPrivateInfo']);
    const txid = params.txid;
    const recipient = params.recipient;
    let travelInfo = params.travelInfo;
    if (!recipient || !lodash_1.default.isObject(recipient)) {
        throw new Error('invalid or missing recipient');
    }
    if (!travelInfo || !lodash_1.default.isObject(travelInfo)) {
        throw new Error('invalid or missing travelInfo');
    }
    if (!params.noValidate) {
        travelInfo = this.validateTravelInfo(travelInfo);
    }
    // Fill in toEnterprise if not already filled
    if (!travelInfo.toEnterprise && recipient.enterprise) {
        travelInfo.toEnterprise = recipient.enterprise;
    }
    // If a key was not provided, create a new random key
    let fromKey = params.fromKey && utxolib.ECPair.fromWIF(params.fromKey, (0, sdk_core_1.getNetwork)());
    if (!fromKey) {
        fromKey = (0, sdk_core_1.makeRandomKey)();
    }
    // Compute the shared key for encryption
    const sharedSecret = (0, sdk_core_1.getSharedSecret)(fromKey, Buffer.from(recipient.pubKey, 'hex')).toString('hex');
    // JSON-ify and encrypt the payload
    const travelInfoJSON = JSON.stringify(travelInfo);
    const encryptedTravelInfo = this.bitgo.encrypt({
        input: travelInfoJSON,
        password: sharedSecret,
    });
    const result = {
        txid: txid,
        outputIndex: recipient.outputIndex,
        toPubKey: recipient.pubKey,
        fromPubKey: fromKey.publicKey.toString('hex'),
        encryptedTravelInfo: encryptedTravelInfo,
        fromPrivateInfo: undefined,
    };
    if (params.fromPrivateInfo) {
        result.fromPrivateInfo = params.fromPrivateInfo;
    }
    return result;
};
/**
 * Send travel data to the server for a transaction
 */
TravelRule.prototype.send = function (params, callback) {
    params = params || {};
    params.txid = params.txid || params.hash;
    sdk_core_1.common.validateParams(params, ['txid', 'toPubKey', 'encryptedTravelInfo'], ['fromPubKey', 'fromPrivateInfo'], callback);
    if (!lodash_1.default.isNumber(params.outputIndex)) {
        throw new Error('invalid outputIndex');
    }
    return Promise.resolve(this.bitgo
        .post(this.url(params.txid + '/' + params.outputIndex))
        .send(params)
        .result())
        .then(callback)
        .catch(callback);
};
/**
 * Send multiple travel rule infos for the outputs of a single transaction.
 * Parameters:
 *   - txid (or hash): txid of the transaction (must be a sender of the tx)
 *   - travelInfos: array of travelInfo objects which look like the following:
 *     {
 *       outputIndex: number,     // tx output index
 *       fromUserName: string,    // name of the sending user
 *       fromUserAccount: string, // account id of the sending user
 *       fromUserAddress: string, // mailing address of the sending user
 *       toUserName: string,      // name of the receiving user
 *       toUserAccount: string,   // account id of the receiving user
 *       toUserAddress: string    // mailing address of the receiving user
 *     }
 *     All fields aside from outputIndex are optional, but at least one must
 *     be defined.
 *
 *  It is not necessary to provide travelInfo for all output indices.
 *  End-to-end encryption of the travel info is handled automatically by this method.
 *
 */
TravelRule.prototype.sendMany = function (params, callback) {
    params = params || {};
    params.txid = params.txid || params.hash;
    sdk_core_1.common.validateParams(params, ['txid'], callback);
    const travelInfos = params.travelInfos;
    if (!lodash_1.default.isArray(travelInfos)) {
        throw new Error('expected parameter travelInfos to be array');
    }
    const self = this;
    const travelInfoMap = (0, lodash_1.default)(travelInfos)
        .keyBy('outputIndex')
        .mapValues(function (travelInfo) {
        return self.validateTravelInfo(travelInfo);
    })
        .value();
    return self.getRecipients({ txid: params.txid }).then(function (recipients) {
        // Build up data to post
        const sendParamsList = [];
        // don't regenerate a new random key for each recipient
        const fromKey = params.fromKey || (0, sdk_core_1.makeRandomKey)().toWIF();
        recipients.forEach(function (recipient) {
            const outputIndex = recipient.outputIndex;
            const info = travelInfoMap[outputIndex];
            if (info) {
                if (info.amount && info.amount !== recipient.amount) {
                    throw new Error('amount did not match for output index ' + outputIndex);
                }
                const sendParams = self.prepareParams({
                    txid: params.txid,
                    recipient: recipient,
                    travelInfo: info,
                    fromKey: fromKey,
                    noValidate: true, // don't re-validate
                });
                sendParamsList.push(sendParams);
            }
        });
        const result = {
            matched: sendParamsList.length,
            results: [],
        };
        const sendSerial = function () {
            const sendParams = sendParamsList.shift();
            if (!sendParams) {
                return result;
            }
            return self
                .send(sendParams)
                .then(function (res) {
                result.results.push({ result: res });
                return sendSerial();
            })
                .catch(function (err) {
                result.results.push({ error: err.toString() });
                return sendSerial();
            });
        };
        return sendSerial();
    });
};
module.exports = TravelRule;
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhdmVsUnVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy92MS90cmF2ZWxSdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7R0FFRzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSDtHQUNHO0FBQ0gsRUFBRTtBQUNGLG9CQUFvQjtBQUNwQiwyQ0FBMkM7QUFDM0MsRUFBRTtBQUNGLG9EQUFvRDtBQUNwRCxFQUFFO0FBQ0YsOENBQXlHO0FBQ3pHLDhDQUF3RDtBQUN4RCx5REFBMkM7QUFDM0Msb0RBQXVCO0FBeUJ2QixFQUFFO0FBQ0YsY0FBYztBQUNkLEVBQUU7QUFDRixNQUFNLFVBQVUsR0FBRyxVQUFVLEtBQUs7SUFDaEMsc0NBQXNDO0lBQ3RDLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDO0FBQ3JCLENBQUMsQ0FBQztBQUVGLFVBQVUsQ0FBQyxTQUFTLENBQUMsR0FBRyxHQUFHLFVBQVUsS0FBSztJQUN4QyxLQUFLLEdBQUcsS0FBSyxJQUFJLEVBQUUsQ0FBQztJQUNwQixPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLFVBQVUsR0FBRyxLQUFLLENBQUMsQ0FBQztBQUM1QyxDQUFDLENBQUM7QUFFRjs7Ozs7O0dBTUc7QUFDSCxVQUFVLENBQUMsU0FBUyxDQUFDLGFBQWEsR0FBRyxVQUFVLE1BQU0sRUFBRSxRQUFRO0lBQzdELE1BQU0sR0FBRyxNQUFNLElBQUksRUFBRSxDQUFDO0lBQ3RCLE1BQU0sQ0FBQyxJQUFJLEdBQUcsTUFBTSxDQUFDLElBQUksSUFBSSxNQUFNLENBQUMsSUFBSSxDQUFDO0lBQ3pDLGlCQUFNLENBQUMsY0FBYyxDQUFDLE1BQU0sRUFBRSxDQUFDLE1BQU0sQ0FBQyxFQUFFLEVBQUUsRUFBRSxRQUFRLENBQUMsQ0FBQztJQUV0RCxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxJQUFJLEdB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TyxFQUFFLE9BQU87b0JBQ2hCLFVBQVUsRUFBRSxJQUFJLEVBQUUsb0JBQW9CO2lCQUN2QyxDQUFDLENBQUM7Z0JBQ0gsY0FBYyxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQztZQUNsQyxDQUFDO1FBQ0gsQ0FBQyxDQUFDLENBQUM7UUFFSCxNQUFNLE1BQU0sR0FNUjtZQUNGLE9BQU8sRUFBRSxjQUFjLENBQUMsTUFBTTtZQUM5QixPQUFPLEVBQUUsRUFBRTtTQUNaLENBQUM7UUFFRixNQUFNLFVBQVUsR0FBRztZQUNqQixNQUFNLFVBQVUsR0FBRyxjQUFjLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDMUMsSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFDO2dCQUNoQixPQUFPLE1BQU0sQ0FBQztZQUNoQixDQUFDO1lBQ0QsT0FBTyxJQUFJO2lCQUNSLElBQUksQ0FBQyxVQUFVLENBQUM7aUJBQ2hCLElBQUksQ0FBQyxVQUFVLEdBQUc7Z0JBQ2pCLE1BQU0sQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEVBQUUsTUFBTSxFQUFFLEdBQUcsRUFBRSxDQUFDLENBQUM7Z0JBQ3JDLE9BQU8sVUFBVSxFQUFFLENBQUM7WUFDdEIsQ0FBQyxDQUFDO2lCQUNELEtBQUssQ0FBQyxVQUFVLEdBQUc7Z0JBQ2xCLE1BQU0sQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEVBQUUsS0FBSyxFQUFFLEdBQUcsQ0FBQyxRQUFRLEVBQUUsRUFBRSxDQUFDLENBQUM7Z0JBQy9DLE9BQU8sVUFBVSxFQUFFLENBQUM7WUFDdEIsQ0FBQyxDQUFDLENBQUM7UUFDUCxDQUFDLENBQUM7UUFFRixPQUFPLFVBQVUsRUFBRSxDQUFDO0lBQ3RCLENBQUMsQ0FBQyxDQUFDO0FBQ0wsQ0FBQyxDQUFDO0FBRUYsTUFBTSxDQUFDLE9BQU8sR0FBRyxVQUFVLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBoaWRkZW5cbiAqL1xuXG4vKipcbiAqL1xuLy9cbi8vIFRyYXZlbFJ1bGUgT2JqZWN0XG4vLyBCaXRHbyBhY2Nlc3NvciBmb3IgYSBzcGVjaWZpYyBlbnRlcnByaXNlXG4vL1xuLy8gQ29weXJpZ2h0IDIwMTQsIEJpdEdvLCBJbmMuICBBbGwgUmlnaHRzIFJlc2VydmVkLlxuLy9cbmltcG9ydCB7IGNvbW1vbiwgZ2V0TmV0d29yaywgZ2V0U2hhcmVkU2VjcmV0LCBtYWtlUmFuZG9tS2V5LCBzYW5pdGl6ZUxlZ2FjeVBhdGggfSBmcm9tICdAYml0Z28vc2RrLWNvcmUnO1xuaW1wb3J0IHsgYmlwMzIsIEJJUDMySW50ZXJmYWNlIH0gZnJvbSAnQGJpdGdvL3V0eG8tbGliJztcbmltcG9ydCAqIGFzIHV0eG9saWIgZnJvbSAnQGJpdGdvL3V0eG8tbGliJztcbmltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5cbmludGVyZmFjZSBEZWNyeXB0UmVjZWl2ZWRUcmF2ZWxSdWxlT3B0aW9ucyB7XG4gIHR4Pzoge1xuICAgIHJlY2VpdmVkVHJhdmVsSW5mbz86IHtcbiAgICAgIHRvUHViS2V5UGF0aDogc3RyaW5nO1xuICAgICAgZnJvbVB1YktleTogc3RyaW5nO1xuICAgICAgZW5jcnlwdGVkVHJhdmVsSW5mbzogc3RyaW5nO1xuICAgICAgdHJhdmVsSW5mbzogc3RyaW5nO1xuICAgICAgdHJhbnNhY3Rpb25JZDogc3RyaW5nO1xuICAgICAgb3V0cHV0SW5kZXg6IG51bWJlcjtcbiAgICB9W107XG4gIH07XG4gIGtleWNoYWluPzoge1xuICAgIHhwcnY/OiBzdHJpbmc7XG4gIH07XG4gIGhkbm9kZT86IEJJUDMySW50ZXJmYWNlO1xufVxuXG5pbnRlcmZhY2UgUmVjaXBpZW50IHtcbiAgZW50ZXJwcmlzZTogc3RyaW5nO1xuICBwdWJLZXk6IHN0cmluZztcbiAgb3V0cHV0SW5kZXg6IHN0cmluZztcbn1cblxuLy9cbi8vIENvbnN0cnVjdG9yXG4vL1xuY29uc3QgVHJhdmVsUnVsZSA9IGZ1bmN0aW9uIChiaXRnbykge1xuICAvLyBAdHMtZXhwZWN0LWVycm9yIC0gbm8gaW1wbGljaXQgdGhpc1xuICB0aGlzLmJpdGdvID0gYml0Z287XG59O1xuXG5UcmF2ZWxSdWxlLnByb3RvdHlwZS51cmwgPSBmdW5jdGlvbiAoZXh0cmEpIHtcbiAgZXh0cmEgPSBleHRyYSB8fCAnJztcbiAgcmV0dXJuIHRoaXMuYml0Z28udXJsKCcvdHJhdmVsLycgKyBleHRyYSk7XG59O1xuXG4vKipcbiAqIEdldCBhdmFpbGFibGUgdHJhdmVsLXJ1bGUgaW5mbyByZWNpcGllbnRzIGZvciBhIHRyYW5zYWN0aW9uXG4gKiBAcGFyYW0gcGFyYW1zXG4gKiAgdHhpZDogdHJhbnNhY3Rpb24gaWRcbiAqIEBwYXJhbSBjYWxsYmFja1xuICogQHJldHVybnMgeyp9XG4gKi9cblRyYXZlbFJ1bGUucHJvdG90eXBlLmdldFJlY2lwaWVudHMgPSBmdW5jdGlvbiAocGFyYW1zLCBjYWxsYmFjaykge1xuICBwYXJhbXMgPSBwYXJhbXMgfHwge307XG4gIHBhcmFtcy50eGlkID0gcGFyYW1zLnR4aWQgfHwgcGFyYW1zLmhhc2g7XG4gIGNvbW1vbi52YWxpZGF0ZVBhcmFtcyhwYXJhbXMsIFsndHhpZCddLCBbXSwgY2FsbGJhY2spO1xuXG4gIGNvbnN0IHVybCA9IHRoaXMudXJsKHBhcmFtcy50eGlkICsgJy9yZWNpcGllbnRzJyk7XG4gIHJldHVybiBQcm9taXNlLnJlc29sdmUodGhpcy5iaXRnby5nZXQodXJsKS5yZXN1bHQoJ3JlY2lwaWVudHMnKSkudGhlbihjYWxsYmFjaykuY2F0Y2goY2FsbGJhY2spO1xufTtcblxuVHJhdmVsUnVsZS5wcm90b3R5cGUudmFsaWRhdGVUcmF2ZWxJbmZvID0gZnVuY3Rpb24gKGluZm8pIHtcbiAgY29uc3QgZmllbGRzID0ge1xuICAgIGFtb3VudDogeyB0eXBlOiAnbnVtYmVyJyB9LFxuICAgIHRvQWRkcmVzczogeyB0eXBlOiAnc3RyaW5nJyB9LFxuICAgIHRvRW50ZXJwcmlzZTogeyB0eXBlOiAnc3RyaW5nJyB9LFxuICAgIGZyb21Vc2VyTmFtZTogeyB0eXBlOiAnc3RyaW5nJyB9LFxuICAgIGZyb21Vc2VyQWNjb3VudDogeyB0eXBlOiAnc3RyaW5nJyB9LFxuICAgIGZyb21Vc2VyQWRkcmVzczogeyB0eXBlOiAnc3RyaW5nJyB9LFxuICAgIHRvVXNlck5hbWU6IHsgdHlwZTogJ3N0cmluZycgfSxcbiAgICB0b1VzZXJBY2NvdW50OiB7IHR5cGU6ICdzdHJpbmcnIH0sXG4gICAgdG9Vc2VyQWRkcmVzczogeyB0eXBlOiAnc3RyaW5nJyB9LFxuICAgIGV4dHJhOiB7IHR5cGU6ICdvYmplY3QnIH0sXG4gIH07XG5cbiAgXy5mb3JFYWNoKGZpZWxkcywgZnVuY3Rpb24gKGZpZWxkOiBhbnksIGZpZWxkTmFtZSkge1xuICAgIC8vIE5vIHJlcXVpcmVkIGZpZWxkcyB5ZXQgLS0gc2hvdWxkIHRoZXJlIGJlP1xuICAgIGlmIChmaWVsZC5yZXF1aXJlZCkge1xuICAgICAgaWYgKGluZm9bZmllbGROYW1lXSA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcignbWlzc2luZyByZXF1aXJlZCBmaWVsZCAnICsgZmllbGROYW1lICsgJyBpbiB0cmF2ZWwgaW5mbycpO1xuICAgICAgfVxuICAgIH1cbiAgICBpZiAoaW5mb1tmaWVsZE5hbWVdICYmIHR5cGVvZiBpbmZvW2ZpZWxkTmFtZV0gIT09IGZpZWxkLnR5cGUpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignaW5jb3JyZWN0IHR5cGUgZm9yIGZpZWxkICcgKyBmaWVsZE5hbWUgKyAnIGluIHRyYXZlbCBpbmZvLCBleHBlY3RlZCAnICsgZmllbGQudHlwZSk7XG4gICAgfVxuICB9KTtcblxuICAvLyBTdHJpcCBvdXQgYW55IG90aGVyIGZpZWxkcyB3ZSBkb24ndCBrbm93IGFib3V0XG4gIGNvbnN0IHJlc3VsdCA9IF8ucGljayhpbmZvLCBfLmtleXMoZmllbGRzKSk7XG4gIGlmIChfLmlzRW1wdHkocmVzdWx0KSkge1xuICAgIHRocm93IG5ldyBFcnJvcignZW1wdHkgdHJhdmVsIGRhdGEnKTtcbiAgfVxuICByZXR1cm4gcmVzdWx0O1xufTtcblxuLyoqXG4gKiBUYWtlcyBhIHRyYW5zYWN0aW9uIG9iamVjdCBhcyByZXR1cm5lZCBieSBnZXRUcmFuc2FjdGlvbiBvciBsaXN0VHJhbnNhY3Rpb25zLCBhbG9uZ1xuICogd2l0aCBhIGtleWNoYWluIChvciBoZG5vZGUgb2JqZWN0KSwgYW5kIGF0dGVtcHRzIHRvIGRlY3J5cHQgYW55IGVuY3J5cHRlZCB0cmF2ZWxcbiAqIGluZm8gaW5jbHVkZWQgaW4gdGhlIHRyYW5zYWN0aW9uJ3MgcmVjZWl2ZWRUcmF2ZWxJbmZvIGZpZWxkLlxuICogUGFyYW1ldGVyczpcbiAqICAgdHg6IGEgdHJhbnNhY3Rpb24gb2JqZWN0XG4gKiAgIGtleWNoYWluOiBrZXljaGFpbiBvYmplY3QgKHdpdGggeHBydilcbiAqIFJldHVybnM6XG4gKiAgIHRoZSB0eCBvYmplY3QsIGF1Z21lbnRlZCB3aXRoIGRlY3J5cHRlZCB0cmF2ZWxJbmZvIGZpZWxkc1xuICovXG5UcmF2ZWxSdWxlLnByb3RvdHlwZS5kZWNyeXB0UmVjZWl2ZWRUcmF2ZWxJbmZvID0gZnVuY3Rpb24gKHBhcmFtczogRGVjcnlwdFJlY2VpdmVkVHJhdmVsUnVsZU9wdGlvbnMgPSB7fSkge1xuICBjb25zdCB0eCA9IHBhcmFtcy50eDtcbiAgaWYgKCFfLmlzT2JqZWN0KHR4KSkge1xuICAgIHRocm93IG5ldyBFcnJvcignZXhwZWN0aW5nIHR4IHBhcmFtIHRvIGJlIG9iamVjdCcpO1xuICB9XG5cbiAgaWYgKCF0eC5yZWNlaXZlZFRyYXZlbEluZm8gfHwgIXR4LnJlY2VpdmVkVHJhdmVsSW5mby5sZW5ndGgpIHtcbiAgICByZXR1cm4gdHg7XG4gIH1cblxuICBjb25zdCBrZXljaGFpbiA9IHBhcmFtcy5rZXljaGFpbjtcbiAgaWYgKCFfLmlzT2JqZWN0KGtleWNoYWluKSB8fCAhXy5pc1N0cmluZyhrZXljaGFpbi54cHJ2KSkge1xuICAgIHRocm93IG5ldyBFcnJvcignZXhwZWN0aW5nIGtleWNoYWluIHBhcmFtIHdpdGggeHBydicpO1xuICB9XG4gIGNvbnN0IGhkTm9kZSA9IGJpcDMyLmZyb21CYXNlNTgoa2V5Y2hhaW4ueHBydik7XG5cbiAgdHgucmVjZWl2ZWRUcmF2ZWxJbmZvLmZvckVhY2goKGluZm8pID0+IHtcbiAgICBjb25zdCBrZXkgPSBoZE5vZGUuZGVyaXZlUGF0aChzYW5pdGl6ZUxlZ2FjeVBhdGgoaW5mby50b1B1YktleVBhdGgpKTtcbiAgICBjb25zdCBzZWNyZXQgPSBnZXRTaGFyZWRTZWNyZXQoa2V5LCBCdWZmZXIuZnJvbShpbmZvLmZyb21QdWJLZXksICdoZXgnKSkudG9TdHJpbmcoJ2hleCcpO1xuICAgIHRyeSB7XG4gICAgICBjb25zdCBkZWNyeXB0ZWQgPSB0aGlzLmJpdGdvLmRlY3J5cHQoe1xuICAgICAgICBpbnB1dDogaW5mby5lbmNyeXB0ZWRUcmF2ZWxJbmZvLFxuICAgICAgICBwYXNzd29yZDogc2VjcmV0LFxuICAgICAgfSk7XG4gICAgICBpbmZvLnRyYXZlbEluZm8gPSBKU09OLnBhcnNlKGRlY3J5cHRlZCk7XG4gICAgfSBjYXRjaCAoZXJyKSB7XG4gICAgICBjb25zb2xlLmVycm9yKCdmYWlsZWQgdG8gZGVjcnlwdCBvciBwYXJzZSB0cmF2ZWwgaW5mbyBmb3IgJywgaW5mby50cmFuc2FjdGlvbklkICsgJzonICsgaW5mby5vdXRwdXRJbmRleCk7XG4gICAgfVxuICB9KTtcblxuICByZXR1cm4gdHg7XG59O1xuXG5UcmF2ZWxSdWxlLnByb3RvdHlwZS5wcmVwYXJlUGFyYW1zID0gZnVuY3Rpb24gKHBhcmFtcykge1xuICBwYXJhbXMgPSBwYXJhbXMgfHwge307XG4gIHBhcmFtcy50eGlkID0gcGFyYW1zLnR4aWQgfHwgcGFyYW1zLmhhc2g7XG4gIGNvbW1vbi52YWxpZGF0ZVBhcmFtcyhwYXJhbXMsIFsndHhpZCddLCBbJ2Zyb21Qcml2YXRlSW5mbyddKTtcbiAgY29uc3QgdHhpZCA9IHBhcmFtcy50eGlkO1xuICBjb25zdCByZWNpcGllbnQ6IFJlY2lwaWVudCB8IHVuZGVmaW5lZCA9IHBhcmFtcy5yZWNpcGllbnQ7XG4gIGxldCB0cmF2ZWxJbmZvID0gcGFyYW1zLnRyYXZlbEluZm87XG4gIGlmICghcmVjaXBpZW50IHx8ICFfLmlzT2JqZWN0KHJlY2lwaWVudCkpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ2ludmFsaWQgb3IgbWlzc2luZyByZWNpcGllbnQnKTtcbiAgfVxuICBpZiAoIXRyYXZlbEluZm8gfHwgIV8uaXNPYmplY3QodHJhdmVsSW5mbykpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ2ludmFsaWQgb3IgbWlzc2luZyB0cmF2ZWxJbmZvJyk7XG4gIH1cbiAgaWYgKCFwYXJhbXMubm9WYWxpZGF0ZSkge1xuICAgIHRyYXZlbEluZm8gPSB0aGlzLnZhbGlkYXRlVHJhdmVsSW5mbyh0cmF2ZWxJbmZvKTtcbiAgfVxuXG4gIC8vIEZpbGwgaW4gdG9FbnRlcnByaXNlIGlmIG5vdCBhbHJlYWR5IGZpbGxlZFxuICBpZiAoIXRyYXZlbEluZm8udG9FbnRlcnByaXNlICYmIHJlY2lwaWVudC5lbnRlcnByaXNlKSB7XG4gICAgdHJhdmVsSW5mby50b0VudGVycHJpc2UgPSByZWNpcGllbnQuZW50ZXJwcmlzZTtcbiAgfVxuXG4gIC8vIElmIGEga2V5IHdhcyBub3QgcHJvdmlkZWQsIGNyZWF0ZSBhIG5ldyByYW5kb20ga2V5XG4gIGxldCBmcm9tS2V5ID0gcGFyYW1zLmZyb21LZXkgJiYgdXR4b2xpYi5FQ1BhaXIuZnJvbVdJRihwYXJhbXMuZnJvbUtleSwgZ2V0TmV0d29yaygpIGFzIHV0eG9saWIuQml0Y29pbkpTTmV0d29yayk7XG4gIGlmICghZnJvbUtleSkge1xuICAgIGZyb21LZXkgPSBtYWtlUmFuZG9tS2V5KCk7XG4gIH1cblxuICAvLyBDb21wdXRlIHRoZSBzaGFyZWQga2V5IGZvciBlbmNyeXB0aW9uXG4gIGNvbnN0IHNoYXJlZFNlY3JldCA9IGdldFNoYXJlZFNlY3JldChmcm9tS2V5LCBCdWZmZXIuZnJvbShyZWNpcGllbnQucHViS2V5LCAnaGV4JykpLnRvU3RyaW5nKCdoZXgnKTtcblxuICAvLyBKU09OLWlmeSBhbmQgZW5jcnlwdCB0aGUgcGF5bG9hZFxuICBjb25zdCB0cmF2ZWxJbmZvSlNPTiA9IEpTT04uc3RyaW5naWZ5KHRyYXZlbEluZm8pO1xuICBjb25zdCBlbmNyeXB0ZWRUcmF2ZWxJbmZvID0gdGhpcy5iaXRnby5lbmNyeXB0KHtcbiAgICBpbnB1dDogdHJhdmVsSW5mb0pTT04sXG4gICAgcGFzc3dvcmQ6IHNoYXJlZFNlY3JldCxcbiAgfSk7XG5cbiAgY29uc3QgcmVzdWx0ID0ge1xuICAgIHR4aWQ6IHR4aWQsXG4gICAgb3V0cHV0SW5kZXg6IHJlY2lwaWVudC5vdXRwdXRJbmRleCxcbiAgICB0b1B1YktleTogcmVjaXBpZW50LnB1YktleSxcbiAgICBmcm9tUHViS2V5OiBmcm9tS2V5LnB1YmxpY0tleS50b1N0cmluZygnaGV4JyksXG4gICAgZW5jcnlwdGVkVHJhdmVsSW5mbzogZW5jcnlwdGVkVHJhdmVsSW5mbyxcbiAgICBmcm9tUHJpdmF0ZUluZm86IHVuZGVmaW5lZCxcbiAgfTtcblxuICBpZiAocGFyYW1zLmZyb21Qcml2YXRlSW5mbykge1xuICAgIHJlc3VsdC5mcm9tUHJpdmF0ZUluZm8gPSBwYXJhbXMuZnJvbVByaXZhdGVJbmZvO1xuICB9XG5cbiAgcmV0dXJuIHJlc3VsdDtcbn07XG5cbi8qKlxuICogU2VuZCB0cmF2ZWwgZGF0YSB0byB0aGUgc2VydmVyIGZvciBhIHRyYW5zYWN0aW9uXG4gKi9cblRyYXZlbFJ1bGUucHJvdG90eXBlLnNlbmQgPSBmdW5jdGlvbiAocGFyYW1zLCBjYWxsYmFjaykge1xuICBwYXJhbXMgPSBwYXJhbXMgfHwge307XG4gIHBhcmFtcy50eGlkID0gcGFyYW1zLnR4aWQgfHwgcGFyYW1zLmhhc2g7XG4gIGNvbW1vbi52YWxpZGF0ZVBhcmFtcyhcbiAgICBwYXJhbXMsXG4gICAgWyd0eGlkJywgJ3RvUHViS2V5JywgJ2VuY3J5cHRlZFRyYXZlbEluZm8nXSxcbiAgICBbJ2Zyb21QdWJLZXknLCAnZnJvbVByaXZhdGVJbmZvJ10sXG4gICAgY2FsbGJhY2tcbiAgKTtcblxuICBpZiAoIV8uaXNOdW1iZXIocGFyYW1zLm91dHB1dEluZGV4KSkge1xuICAgIHRocm93IG5ldyBFcnJvcignaW52YWxpZCBvdXRwdXRJbmRleCcpO1xuICB9XG5cbiAgcmV0dXJuIFByb21pc2UucmVzb2x2ZShcbiAgICB0aGlzLmJpdGdvXG4gICAgICAucG9zdCh0aGlzLnVybChwYXJhbXMudHhpZCArICcvJyArIHBhcmFtcy5vdXRwdXRJbmRleCkpXG4gICAgICAuc2VuZChwYXJhbXMpXG4gICAgICAucmVzdWx0KClcbiAgKVxuICAgIC50aGVuKGNhbGxiYWNrKVxuICAgIC5jYXRjaChjYWxsYmFjayk7XG59O1xuXG4vKipcbiAqIFNlbmQgbXVsdGlwbGUgdHJhdmVsIHJ1bGUgaW5mb3MgZm9yIHRoZSBvdXRwdXRzIG9mIGEgc2luZ2xlIHRyYW5zYWN0aW9uLlxuICogUGFyYW1ldGVyczpcbiAqICAgLSB0eGlkIChvciBoYXNoKTogdHhpZCBvZiB0aGUgdHJhbnNhY3Rpb24gKG11c3QgYmUgYSBzZW5kZXIgb2YgdGhlIHR4KVxuICogICAtIHRyYXZlbEluZm9zOiBhcnJheSBvZiB0cmF2ZWxJbmZvIG9iamVjdHMgd2hpY2ggbG9vayBsaWtlIHRoZSBmb2xsb3dpbmc6XG4gKiAgICAge1xuICogICAgICAgb3V0cHV0SW5kZXg6IG51bWJlciwgICAgIC8vIHR4IG91dHB1dCBpbmRleFxuICogICAgICAgZnJvbVVzZXJOYW1lOiBzdHJpbmcsICAgIC8vIG5hbWUgb2YgdGhlIHNlbmRpbmcgdXNlclxuICogICAgICAgZnJvbVVzZXJBY2NvdW50OiBzdHJpbmcsIC8vIGFjY291bnQgaWQgb2YgdGhlIHNlbmRpbmcgdXNlclxuICogICAgICAgZnJvbVVzZXJBZGRyZXNzOiBzdHJpbmcsIC8vIG1haWxpbmcgYWRkcmVzcyBvZiB0aGUgc2VuZGluZyB1c2VyXG4gKiAgICAgICB0b1VzZXJOYW1lOiBzdHJpbmcsICAgICAgLy8gbmFtZSBvZiB0aGUgcmVjZWl2aW5nIHVzZXJcbiAqICAgICAgIHRvVXNlckFjY291bnQ6IHN0cmluZywgICAvLyBhY2NvdW50IGlkIG9mIHRoZSByZWNlaXZpbmcgdXNlclxuICogICAgICAgdG9Vc2VyQWRkcmVzczogc3RyaW5nICAgIC8vIG1haWxpbmcgYWRkcmVzcyBvZiB0aGUgcmVjZWl2aW5nIHVzZXJcbiAqICAgICB9XG4gKiAgICAgQWxsIGZpZWxkcyBhc2lkZSBmcm9tIG91dHB1dEluZGV4IGFyZSBvcHRpb25hbCwgYnV0IGF0IGxlYXN0IG9uZSBtdXN0XG4gKiAgICAgYmUgZGVmaW5lZC5cbiAqXG4gKiAgSXQgaXMgbm90IG5lY2Vzc2FyeSB0byBwcm92aWRlIHRyYXZlbEluZm8gZm9yIGFsbCBvdXRwdXQgaW5kaWNlcy5cbiAqICBFbmQtdG8tZW5kIGVuY3J5cHRpb24gb2YgdGhlIHRyYXZlbCBpbmZvIGlzIGhhbmRsZWQgYXV0b21hdGljYWxseSBieSB0aGlzIG1ldGhvZC5cbiAqXG4gKi9cblRyYXZlbFJ1bGUucHJvdG90eXBlLnNlbmRNYW55ID0gZnVuY3Rpb24gKHBhcmFtcywgY2FsbGJhY2spIHtcbiAgcGFyYW1zID0gcGFyYW1zIHx8IHt9O1xuICBwYXJhbXMudHhpZCA9IHBhcmFtcy50eGlkIHx8IHBhcmFtcy5oYXNoO1xuICBjb21tb24udmFsaWRhdGVQYXJhbXMocGFyYW1zLCBbJ3R4aWQnXSwgY2FsbGJhY2spO1xuXG4gIGNvbnN0IHRyYXZlbEluZm9zID0gcGFyYW1zLnRyYXZlbEluZm9zO1xuICBpZiAoIV8uaXNBcnJheSh0cmF2ZWxJbmZvcykpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ2V4cGVjdGVkIHBhcmFtZXRlciB0cmF2ZWxJbmZvcyB0byBiZSBhcnJheScpO1xuICB9XG5cbiAgY29uc3Qgc2VsZiA9IHRoaXM7XG4gIGNvbnN0IHRyYXZlbEluZm9NYXAgPSBfKHRyYXZlbEluZm9zKVxuICAgIC5rZXlCeSgnb3V0cHV0SW5kZXgnKVxuICAgIC5tYXBWYWx1ZXMoZnVuY3Rpb24gKHRyYXZlbEluZm8pIHtcbiAgICAgIHJldHVybiBzZWxmLnZhbGlkYXRlVHJhdmVsSW5mbyh0cmF2ZWxJbmZvKTtcbiAgICB9KVxuICAgIC52YWx1ZSgpO1xuXG4gIHJldHVybiBzZWxmLmdldFJlY2lwaWVudHMoeyB0eGlkOiBwYXJhbXMudHhpZCB9KS50aGVuKGZ1bmN0aW9uIChyZWNpcGllbnRzKSB7XG4gICAgLy8gQnVpbGQgdXAgZGF0YSB0byBwb3N0XG4gICAgY29uc3Qgc2VuZFBhcmFtc0xpc3Q6IGFueVtdID0gW107XG4gICAgLy8gZG9uJ3QgcmVnZW5lcmF0ZSBhIG5ldyByYW5kb20ga2V5IGZvciBlYWNoIHJlY2lwaWVudFxuICAgIGNvbnN0IGZyb21LZXkgPSBwYXJhbXMuZnJvbUtleSB8fCBtYWtlUmFuZG9tS2V5KCkudG9XSUYoKTtcblxuICAgIHJlY2lwaWVudHMuZm9yRWFjaChmdW5jdGlvbiAocmVjaXBpZW50KSB7XG4gICAgICBjb25zdCBvdXRwdXRJbmRleCA9IHJlY2lwaWVudC5vdXRwdXRJbmRleDtcbiAgICAgIGNvbnN0IGluZm8gPSB0cmF2ZWxJbmZvTWFwW291dHB1dEluZGV4XTtcbiAgICAgIGlmIChpbmZvKSB7XG4gICAgICAgIGlmIChpbmZvLmFtb3VudCAmJiBpbmZvLmFtb3VudCAhPT0gcmVjaXBpZW50LmFtb3VudCkge1xuICAgICAgICAgIHRocm93IG5ldyBFcnJvcignYW1vdW50IGRpZCBub3QgbWF0Y2ggZm9yIG91dHB1dCBpbmRleCAnICsgb3V0cHV0SW5kZXgpO1xuICAgICAgICB9XG4gICAgICAgIGNvbnN0IHNlbmRQYXJhbXMgPSBzZWxmLnByZXBhcmVQYXJhbXMoe1xuICAgICAgICAgIHR4aWQ6IHBhcmFtcy50eGlkLFxuICAgICAgICAgIHJlY2lwaWVudDogcmVjaXBpZW50LFxuICAgICAgICAgIHRyYXZlbEluZm86IGluZm8sXG4gICAgICAgICAgZnJvbUtleTogZnJvbUtleSxcbiAgICAgICAgICBub1ZhbGlkYXRlOiB0cnVlLCAvLyBkb24ndCByZS12YWxpZGF0ZVxuICAgICAgICB9KTtcbiAgICAgICAgc2VuZFBhcmFtc0xpc3QucHVzaChzZW5kUGFyYW1zKTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIGNvbnN0IHJlc3VsdDoge1xuICAgICAgbWF0Y2hlZDogbnVtYmVyO1xuICAgICAgcmVzdWx0czoge1xuICAgICAgICByZXN1bHQ/OiBhbnk7XG4gICAgICAgIGVycm9yPzogc3RyaW5nO1xuICAgICAgfVtdO1xuICAgIH0gPSB7XG4gICAgICBtYXRjaGVkOiBzZW5kUGFyYW1zTGlzdC5sZW5ndGgsXG4gICAgICByZXN1bHRzOiBbXSxcbiAgICB9O1xuXG4gICAgY29uc3Qgc2VuZFNlcmlhbCA9IGZ1bmN0aW9uICgpIHtcbiAgICAgIGNvbnN0IHNlbmRQYXJhbXMgPSBzZW5kUGFyYW1zTGlzdC5zaGlmdCgpO1xuICAgICAgaWYgKCFzZW5kUGFyYW1zKSB7XG4gICAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgICB9XG4gICAgICByZXR1cm4gc2VsZlxuICAgICAgICAuc2VuZChzZW5kUGFyYW1zKVxuICAgICAgICAudGhlbihmdW5jdGlvbiAocmVzKSB7XG4gICAgICAgICAgcmVzdWx0LnJlc3VsdHMucHVzaCh7IHJlc3VsdDogcmVzIH0pO1xuICAgICAgICAgIHJldHVybiBzZW5kU2VyaWFsKCk7XG4gICAgICAgIH0pXG4gICAgICAgIC5jYXRjaChmdW5jdGlvbiAoZXJyKSB7XG4gICAgICAgICAgcmVzdWx0LnJlc3VsdHMucHVzaCh7IGVycm9yOiBlcnIudG9TdHJpbmcoKSB9KTtcbiAgICAgICAgICByZXR1cm4gc2VuZFNlcmlhbCgpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgcmV0dXJuIHNlbmRTZXJpYWwoKTtcbiAgfSk7XG59O1xuXG5tb2R1bGUuZXhwb3J0cyA9IFRyYXZlbFJ1bGU7XG4iXX0=

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


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