PHP WebShell
Текущая директория: /opt/BitGoJS/modules/sdk-coin-ada/dist/src/lib
Просмотр файла: utils.js
"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.Utils = exports.VOTE_ALWAYS_NO_CONFIDENCE = exports.VOTE_ALWAYS_ABSTAIN = exports.MIN_ADA_FOR_ONE_ASSET = void 0;
const sdk_core_1 = require("@bitgo/sdk-core");
const cardano_serialization_lib_nodejs_1 = require("@emurgo/cardano-serialization-lib-nodejs");
const keyPair_1 = require("./keyPair");
const bech32_1 = require("bech32");
const bs58_1 = __importDefault(require("bs58"));
const cbor_1 = __importDefault(require("cbor"));
exports.MIN_ADA_FOR_ONE_ASSET = '1500000';
exports.VOTE_ALWAYS_ABSTAIN = 'always-abstain';
exports.VOTE_ALWAYS_NO_CONFIDENCE = 'always-no-confidence';
class Utils {
createBaseAddressWithStakeAndPaymentKey(stakeKeyPair, paymentKeyPair, network) {
let baseAddr;
if (network === sdk_core_1.AddressFormat.mainnet) {
// 1. create stake pubKey
const key = stakeKeyPair.getKeys().pub;
const stakePub = cardano_serialization_lib_nodejs_1.PublicKey.from_bytes(Buffer.from(key, 'hex'));
// 2. create payment pubKey
const paymentPub = cardano_serialization_lib_nodejs_1.PublicKey.from_bytes(Buffer.from(paymentKeyPair.getKeys().pub, 'hex'));
// 3. create full base address for staking
baseAddr = cardano_serialization_lib_nodejs_1.BaseAddress.new(cardano_serialization_lib_nodejs_1.NetworkInfo.mainnet().network_id(), cardano_serialization_lib_nodejs_1.Credential.from_keyhash(paymentPub.hash()), cardano_serialization_lib_nodejs_1.Credential.from_keyhash(stakePub.hash()));
return baseAddr.to_address().to_bech32();
}
else if (network === sdk_core_1.AddressFormat.testnet) {
// 1. create stake pubKey
const stakePub = cardano_serialization_lib_nodejs_1.PublicKey.from_bytes(Buffer.from(stakeKeyPair.getKeys().pub, 'hex'));
// 2. create payment pubKey
const paymentPub = cardano_serialization_lib_nodejs_1.PublicKey.from_bytes(Buffer.from(paymentKeyPair.getKeys().pub, 'hex'));
// 3. create full base address for staking
const baseAddr = cardano_serialization_lib_nodejs_1.BaseAddress.new(cardano_serialization_lib_nodejs_1.NetworkInfo.testnet_preprod().network_id(), cardano_serialization_lib_nodejs_1.Credential.from_keyhash(paymentPub.hash()), cardano_serialization_lib_nodejs_1.Credential.from_keyhash(stakePub.hash()));
return baseAddr.to_address().to_bech32();
}
else {
throw new Error('Improper Network Type!');
}
}
validateBlake2b(hash) {
if (!hash) {
return false;
}
if (hash.length !== 64) {
return false;
}
return hash.match(/^[a-zA-Z0-9]+$/) !== null;
}
getRewardAddress(stakingPubKey, coinName) {
const stakePub = cardano_serialization_lib_nodejs_1.PublicKey.from_bytes(Buffer.from(stakingPubKey, 'hex'));
let rewardAddress;
if (coinName === 'ada') {
rewardAddress = cardano_serialization_lib_nodejs_1.RewardAddress.new(cardano_serialization_lib_nodejs_1.NetworkInfo.mainnet().network_id(), cardano_serialization_lib_nodejs_1.Credential.from_keyhash(stakePub.hash()));
}
else {
rewardAddress = cardano_serialization_lib_nodejs_1.RewardAddress.new(cardano_serialization_lib_nodejs_1.NetworkInfo.testnet_preprod().network_id(), cardano_serialization_lib_nodejs_1.Credential.from_keyhash(stakePub.hash()));
}
return rewardAddress.to_address().to_bech32();
}
isValidDRepId(dRepId) {
try {
this.getDRepFromDRepId(dRepId);
return true;
}
catch (err) {
return false;
}
}
getDRepFromDRepId(dRepId) {
switch (dRepId) {
case 'always-abstain':
return cardano_serialization_lib_nodejs_1.DRep.new_always_abstain();
case 'always-no-confidence':
return cardano_serialization_lib_nodejs_1.DRep.new_always_no_confidence();
default:
try {
// for parsing CIP-105 standard DRep ID
return cardano_serialization_lib_nodejs_1.DRep.from_bech32(dRepId);
}
catch (err) {
// for parsing CIP-129 standard DRep ID
// https://cips.cardano.org/cip/CIP-0129
const decodedBech32 = bech32_1.bech32.decode(dRepId);
const decodedBytes = Buffer.from(bech32_1.bech32.fromWords(decodedBech32.words));
const header = decodedBytes[0];
const keyBytes = decodedBytes.subarray(1);
const keyType = (header & 0xf0) >> 4;
const credentialType = header & 0x0f;
if (keyType !== 0x02) {
throw new Error('Invalid key type for DRep');
}
switch (credentialType) {
case 0x02:
const ed25519KeyHash = cardano_serialization_lib_nodejs_1.Ed25519KeyHash.from_bytes(keyBytes);
return cardano_serialization_lib_nodejs_1.DRep.new_key_hash(ed25519KeyHash);
case 0x03:
const scriptHash = cardano_serialization_lib_nodejs_1.ScriptHash.from_bytes(keyBytes);
return cardano_serialization_lib_nodejs_1.DRep.new_script_hash(scriptHash);
default:
throw new Error('Invalid credential type for DRep');
}
}
}
}
getDRepIdFromDRep(dRep) {
switch (dRep.kind()) {
case cardano_serialization_lib_nodejs_1.DRepKind.AlwaysAbstain:
return exports.VOTE_ALWAYS_ABSTAIN;
case cardano_serialization_lib_nodejs_1.DRepKind.AlwaysNoConfidence:
return exports.VOTE_ALWAYS_NO_CONFIDENCE;
default:
return dRep.to_bech32();
}
}
/** @inheritdoc */
// this will validate both stake and payment addresses
isValidAddress(address) {
const bech32PrefixList = ['addr', 'addr_test', 'stake', 'stake_test'];
const BASE_ADDR_LEN = 92;
const REWARD_AND_ENTERPRISE_ADDR_LEN = 47;
const POINTER_ADDR_LEN = 52;
const VALIDATOR_ADDR_LEN = 56;
//Check for Shelley-era (Bech32) addresses
if (new RegExp(`^(${bech32PrefixList.join('|')})`).test(address)) {
try {
const decodedBech = bech32_1.bech32.decode(address, 108);
const wordLength = decodedBech.words.length;
if (bech32PrefixList.includes(decodedBech.prefix) &&
(wordLength === BASE_ADDR_LEN ||
wordLength === REWARD_AND_ENTERPRISE_ADDR_LEN ||
wordLength === POINTER_ADDR_LEN)) {
return true;
}
}
catch (e) {
console.log(`Address: ${address} failed Bech32 test with error: ${e}`);
}
}
//Check for Validator addresses
if (new RegExp(`^(?!pool)[a-z0-9]{${VALIDATOR_ADDR_LEN}}$`).test(address)) {
return true;
}
//Check for Byron-era address
try {
// Reject Daedalus wallet addresses (Byron-era addresses starting with "DdzFF")
if (address.startsWith('DdzFF')) {
console.log(`Rejecting Daedalus wallet address: ${address}`);
return false;
}
const decoded = bs58_1.default.decode(address);
const cborData = cbor_1.default.decodeFirstSync(decoded);
return Array.isArray(cborData) && cborData.length >= 2;
}
catch (e) {
console.log(`Address: ${address} failed Byron test with error: ${e}`);
console.log(e.stack);
}
return false;
}
/** @inheritdoc */
isValidBlockId(hash) {
return this.validateBlake2b(hash);
}
/** @inheritdoc */
isValidPrivateKey(key) {
// this will return true for both extended and non-extended ED25519 keys
return this.isValidKey(key);
}
isValidKey(key) {
try {
new keyPair_1.KeyPair({ prv: key });
return true;
}
catch {
return false;
}
}
/** @inheritdoc */
isValidPublicKey(pubKey) {
try {
new keyPair_1.KeyPair({ pub: pubKey });
return true;
}
catch {
return false;
}
}
/** @inheritdoc */
isValidSignature(signature) {
try {
cardano_serialization_lib_nodejs_1.Ed25519Signature.from_hex(signature);
return true;
}
catch (err) {
return false;
}
}
/** @inheritdoc */
isValidTransactionId(txId) {
return this.validateBlake2b(txId);
}
/**
* Get the transaction body from a serialized transaction
* @param {string} serializedTx - serialized transaction in hex or base64 format
* @returns {string} transaction body in hex format
*/
getTransactionBody(serializedTx) {
const HEX_REGEX = /^[0-9a-fA-F]+$/;
const bufferRawTransaction = HEX_REGEX.test(serializedTx)
? Buffer.from(serializedTx, 'hex')
: Buffer.from(serializedTx, 'base64');
return Buffer.from(cardano_serialization_lib_nodejs_1.Transaction.from_bytes(bufferRawTransaction).body().to_bytes()).toString('hex');
}
/**
* Decode wallet address from string.
* Attempts to decode as Shelley (bech32) first, then Byron (base58).
* @param {string} address - Valid Byron or Shelley-era address.
* @returns {Address} - Valid address object.
* @throws {InvalidAddressError} If the address is neither valid Shelley nor Byron.
*/
getWalletAddress(address) {
if (!address || typeof address !== 'string') {
throw new sdk_core_1.InvalidAddressError('Provided address is not a valid string');
}
// Try decoding as a Shelley (bech32) address first
try {
return cardano_serialization_lib_nodejs_1.Address.from_bech32(address);
}
catch (e) {
console.error(`Could not decode shelly address from string '${address}'`);
}
// Try decoding as a Byron (base58) address later
try {
// Reject Daedalus wallet addresses (Byron-era addresses starting with "DdzFF")
if (address.startsWith('DdzFF')) {
throw new sdk_core_1.InvalidAddressError('Provided string is a Daedalus address');
}
return cardano_serialization_lib_nodejs_1.ByronAddress.from_base58(address).to_address();
}
catch (e) {
console.error(`Could not decode byron address from string '${address}'`);
}
throw new sdk_core_1.InvalidAddressError('Provided string is not a valid Shelley or Byron address');
}
/**
* Decode address string from Address object.
* Attempts to decode as Shelley (bech32) first, then Byron (base58).
* @param {Address} address - Valid Address object
* @returns {string} - Valid Byron or Shelley-era address string.
* @throws {InvalidAddressError} If the Address object is neither valid Shelley nor Byron.
*/
getAddressString(address) {
// Check all Shelley address types
if (cardano_serialization_lib_nodejs_1.BaseAddress.from_address(address) ||
cardano_serialization_lib_nodejs_1.EnterpriseAddress.from_address(address) ||
cardano_serialization_lib_nodejs_1.RewardAddress.from_address(address) ||
cardano_serialization_lib_nodejs_1.PointerAddress.from_address(address)) {
return address.to_bech32();
}
const byronAddress = cardano_serialization_lib_nodejs_1.ByronAddress.from_address(address);
// Reject Daedalus wallet addresses (Byron-era addresses starting with "DdzFF")
if (byronAddress) {
if (byronAddress.to_base58().startsWith('DdzFF')) {
throw new sdk_core_1.InvalidAddressError('Provided address is a Daedalus address');
}
return byronAddress.to_base58();
}
// If neither, it's invalid
throw new sdk_core_1.InvalidAddressError('Provided Address is not a valid Shelley or Byron address');
}
}
exports.Utils = Utils;
const utils = new Utils();
exports.default = utils;
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL3V0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7OztBQUFBLDhDQUFnRjtBQUNoRiwrRkFnQmtEO0FBQ2xELHVDQUFvQztBQUNwQyxtQ0FBZ0M7QUFDaEMsZ0RBQXdCO0FBQ3hCLGdEQUF3QjtBQUVYLFFBQUEscUJBQXFCLEdBQUcsU0FBUyxDQUFDO0FBQ2xDLFFBQUEsbUJBQW1CLEdBQUcsZ0JBQWdCLENBQUM7QUFDdkMsUUFBQSx5QkFBeUIsR0FBRyxzQkFBc0IsQ0FBQztBQUVoRSxNQUFhLEtBQUs7SUFDaEIsdUNBQXVDLENBQ3JDLFlBQXFCLEVBQ3JCLGNBQXVCLEVBQ3ZCLE9BQXNCO1FBRXRCLElBQUksUUFBUSxDQUFDO1FBQ2IsSUFBSSxPQUFPLEtBQUssd0JBQWEsQ0FBQyxPQUFPLEVBQUUsQ0FBQztZQUN0Qyx5QkFBeUI7WUFDekIsTUFBTSxHQUFHLEdBQUcsWUFBWSxDQUFDLE9BQU8sRUFBRSxDQUFDLEdBQUcsQ0FBQztZQUV2QyxNQUFNLFFBQVEsR0FBRyw0Q0FBUyxDQUFDLFVBQVUsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDO1lBQy9ELDJCQUEyQjtZQUMzQixNQUFNLFVBQVUsR0FBRyw0Q0FBUyxDQUFDLFVBQVUsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLGNBQWMsQ0FBQyxPQUFPLEVBQUUsQ0FBQyxHQUFHLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQztZQUMxRiwwQ0FBMEM7WUFDMUMsUUFBUSxHQUFHLDhDQUFXLENBQUMsR0FBRyxDQUN4Qiw4Q0FBVyxDQUFDLE9BQU8sRUFBRSxDQUFDLFVBQVUsRUFBRSxFQUNsQyw2Q0FBVSxDQUFDLFlBQVksQ0FBQyxVQUFVLENBQUMsSUFBSSxFQUFFLENBQUMsRUFDMUMsNkNBQVUsQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLElBQUks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QUFDO1lBQ1gsT0FBTyxDQUFDLEtBQUssQ0FBQywrQ0FBK0MsT0FBTyxHQUFHLENBQUMsQ0FBQztRQUMzRSxDQUFDO1FBQ0QsTUFBTSxJQUFJLDhCQUFtQixDQUFDLHlEQUF5RCxDQUFDLENBQUM7SUFDM0YsQ0FBQztJQUVEOzs7Ozs7T0FNRztJQUNILGdCQUFnQixDQUFDLE9BQWdCO1FBQy9CLGtDQUFrQztRQUNsQyxJQUNFLDhDQUFXLENBQUMsWUFBWSxDQUFDLE9BQU8sQ0FBQztZQUNqQyxvREFBaUIsQ0FBQyxZQUFZLENBQUMsT0FBTyxDQUFDO1lBQ3ZDLGdEQUFhLENBQUMsWUFBWSxDQUFDLE9BQU8sQ0FBQztZQUNuQyxpREFBYyxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsRUFDcEMsQ0FBQztZQUNELE9BQU8sT0FBTyxDQUFDLFNBQVMsRUFBRSxDQUFDO1FBQzdCLENBQUM7UUFFRCxNQUFNLFlBQVksR0FBRywrQ0FBWSxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUN4RCwrRUFBK0U7UUFDL0UsSUFBSSxZQUFZLEVBQUUsQ0FBQztZQUNqQixJQUFJLFlBQVksQ0FBQyxTQUFTLEVBQUUsQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztnQkFDakQsTUFBTSxJQUFJLDhCQUFtQixDQUFDLHdDQUF3QyxDQUFDLENBQUM7WUFDMUUsQ0FBQztZQUNELE9BQU8sWUFBWSxDQUFDLFNBQVMsRUFBRSxDQUFDO1FBQ2xDLENBQUM7UUFFRCwyQkFBMkI7UUFDM0IsTUFBTSxJQUFJLDhCQUFtQixDQUFDLDBEQUEwRCxDQUFDLENBQUM7SUFDNUYsQ0FBQztDQUNGO0FBclNELHNCQXFTQztBQUVELE1BQU0sS0FBSyxHQUFHLElBQUksS0FBSyxFQUFFLENBQUM7QUFFMUIsa0JBQWUsS0FBSyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQWRkcmVzc0Zvcm1hdCwgQmFzZVV0aWxzLCBJbnZhbGlkQWRkcmVzc0Vycm9yIH0gZnJvbSAnQGJpdGdvL3Nkay1jb3JlJztcbmltcG9ydCB7XG4gIEJhc2VBZGRyZXNzLFxuICBQdWJsaWNLZXksXG4gIEVkMjU1MTlTaWduYXR1cmUsXG4gIE5ldHdvcmtJbmZvLFxuICBDcmVkZW50aWFsLFxuICBSZXdhcmRBZGRyZXNzLFxuICBUcmFuc2FjdGlvbiBhcyBDYXJkYW5vVHJhbnNhY3Rpb24sXG4gIERSZXAsXG4gIEVkMjU1MTlLZXlIYXNoLFxuICBTY3JpcHRIYXNoLFxuICBEUmVwS2luZCxcbiAgQWRkcmVzcyxcbiAgRW50ZXJwcmlzZUFkZHJlc3MsXG4gIFBvaW50ZXJBZGRyZXNzLFxuICBCeXJvbkFkZHJlc3MsXG59IGZyb20gJ0BlbXVyZ28vY2FyZGFuby1zZXJpYWxpemF0aW9uLWxpYi1ub2RlanMnO1xuaW1wb3J0IHsgS2V5UGFpciB9IGZyb20gJy4va2V5UGFpcic7XG5pbXBvcnQgeyBiZWNoMzIgfSBmcm9tICdiZWNoMzInO1xuaW1wb3J0IGJzNTggZnJvbSAnYnM1OCc7XG5pbXBvcnQgY2JvciBmcm9tICdjYm9yJztcblxuZXhwb3J0IGNvbnN0IE1JTl9BREFfRk9SX09ORV9BU1NFVCA9ICcxNTAwMDAwJztcbmV4cG9ydCBjb25zdCBWT1RFX0FMV0FZU19BQlNUQUlOID0gJ2Fsd2F5cy1hYnN0YWluJztcbmV4cG9ydCBjb25zdCBWT1RFX0FMV0FZU19OT19DT05GSURFTkNFID0gJ2Fsd2F5cy1uby1jb25maWRlbmNlJztcblxuZXhwb3J0IGNsYXNzIFV0aWxzIGltcGxlbWVudHMgQmFzZVV0aWxzIHtcbiAgY3JlYXRlQmFzZUFkZHJlc3NXaXRoU3Rha2VBbmRQYXltZW50S2V5KFxuICAgIHN0YWtlS2V5UGFpcjogS2V5UGFpcixcbiAgICBwYXltZW50S2V5UGFpcjogS2V5UGFpcixcbiAgICBuZXR3b3JrOiBBZGRyZXNzRm9ybWF0XG4gICk6IHN0cmluZyB7XG4gICAgbGV0IGJhc2VBZGRyO1xuICAgIGlmIChuZXR3b3JrID09PSBBZGRyZXNzRm9ybWF0Lm1haW5uZXQpIHtcbiAgICAgIC8vIDEuIGNyZWF0ZSBzdGFrZSBwdWJLZXlcbiAgICAgIGNvbnN0IGtleSA9IHN0YWtlS2V5UGFpci5nZXRLZXlzKCkucHViO1xuXG4gICAgICBjb25zdCBzdGFrZVB1YiA9IFB1YmxpY0tleS5mcm9tX2J5dGVzKEJ1ZmZlci5mcm9tKGtleSwgJ2hleCcpKTtcbiAgICAgIC8vIDIuIGNyZWF0ZSBwYXltZW50IHB1YktleVxuICAgICAgY29uc3QgcGF5bWVudFB1YiA9IFB1YmxpY0tleS5mcm9tX2J5dGVzKEJ1ZmZlci5mcm9tKHBheW1lbnRLZXlQYWlyLmdldEtleXMoKS5wdWIsICdoZXgnKSk7XG4gICAgICAvLyAzLiBjcmVhdGUgZnVsbCBiYXNlIGFkZHJlc3MgZm9yIHN0YWtpbmdcbiAgICAgIGJhc2VBZGRyID0gQmFzZUFkZHJlc3MubmV3KFxuICAgICAgICBOZXR3b3JrSW5mby5tYWlubmV0KCkubmV0d29ya19pZCgpLFxuICAgICAgICBDcmVkZW50aWFsLmZyb21fa2V5aGFzaChwYXltZW50UHViLmhhc2goKSksXG4gICAgICAgIENyZWRlbnRpYWwuZnJvbV9rZXloYXNoKHN0YWtlUHViLmhhc2goKSlcbiAgICAgICk7XG4gICAgICByZXR1cm4gYmFzZUFkZHIudG9fYWRkcmVzcygpLnRvX2JlY2gzMigpO1xuICAgIH0gZWxzZSBpZiAobmV0d29yayA9PT0gQWRkcmVzc0Zvcm1hdC50ZXN0bmV0KSB7XG4gICAgICAvLyAxLiBjcmVhdGUgc3Rha2UgcHViS2V5XG4gICAgICBjb25zdCBzdGFrZVB1YiA9IFB1YmxpY0tleS5mcm9tX2J5dGVzKEJ1ZmZlci5mcm9tKHN0YWtlS2V5UGFpci5nZXRLZXlzKCkucHViLCAnaGV4JykpO1xuICAgICAgLy8gMi4gY3JlYXRlIHBheW1lbnQgcHViS2V5XG4gICAgICBjb25zdCBwYXltZW50UHViID0gUHVibGljS2V5LmZyb21fYnl0ZXMoQnVmZmVyLmZyb20ocGF5bWVudEtleVBhaXIuZ2V0S2V5cygpLnB1YiwgJ2hleCcpKTtcbiAgICAgIC8vIDMuIGNyZWF0ZSBmdWxsIGJhc2UgYWRkcmVzcyBmb3Igc3Rha2luZ1xuICAgICAgY29uc3QgYmFzZUFkZHIgPSBCYXNlQWRkcmVzcy5uZXcoXG4gICAgICAgIE5ldHdvcmtJbmZvLnRlc3RuZXRfcHJlcHJvZCgpLm5ldHdvcmtfaWQoKSxcbiAgICAgICAgQ3JlZGVudGlhbC5mcm9tX2tleWhhc2gocGF5bWVudFB1Yi5oYXNoKCkpLFxuICAgICAgICBDcmVkZW50aWFsLmZyb21fa2V5aGFzaChzdGFrZVB1Yi5oYXNoKCkpXG4gICAgICApO1xuICAgICAgcmV0dXJuIGJhc2VBZGRyLnRvX2FkZHJlc3MoKS50b19iZWNoMzIoKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCdJbXByb3BlciBOZXR3b3JrIFR5cGUhJyk7XG4gICAgfVxuICB9XG5cbiAgdmFsaWRhdGVCbGFrZTJiKGhhc2g6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIGlmICghaGFzaCkge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgICBpZiAoaGFzaC5sZW5ndGggIT09IDY0KSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICAgIHJldHVybiBoYXNoLm1hdGNoKC9eW2EtekEtWjAtOV0rJC8pICE9PSBudWxsO1xuICB9XG5cbiAgZ2V0UmV3YXJkQWRkcmVzcyhzdGFraW5nUHViS2V5OiBzdHJpbmcsIGNvaW5OYW1lOiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIGNvbnN0IHN0YWtlUHViID0gUHVibGljS2V5LmZyb21fYnl0ZXMoQnVmZmVyLmZyb20oc3Rha2luZ1B1YktleSwgJ2hleCcpKTtcbiAgICBsZXQgcmV3YXJkQWRkcmVzcztcbiAgICBpZiAoY29pbk5hbWUgPT09ICdhZGEnKSB7XG4gICAgICByZXdhcmRBZGRyZXNzID0gUmV3YXJkQWRkcmVzcy5uZXcoTmV0d29ya0luZm8ubWFpbm5ldCgpLm5ldHdvcmtfaWQoKSwgQ3JlZGVudGlhbC5mcm9tX2tleWhhc2goc3Rha2VQdWIuaGFzaCgpKSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHJld2FyZEFkZHJlc3MgPSBSZXdhcmRBZGRyZXNzLm5ldyhcbiAgICAgICAgTmV0d29ya0luZm8udGVzdG5ldF9wcmVwcm9kKCkubmV0d29ya19pZCgpLFxuICAgICAgICBDcmVkZW50aWFsLmZyb21fa2V5aGFzaChzdGFrZVB1Yi5oYXNoKCkpXG4gICAgICApO1xuICAgIH1cbiAgICByZXR1cm4gcmV3YXJkQWRkcmVzcy50b19hZGRyZXNzKCkudG9fYmVjaDMyKCk7XG4gIH1cblxuICBpc1ZhbGlkRFJlcElkKGRSZXBJZDogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgdHJ5IHtcbiAgICAgIHRoaXMuZ2V0RFJlcEZyb21EUmVwSWQoZFJlcElkKTtcbiAgICAgIHJldHVybiB0cnVlO1xuICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgfVxuXG4gIGdldERSZXBGcm9tRFJlcElkKGRSZXBJZDogc3RyaW5nKTogRFJlcCB7XG4gICAgc3dpdGNoIChkUmVwSWQpIHtcbiAgICAgIGNhc2UgJ2Fsd2F5cy1hYnN0YWluJzpcbiAgICAgICAgcmV0dXJuIERSZXAubmV3X2Fsd2F5c19hYnN0YWluKCk7XG4gICAgICBjYXNlICdhbHdheXMtbm8tY29uZmlkZW5jZSc6XG4gICAgICAgIHJldHVybiBEUmVwLm5ld19hbHdheXNfbm9fY29uZmlkZW5jZSgpO1xuICAgICAgZGVmYXVsdDpcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAvLyBmb3IgcGFyc2luZyBDSVAtMTA1IHN0YW5kYXJkIERSZXAgSURcbiAgICAgICAgICByZXR1cm4gRFJlcC5mcm9tX2JlY2gzMihkUmVwSWQpO1xuICAgICAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgICAgICAvLyBmb3IgcGFyc2luZyBDSVAtMTI5IHN0YW5kYXJkIERSZXAgSURcbiAgICAgICAgICAvLyBodHRwczovL2NpcHMuY2FyZGFuby5vcmcvY2lwL0NJUC0wMTI5XG4gICAgICAgICAgY29uc3QgZGVjb2RlZEJlY2gzMiA9IGJlY2gzMi5kZWNvZGUoZFJlcElkKTtcbiAgICAgICAgICBjb25zdCBkZWNvZGVkQnl0ZXMgPSBCdWZmZXIuZnJvbShiZWNoMzIuZnJvbVdvcmRzKGRlY29kZWRCZWNoMzIud29yZHMpKTtcbiAgICAgICAgICBjb25zdCBoZWFkZXIgPSBkZWNvZGVkQnl0ZXNbMF07XG4gICAgICAgICAgY29uc3Qga2V5Qnl0ZXMgPSBkZWNvZGVkQnl0ZXMuc3ViYXJyYXkoMSk7XG5cbiAgICAgICAgICBjb25zdCBrZXlUeXBlID0gKGhlYWRlciAmIDB4ZjApID4+IDQ7XG4gICAgICAgICAgY29uc3QgY3JlZGVudGlhbFR5cGUgPSBoZWFkZXIgJiAweDBmO1xuXG4gICAgICAgICAgaWYgKGtleVR5cGUgIT09IDB4MDIpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcignSW52YWxpZCBrZXkgdHlwZSBmb3IgRFJlcCcpO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHN3aXRjaCAoY3JlZGVudGlhbFR5cGUpIHtcbiAgICAgICAgICAgIGNhc2UgMHgwMjpcbiAgICAgICAgICAgICAgY29uc3QgZWQyNTUxOUtleUhhc2ggPSBFZDI1NTE5S2V5SGFzaC5mcm9tX2J5dGVzKGtleUJ5dGVzKTtcbiAgICAgICAgICAgICAgcmV0dXJuIERSZXAubmV3X2tleV9oYXNoKGVkMjU1MTlLZXlIYXNoKTtcbiAgICAgICAgICAgIGNhc2UgMHgwMzpcbiAgICAgICAgICAgICAgY29uc3Qgc2NyaXB0SGFzaCA9IFNjcmlwdEhhc2guZnJvbV9ieXRlcyhrZXlCeXRlcyk7XG4gICAgICAgICAgICAgIHJldHVybiBEUmVwLm5ld19zY3JpcHRfaGFzaChzY3JpcHRIYXNoKTtcbiAgICAgICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcignSW52YWxpZCBjcmVkZW50aWFsIHR5cGUgZm9yIERSZXAnKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICB9XG4gIH1cblxuICBnZXREUmVwSWRGcm9tRFJlcChkUmVwOiBEUmVwKTogc3RyaW5nIHtcbiAgICBzd2l0Y2ggKGRSZXAua2luZCgpKSB7XG4gICAgICBjYXNlIERSZXBLaW5kLkFsd2F5c0Fic3RhaW46XG4gICAgICAgIHJldHVybiBWT1RFX0FMV0FZU19BQlNUQUlOO1xuICAgICAgY2FzZSBEUmVwS2luZC5BbHdheXNOb0NvbmZpZGVuY2U6XG4gICAgICAgIHJldHVybiBWT1RFX0FMV0FZU19OT19DT05GSURFTkNFO1xuICAgICAgZGVmYXVsdDpcbiAgICAgICAgcmV0dXJuIGRSZXAudG9fYmVjaDMyKCk7XG4gICAgfVxuICB9XG5cbiAgLyoqIEBpbmhlcml0ZG9jICovXG4gIC8vIHRoaXMgd2lsbCB2YWxpZGF0ZSBib3RoIHN0YWtlIGFuZCBwYXltZW50IGFkZHJlc3Nlc1xuICBpc1ZhbGlkQWRkcmVzcyhhZGRyZXNzOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICBjb25zdCBiZWNoMzJQcmVmaXhMaXN0ID0gWydhZGRyJywgJ2FkZHJfdGVzdCcsICdzdGFrZScsICdzdGFrZV90ZXN0J107XG4gICAgY29uc3QgQkFTRV9BRERSX0xFTiA9IDkyO1xuICAgIGNvbnN0IFJFV0FSRF9BTkRfRU5URVJQUklTRV9BRERSX0xFTiA9IDQ3O1xuICAgIGNvbnN0IFBPSU5URVJfQUREUl9MRU4gPSA1MjtcbiAgICBjb25zdCBWQUxJREFUT1JfQUREUl9MRU4gPSA1NjtcblxuICAgIC8vQ2hlY2sgZm9yIFNoZWxsZXktZXJhIChCZWNoMzIpIGFkZHJlc3Nlc1xuICAgIGlmIChuZXcgUmVnRXhwKGBeKCR7YmVjaDMyUHJlZml4TGlzdC5qb2luKCd8Jyl9KWApLnRlc3QoYWRkcmVzcykpIHtcbiAgICAgIHRyeSB7XG4gICAgICAgIGNvbnN0IGRlY29kZWRCZWNoID0gYmVjaDMyLmRlY29kZShhZGRyZXNzLCAxMDgpO1xuICAgICAgICBjb25zdCB3b3JkTGVuZ3RoID0gZGVjb2RlZEJlY2gud29yZHMubGVuZ3RoO1xuICAgICAgICBpZiAoXG4gICAgICAgICAgYmVjaDMyUHJlZml4TGlzdC5pbmNsdWRlcyhkZWNvZGVkQmVjaC5wcmVmaXgpICYmXG4gICAgICAgICAgKHdvcmRMZW5ndGggPT09IEJBU0VfQUREUl9MRU4gfHxcbiAgICAgICAgICAgIHdvcmRMZW5ndGggPT09IFJFV0FSRF9BTkRfRU5URVJQUklTRV9BRERSX0xFTiB8fFxuICAgICAgICAgICAgd29yZExlbmd0aCA9PT0gUE9JTlRFUl9BRERSX0xFTilcbiAgICAgICAgKSB7XG4gICAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgICAgIH1cbiAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgY29uc29sZS5sb2coYEFkZHJlc3M6ICR7YWRkcmVzc30gZmFpbGVkIEJlY2gzMiB0ZXN0IHdpdGggZXJyb3I6ICR7ZX1gKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvL0NoZWNrIGZvciBWYWxpZGF0b3IgYWRkcmVzc2VzXG4gICAgaWYgKG5ldyBSZWdFeHAoYF4oPyFwb29sKVthLXowLTldeyR7VkFMSURBVE9SX0FERFJfTEVOfX0kYCkudGVzdChhZGRyZXNzKSkge1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuXG4gICAgLy9DaGVjayBmb3IgQnlyb24tZXJhIGFkZHJlc3NcbiAgICB0cnkge1xuICAgICAgLy8gUmVqZWN0IERhZWRhbHVzIHdhbGxldCBhZGRyZXNzZXMgKEJ5cm9uLWVyYSBhZGRyZXNzZXMgc3RhcnRpbmcgd2l0aCBcIkRkekZGXCIpXG4gICAgICBpZiAoYWRkcmVzcy5zdGFydHNXaXRoKCdEZHpGRicpKSB7XG4gICAgICAgIGNvbnNvbGUubG9nKGBSZWplY3RpbmcgRGFlZGFsdXMgd2FsbGV0IGFkZHJlc3M6ICR7YWRkcmVzc31gKTtcbiAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBkZWNvZGVkID0gYnM1OC5kZWNvZGUoYWRkcmVzcyk7XG4gICAgICBjb25zdCBjYm9yRGF0YSA9IGNib3IuZGVjb2RlRmlyc3RTeW5jKGRlY29kZWQpO1xuICAgICAgcmV0dXJuIEFycmF5LmlzQXJyYXkoY2JvckRhdGEpICYmIGNib3JEYXRhLmxlbmd0aCA+PSAyO1xuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgIGNvbnNvbGUubG9nKGBBZGRyZXNzOiAke2FkZHJlc3N9IGZhaWxlZCBCeXJvbiB0ZXN0IHdpdGggZXJyb3I6ICR7ZX1gKTtcbiAgICAgIGNvbnNvbGUubG9nKGUuc3RhY2spO1xuICAgIH1cblxuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIC8qKiBAaW5oZXJpdGRvYyAqL1xuICBpc1ZhbGlkQmxvY2tJZChoYXNoOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy52YWxpZGF0ZUJsYWtlMmIoaGFzaCk7XG4gIH1cblxuICAvKiogQGluaGVyaXRkb2MgKi9cbiAgaXNWYWxpZFByaXZhdGVLZXkoa2V5OiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICAvLyB0aGlzIHdpbGwgcmV0dXJuIHRydWUgZm9yIGJvdGggZXh0ZW5kZWQgYW5kIG5vbi1leHRlbmRlZCBFRDI1NTE5IGtleXNcbiAgICByZXR1cm4gdGhpcy5pc1ZhbGlkS2V5KGtleSk7XG4gIH1cblxuICBpc1ZhbGlkS2V5KGtleTogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgdHJ5IHtcbiAgICAgIG5ldyBLZXlQYWlyKHsgcHJ2OiBrZXkgfSk7XG4gICAgICByZXR1cm4gdHJ1ZTtcbiAgICB9IGNhdGNoIHtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG4gIH1cblxuICAvKiogQGluaGVyaXRkb2MgKi9cbiAgaXNWYWxpZFB1YmxpY0tleShwdWJLZXk6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIHRyeSB7XG4gICAgICBuZXcgS2V5UGFpcih7IHB1YjogcHViS2V5IH0pO1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfSBjYXRjaCB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICB9XG5cbiAgLyoqIEBpbmhlcml0ZG9jICovXG4gIGlzVmFsaWRTaWduYXR1cmUoc2lnbmF0dXJlOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICB0cnkge1xuICAgICAgRWQyNTUxOVNpZ25hdHVyZS5mcm9tX2hleChzaWduYXR1cmUpO1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfSBjYXRjaCAoZXJyKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICB9XG5cbiAgLyoqIEBpbmhlcml0ZG9jICovXG4gIGlzVmFsaWRUcmFuc2FjdGlvbklkKHR4SWQ6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLnZhbGlkYXRlQmxha2UyYih0eElkKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgdGhlIHRyYW5zYWN0aW9uIGJvZHkgZnJvbSBhIHNlcmlhbGl6ZWQgdHJhbnNhY3Rpb25cbiAgICogQHBhcmFtIHtzdHJpbmd9IHNlcmlhbGl6ZWRUeCAtIHNlcmlhbGl6ZWQgdHJhbnNhY3Rpb24gaW4gaGV4IG9yIGJhc2U2NCBmb3JtYXRcbiAgICogQHJldHVybnMge3N0cmluZ30gdHJhbnNhY3Rpb24gYm9keSBpbiBoZXggZm9ybWF0XG4gICAqL1xuICBnZXRUcmFuc2FjdGlvbkJvZHkoc2VyaWFsaXplZFR4OiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIGNvbnN0IEhFWF9SRUdFWCA9IC9eWzAtOWEtZkEtRl0rJC87XG4gICAgY29uc3QgYnVmZmVyUmF3VHJhbnNhY3Rpb24gPSBIRVhfUkVHRVgudGVzdChzZXJpYWxpemVkVHgpXG4gICAgICA/IEJ1ZmZlci5mcm9tKHNlcmlhbGl6ZWRUeCwgJ2hleCcpXG4gICAgICA6IEJ1ZmZlci5mcm9tKHNlcmlhbGl6ZWRUeCwgJ2Jhc2U2NCcpO1xuICAgIHJldHVybiBCdWZmZXIuZnJvbShDYXJkYW5vVHJhbnNhY3Rpb24uZnJvbV9ieXRlcyhidWZmZXJSYXdUcmFuc2FjdGlvbikuYm9keSgpLnRvX2J5dGVzKCkpLnRvU3RyaW5nKCdoZXgnKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBEZWNvZGUgd2FsbGV0IGFkZHJlc3MgZnJvbSBzdHJpbmcuXG4gICAqIEF0dGVtcHRzIHRvIGRlY29kZSBhcyBTaGVsbGV5IChiZWNoMzIpIGZpcnN0LCB0aGVuIEJ5cm9uIChiYXNlNTgpLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gYWRkcmVzcyAtIFZhbGlkIEJ5cm9uIG9yIFNoZWxsZXktZXJhIGFkZHJlc3MuXG4gICAqIEByZXR1cm5zIHtBZGRyZXNzfSAtIFZhbGlkIGFkZHJlc3Mgb2JqZWN0LlxuICAgKiBAdGhyb3dzIHtJbnZhbGlkQWRkcmVzc0Vycm9yfSBJZiB0aGUgYWRkcmVzcyBpcyBuZWl0aGVyIHZhbGlkIFNoZWxsZXkgbm9yIEJ5cm9uLlxuICAgKi9cbiAgZ2V0V2FsbGV0QWRkcmVzcyhhZGRyZXNzOiBzdHJpbmcpOiBBZGRyZXNzIHtcbiAgICBpZiAoIWFkZHJlc3MgfHwgdHlwZW9mIGFkZHJlc3MgIT09ICdzdHJpbmcnKSB7XG4gICAgICB0aHJvdyBuZXcgSW52YWxpZEFkZHJlc3NFcnJvcignUHJvdmlkZWQgYWRkcmVzcyBpcyBub3QgYSB2YWxpZCBzdHJpbmcnKTtcbiAgICB9XG5cbiAgICAvLyBUcnkgZGVjb2RpbmcgYXMgYSBTaGVsbGV5IChiZWNoMzIpIGFkZHJlc3MgZmlyc3RcbiAgICB0cnkge1xuICAgICAgcmV0dXJuIEFkZHJlc3MuZnJvbV9iZWNoMzIoYWRkcmVzcyk7XG4gICAgfSBjYXRjaCAoZSkge1xuICAgICAgY29uc29sZS5lcnJvcihgQ291bGQgbm90IGRlY29kZSBzaGVsbHkgYWRkcmVzcyBmcm9tIHN0cmluZyAnJHthZGRyZXNzfSdgKTtcbiAgICB9XG5cbiAgICAvLyBUcnkgZGVjb2RpbmcgYXMgYSBCeXJvbiAoYmFzZTU4KSBhZGRyZXNzIGxhdGVyXG4gICAgdHJ5IHtcbiAgICAgIC8vIFJlamVjdCBEYWVkYWx1cyB3YWxsZXQgYWRkcmVzc2VzIChCeXJvbi1lcmEgYWRkcmVzc2VzIHN0YXJ0aW5nIHdpdGggXCJEZHpGRlwiKVxuICAgICAgaWYgKGFkZHJlc3Muc3RhcnRzV2l0aCgnRGR6RkYnKSkge1xuICAgICAgICB0aHJvdyBuZXcgSW52YWxpZEFkZHJlc3NFcnJvcignUHJvdmlkZWQgc3RyaW5nIGlzIGEgRGFlZGFsdXMgYWRkcmVzcycpO1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gQnlyb25BZGRyZXNzLmZyb21fYmFzZTU4KGFkZHJlc3MpLnRvX2FkZHJlc3MoKTtcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICBjb25zb2xlLmVycm9yKGBDb3VsZCBub3QgZGVjb2RlIGJ5cm9uIGFkZHJlc3MgZnJvbSBzdHJpbmcgJyR7YWRkcmVzc30nYCk7XG4gICAgfVxuICAgIHRocm93IG5ldyBJbnZhbGlkQWRkcmVzc0Vycm9yKCdQcm92aWRlZCBzdHJpbmcgaXMgbm90IGEgdmFsaWQgU2hlbGxleSBvciBCeXJvbiBhZGRyZXNzJyk7XG4gIH1cblxuICAvKipcbiAgICogRGVjb2RlIGFkZHJlc3Mgc3RyaW5nIGZyb20gQWRkcmVzcyBvYmplY3QuXG4gICAqIEF0dGVtcHRzIHRvIGRlY29kZSBhcyBTaGVsbGV5IChiZWNoMzIpIGZpcnN0LCB0aGVuIEJ5cm9uIChiYXNlNTgpLlxuICAgKiBAcGFyYW0ge0FkZHJlc3N9IGFkZHJlc3MgLSBWYWxpZCBBZGRyZXNzIG9iamVjdFxuICAgKiBAcmV0dXJucyB7c3RyaW5nfSAtIFZhbGlkIEJ5cm9uIG9yIFNoZWxsZXktZXJhIGFkZHJlc3Mgc3RyaW5nLlxuICAgKiBAdGhyb3dzIHtJbnZhbGlkQWRkcmVzc0Vycm9yfSBJZiB0aGUgQWRkcmVzcyBvYmplY3QgaXMgbmVpdGhlciB2YWxpZCBTaGVsbGV5IG5vciBCeXJvbi5cbiAgICovXG4gIGdldEFkZHJlc3NTdHJpbmcoYWRkcmVzczogQWRkcmVzcyk6IHN0cmluZyB7XG4gICAgLy8gQ2hlY2sgYWxsIFNoZWxsZXkgYWRkcmVzcyB0eXBlc1xuICAgIGlmIChcbiAgICAgIEJhc2VBZGRyZXNzLmZyb21fYWRkcmVzcyhhZGRyZXNzKSB8fFxuICAgICAgRW50ZXJwcmlzZUFkZHJlc3MuZnJvbV9hZGRyZXNzKGFkZHJlc3MpIHx8XG4gICAgICBSZXdhcmRBZGRyZXNzLmZyb21fYWRkcmVzcyhhZGRyZXNzKSB8fFxuICAgICAgUG9pbnRlckFkZHJlc3MuZnJvbV9hZGRyZXNzKGFkZHJlc3MpXG4gICAgKSB7XG4gICAgICByZXR1cm4gYWRkcmVzcy50b19iZWNoMzIoKTtcbiAgICB9XG5cbiAgICBjb25zdCBieXJvbkFkZHJlc3MgPSBCeXJvbkFkZHJlc3MuZnJvbV9hZGRyZXNzKGFkZHJlc3MpO1xuICAgIC8vIFJlamVjdCBEYWVkYWx1cyB3YWxsZXQgYWRkcmVzc2VzIChCeXJvbi1lcmEgYWRkcmVzc2VzIHN0YXJ0aW5nIHdpdGggXCJEZHpGRlwiKVxuICAgIGlmIChieXJvbkFkZHJlc3MpIHtcbiAgICAgIGlmIChieXJvbkFkZHJlc3MudG9fYmFzZTU4KCkuc3RhcnRzV2l0aCgnRGR6RkYnKSkge1xuICAgICAgICB0aHJvdyBuZXcgSW52YWxpZEFkZHJlc3NFcnJvcignUHJvdmlkZWQgYWRkcmVzcyBpcyBhIERhZWRhbHVzIGFkZHJlc3MnKTtcbiAgICAgIH1cbiAgICAgIHJldHVybiBieXJvbkFkZHJlc3MudG9fYmFzZTU4KCk7XG4gICAgfVxuXG4gICAgLy8gSWYgbmVpdGhlciwgaXQncyBpbnZhbGlkXG4gICAgdGhyb3cgbmV3IEludmFsaWRBZGRyZXNzRXJyb3IoJ1Byb3ZpZGVkIEFkZHJlc3MgaXMgbm90IGEgdmFsaWQgU2hlbGxleSBvciBCeXJvbiBhZGRyZXNzJyk7XG4gIH1cbn1cblxuY29uc3QgdXRpbHMgPSBuZXcgVXRpbHMoKTtcblxuZXhwb3J0IGRlZmF1bHQgdXRpbHM7XG4iXX0=Выполнить команду
Для локальной разработки. Не используйте в интернете!