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=

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


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