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=Выполнить команду
Для локальной разработки. Не используйте в интернете!