PHP WebShell

Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/utxo-core/dist/src/paygo/psbt

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

"use strict";
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;
    };
})();
Object.defineProperty(exports, "__esModule", { value: true });
exports.addPayGoAddressProof = addPayGoAddressProof;
exports.verifyPayGoAddressProof = verifyPayGoAddressProof;
exports.getPayGoAddressProofOutputIndex = getPayGoAddressProofOutputIndex;
exports.psbtOutputIncludesPaygoAddressProof = psbtOutputIncludesPaygoAddressProof;
const utxolib = __importStar(require("@bitgo/utxo-lib"));
const utils_1 = require("bip174/src/lib/utils");
const bip32utils_1 = require("../../bip32utils");
const attestation_1 = require("../attestation");
const Errors_1 = require("./Errors");
/** This function adds the entropy and signature into the PSBT output unknown key vals.
 * We store the entropy so that we reconstruct the message <ENTROPY><ADDRESS><UUID>
 * to later verify.
 *
 * @param psbt - PSBT that we need to encode our paygo address into
 * @param outputIndex - the index of the address in our output
 * @param sig - the signature that we want to encode
 * @param entropy - the arbitrary entropy bytes from our vasp proof
 */
function addPayGoAddressProof(psbt, outputIndex, sig, entropy) {
    utxolib.bitgo.addProprietaryKeyValuesFromUnknownKeyValues(psbt, 'output', outputIndex, {
        key: {
            identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,
            subtype: utxolib.bitgo.ProprietaryKeySubtype.PAYGO_ADDRESS_ATTESTATION_PROOF,
            keydata: entropy,
        },
        value: sig,
    });
}
/** Verify the paygo address signature is valid using verification pub key.
 *
 * @param psbt - PSBT we want to verify that the paygo address is in
 * @param outputIndex - we have the output index that address is in
 * @param verificationPubkey - the pubkey signed by the HSM to verify our message
 * @returns
 */
function verifyPayGoAddressProof(psbt, outputIndex, verificationPubkey) {
    const psbtOutputs = (0, utils_1.checkForOutput)(psbt.data.outputs, outputIndex);
    const stored = utxolib.bitgo.getProprietaryKeyValuesFromUnknownKeyValues(psbtOutputs, {
        identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,
        subtype: utxolib.bitgo.ProprietaryKeySubtype.PAYGO_ADDRESS_ATTESTATION_PROOF,
    });
    // assert stored length is 0 or 1
    if (stored.length === 0) {
        throw new Errors_1.ErrorNoPayGoProof(outputIndex);
    }
    else if (stored.length > 1) {
        throw new Errors_1.ErrorMultiplePayGoProof();
    }
    // We get the signature and entropy from our PSBT unknown key vals
    const signature = stored[0].value;
    const entropy = stored[0].key.keydata;
    // Get the the PayGo address from the txOutputs
    const txOutputs = psbt.txOutputs;
    if (outputIndex >= txOutputs.length) {
        throw new Errors_1.ErrorOutputIndexOutOfBounds(outputIndex);
    }
    const output = txOutputs[outputIndex];
    const addressFromOutput = utxolib.address.fromOutputScript(output.script, psbt.network);
    // We construct our message <ENTROPY><ADDRESS><UUID>
    const message = (0, attestation_1.createPayGoAttestationBuffer)(addressFromOutput, entropy, psbt.network);
    // bip32utils.verifyMessage now takes in message as a Buffer
    if (!(0, bip32utils_1.verifyMessage)(message, verificationPubkey, signature, utxolib.networks.bitcoin)) {
        throw new Errors_1.ErrorPayGoAddressProofFailedVerification();
    }
}
/** Get the output index of the paygo output if there is one. It does this by
 * checking if the metadata is on one of the outputs of the PSBT. If there is
 * no paygo output, return undefined
 *
 * @param psbt
 * @returns number - the index of the output address
 */
function getPayGoAddressProofOutputIndex(psbt) {
    const res = psbt.data.outputs.flatMap((output, outputIndex) => {
        const proprietaryKeyVals = utxolib.bitgo.getPsbtOutputProprietaryKeyVals(output, {
            identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,
            subtype: utxolib.bitgo.ProprietaryKeySubtype.PAYGO_ADDRESS_ATTESTATION_PROOF,
        });
        if (proprietaryKeyVals.length > 1) {
            throw new Errors_1.ErrorMultiplePayGoProofAtPsbtIndex(outputIndex);
        }
        return proprietaryKeyVals.length === 0 ? [] : [outputIndex];
    });
    return res.length === 0 ? undefined : res[0];
}
function psbtOutputIncludesPaygoAddressProof(psbt) {
    return getPayGoAddressProofOutputIndex(psbt) !== undefined;
}
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGF5R29BZGRyZXNzUHJvb2YuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvcGF5Z28vcHNidC9wYXlHb0FkZHJlc3NQcm9vZi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQXVCQSxvREFjQztBQVNELDBEQXFDQztBQVNELDBFQWVDO0FBRUQsa0ZBRUM7QUEvR0QseURBQTJDO0FBQzNDLGdEQUFzRDtBQUV0RCxpREFBaUQ7QUFDakQsZ0RBQThEO0FBRTlELHFDQU1rQjtBQUVsQjs7Ozs7Ozs7R0FRRztBQUNILFNBQWdCLG9CQUFvQixDQUNsQyxJQUE0QixFQUM1QixXQUFtQixFQUNuQixHQUFXLEVBQ1gsT0FBZTtJQUVmLE9BQU8sQ0FBQyxLQUFLLENBQUMsMkNBQTJDLENBQUMsSUFBSSxFQUFFLFFBQVEsRUFBRSxXQUFXLEVBQUU7UUFDckYsR0FBRyxFQUFFO1lBQ0gsVUFBVSxFQUFFLE9BQU8sQ0FBQyxLQUFLLENBQUMsMkJBQTJCO1lBQ3JELE9BQU8sRUFBRSxPQUFPLENBQUMsS0FBSyxDQUFDLHFCQUFxQixDQUFDLCtCQUErQjtZQUM1RSxPQUFPLEVBQUUsT0FBTztTQUNqQjtRQUNELEtBQUssRUFBRSxHQUFHO0tBQ1gsQ0FBQyxDQUFDO0FBQ0wsQ0FBQztBQUVEOzs7Ozs7R0FNRztBQUNILFNBQWdCLHVCQUF1QixDQUNyQyxJQUE0QixFQUM1QixXQUFtQixFQUNuQixrQkFBMEI7SUFFMUIsTUFBTSxXQUFXLEdBQUcsSUFBQSxzQkFBYyxFQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLFdBQVcsQ0FBQyxDQUFDO0lBQ25FLE1BQU0sTUFBTSxHQUFHLE9BQU8sQ0FBQyxLQUFLLENBQUMsMkNBQTJDLENBQUMsV0FBVyxFQUFFO1FBQ3BGLFVBQVUsRUFBRSxPQUFPLENBQUMsS0FBSyxDQUFDLDJCQUEyQjtRQUNyRCxPQUFPLEVBQUUsT0FBTyxDQUFDLEtBQUssQ0FBQyxxQkFBcUIsQ0FBQywrQkFBK0I7S0FDN0UsQ0FBQyxDQUFDO0lBRUgsaUNBQWlDO0lBQ2pDLElBQUksTUFBTSxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUUsQ0FBQztRQUN4QixNQUFNLElBQUksMEJBQWlCLENBQUMsV0FBVyxDQUFDLENBQUM7SUFDM0MsQ0FBQztTQUFNLElBQUksTUFBTSxDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQztRQUM3QixNQUFNLElBQUksZ0NBQXVCLEVBQUUsQ0FBQztJQUN0QyxDQUFDO0lBRUQsa0VBQWtFO0lBQ2xFLE1BQU0sU0FBUyxHQUFHLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUM7SUFDbEMsTUFBTSxPQUFPLEdBQUcsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUM7SUFFdEMsK0NBQStDO0lBQy9DLE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUM7SUFDakMsSUFBSSxXQUFXLElBQUksU0FBUyxDQUFDLE1BQU0sRUFBRSxDQUFDO1FBQ3BDLE1BQU0sSUFBSSxvQ0FBMkIsQ0FBQyxXQUFXLENBQUMsQ0FBQztJQUNyRCxDQUFDO0lBQ0QsTUFBTSxNQUFNLEdBQUcsU0FBUyxDQUFDLFdBQVcsQ0FBQyxDQUFDO0lBQ3RDLE1BQU0saUJBQWlCLEdBQUcsT0FBTyxDQUFDLE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNLENBQUMsTUFBTSxFQUFFLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUV4RixvREFBb0Q7SUFDcEQsTUFBTSxPQUFPLEdBQUcsSUFBQSwwQ0FBNEIsRUFBQyxpQkFBaUIsRUFBRSxPQUFPLEVBQUUsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBRXZGLDREQUE0RDtJQUM1RCxJQUFJLENBQUMsSUFBQSwwQkFBYSxFQUFDLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxTQUFTLEVBQUUsT0FBTyxDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1FBQ3JGLE1BQU0sSUFBSSxpREFBd0MsRUFBRSxDQUFDO0lBQ3ZELENBQUM7QUFDSCxDQUFDO0FBRUQ7Ozs7OztHQU1HO0FBQ0gsU0FBZ0IsK0JBQStCLENBQUMsSUFBNEI7SUFDMUUsTUFBTSxHQUFHLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDLENBQUMsTUFBTSxFQUFFLFdBQVcsRUFBRSxFQUFFO1FBQzVELE1BQU0sa0JBQWtCLEdBQUcsT0FBTyxDQUFDLEtBQUssQ0FBQywrQkFBK0IsQ0FBQyxNQUFNLEVBQUU7WUFDL0UsVUFBVSxFQUFFLE9BQU8sQ0FBQyxLQUFLLENBQUMsMkJBQTJCO1lBQ3JELE9BQU8sRUFBRSxPQUFPLENBQUMsS0FBSyxDQUFDLHFCQUFxQixDQUFDLCtCQUErQjtTQUM3RSxDQUFDLENBQUM7UUFFSCxJQUFJLGtCQUFrQixDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUNsQyxNQUFNLElBQUksMkNBQWtDLENBQUMsV0FBVyxDQUFDLENBQUM7UUFDNUQsQ0FBQztRQUVELE9BQU8sa0JBQWtCLENBQUMsTUFBTSxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxDQUFDO0lBQzlELENBQUMsQ0FBQyxDQUFDO0lBRUgsT0FBTyxHQUFHLENBQUMsTUFBTSxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7QUFDL0MsQ0FBQztBQUVELFNBQWdCLG1DQUFtQyxDQUFDLElBQTRCO0lBQzlFLE9BQU8sK0JBQStCLENBQUMsSUFBSSxDQUFDLEtBQUssU0FBUyxDQUFDO0FBQzdELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyB1dHhvbGliIGZyb20gJ0BiaXRnby91dHhvLWxpYic7XG5pbXBvcnQgeyBjaGVja0Zvck91dHB1dCB9IGZyb20gJ2JpcDE3NC9zcmMvbGliL3V0aWxzJztcblxuaW1wb3J0IHsgdmVyaWZ5TWVzc2FnZSB9IGZyb20gJy4uLy4uL2JpcDMydXRpbHMnO1xuaW1wb3J0IHsgY3JlYXRlUGF5R29BdHRlc3RhdGlvbkJ1ZmZlciB9IGZyb20gJy4uL2F0dGVzdGF0aW9uJztcblxuaW1wb3J0IHtcbiAgRXJyb3JNdWx0aXBsZVBheUdvUHJvb2YsXG4gIEVycm9yTXVsdGlwbGVQYXlHb1Byb29mQXRQc2J0SW5kZXgsXG4gIEVycm9yTm9QYXlHb1Byb29mLFxuICBFcnJvck91dHB1dEluZGV4T3V0T2ZCb3VuZHMsXG4gIEVycm9yUGF5R29BZGRyZXNzUHJvb2ZGYWlsZWRWZXJpZmljYXRpb24sXG59IGZyb20gJy4vRXJyb3JzJztcblxuLyoqIFRoaXMgZnVuY3Rpb24gYWRkcyB0aGUgZW50cm9weSBhbmQgc2lnbmF0dXJlIGludG8gdGhlIFBTQlQgb3V0cHV0IHVua25vd24ga2V5IHZhbHMuXG4gKiBXZSBzdG9yZSB0aGUgZW50cm9weSBzbyB0aGF0IHdlIHJlY29uc3RydWN0IHRoZSBtZXNzYWdlIDxFTlRST1BZPjxBRERSRVNTPjxVVUlEPlxuICogdG8gbGF0ZXIgdmVyaWZ5LlxuICpcbiAqIEBwYXJhbSBwc2J0IC0gUFNCVCB0aGF0IHdlIG5lZWQgdG8gZW5jb2RlIG91ciBwYXlnbyBhZGRyZXNzIGludG9cbiAqIEBwYXJhbSBvdXRwdXRJbmRleCAtIHRoZSBpbmRleCBvZiB0aGUgYWRkcmVzcyBpbiBvdXIgb3V0cHV0XG4gKiBAcGFyYW0gc2lnIC0gdGhlIHNpZ25hdHVyZSB0aGF0IHdlIHdhbnQgdG8gZW5jb2RlXG4gKiBAcGFyYW0gZW50cm9weSAtIHRoZSBhcmJpdHJhcnkgZW50cm9weSBieXRlcyBmcm9tIG91ciB2YXNwIHByb29mXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBhZGRQYXlHb0FkZHJlc3NQcm9vZihcbiAgcHNidDogdXR4b2xpYi5iaXRnby5VdHhvUHNidCxcbiAgb3V0cHV0SW5kZXg6IG51bWJlcixcbiAgc2lnOiBCdWZmZXIsXG4gIGVudHJvcHk6IEJ1ZmZlclxuKTogdm9pZCB7XG4gIHV0eG9saWIuYml0Z28uYWRkUHJvcHJpZXRhcnlLZXlWYWx1ZXNGcm9tVW5rbm93bktleVZhbHVlcyhwc2J0LCAnb3V0cHV0Jywgb3V0cHV0SW5kZXgsIHtcbiAgICBrZXk6IHtcbiAgICAgIGlkZW50aWZpZXI6IHV0eG9saWIuYml0Z28uUFNCVF9QUk9QUklFVEFSWV9JREVOVElGSUVSLFxuICAgICAgc3VidHlwZTogdXR4b2xpYi5iaXRnby5Qcm9wcmlldGFyeUtleVN1YnR5cGUuUEFZR09fQUREUkVTU19BVFRFU1RBVElPTl9QUk9PRixcbiAgICAgIGtleWRhdGE6IGVudHJvcHksXG4gICAgfSxcbiAgICB2YWx1ZTogc2lnLFxuICB9KTtcbn1cblxuLyoqIFZlcmlmeSB0aGUgcGF5Z28gYWRkcmVzcyBzaWduYXR1cmUgaXMgdmFsaWQgdXNpbmcgdmVyaWZpY2F0aW9uIHB1YiBrZXkuXG4gKlxuICogQHBhcmFtIHBzYnQgLSBQU0JUIHdlIHdhbnQgdG8gdmVyaWZ5IHRoYXQgdGhlIHBheWdvIGFkZHJlc3MgaXMgaW5cbiAqIEBwYXJhbSBvdXRwdXRJbmRleCAtIHdlIGhhdmUgdGhlIG91dHB1dCBpbmRleCB0aGF0IGFkZHJlc3MgaXMgaW5cbiAqIEBwYXJhbSB2ZXJpZmljYXRpb25QdWJrZXkgLSB0aGUgcHVia2V5IHNpZ25lZCBieSB0aGUgSFNNIHRvIHZlcmlmeSBvdXIgbWVzc2FnZVxuICogQHJldHVybnNcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHZlcmlmeVBheUdvQWRkcmVzc1Byb29mKFxuICBwc2J0OiB1dHhvbGliLmJpdGdvLlV0eG9Qc2J0LFxuICBvdXRwdXRJbmRleDogbnVtYmVyLFxuICB2ZXJpZmljYXRpb25QdWJrZXk6IEJ1ZmZlclxuKTogdm9pZCB7XG4gIGNvbnN0IHBzYnRPdXRwdXRzID0gY2hlY2tGb3JPdXRwdXQocHNidC5kYXRhLm91dHB1dHMsIG91dHB1dEluZGV4KTtcbiAgY29uc3Qgc3RvcmVkID0gdXR4b2xpYi5iaXRnby5nZXRQcm9wcmlldGFyeUtleVZhbHVlc0Zyb21Vbmtub3duS2V5VmFsdWVzKHBzYnRPdXRwdXRzLCB7XG4gICAgaWRlbnRpZmllcjogdXR4b2xpYi5iaXRnby5QU0JUX1BST1BSSUVUQVJZX0lERU5USUZJRVIsXG4gICAgc3VidHlwZTogdXR4b2xpYi5iaXRnby5Qcm9wcmlldGFyeUtleVN1YnR5cGUuUEFZR09fQUREUkVTU19BVFRFU1RBVElPTl9QUk9PRixcbiAgfSk7XG5cbiAgLy8gYXNzZXJ0IHN0b3JlZCBsZW5ndGggaXMgMCBvciAxXG4gIGlmIChzdG9yZWQubGVuZ3RoID09PSAwKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yTm9QYXlHb1Byb29mKG91dHB1dEluZGV4KTtcbiAgfSBlbHNlIGlmIChzdG9yZWQubGVuZ3RoID4gMSkge1xuICAgIHRocm93IG5ldyBFcnJvck11bHRpcGxlUGF5R29Qcm9vZigpO1xuICB9XG5cbiAgLy8gV2UgZ2V0IHRoZSBzaWduYXR1cmUgYW5kIGVudHJvcHkgZnJvbSBvdXIgUFNCVCB1bmtub3duIGtleSB2YWxzXG4gIGNvbnN0IHNpZ25hdHVyZSA9IHN0b3JlZFswXS52YWx1ZTtcbiAgY29uc3QgZW50cm9weSA9IHN0b3JlZFswXS5rZXkua2V5ZGF0YTtcblxuICAvLyBHZXQgdGhlIHRoZSBQYXlHbyBhZGRyZXNzIGZyb20gdGhlIHR4T3V0cHV0c1xuICBjb25zdCB0eE91dHB1dHMgPSBwc2J0LnR4T3V0cHV0cztcbiAgaWYgKG91dHB1dEluZGV4ID49IHR4T3V0cHV0cy5sZW5ndGgpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3JPdXRwdXRJbmRleE91dE9mQm91bmRzKG91dHB1dEluZGV4KTtcbiAgfVxuICBjb25zdCBvdXRwdXQgPSB0eE91dHB1dHNbb3V0cHV0SW5kZXhdO1xuICBjb25zdCBhZGRyZXNzRnJvbU91dHB1dCA9IHV0eG9saWIuYWRkcmVzcy5mcm9tT3V0cHV0U2NyaXB0KG91dHB1dC5zY3JpcHQsIHBzYnQubmV0d29yayk7XG5cbiAgLy8gV2UgY29uc3RydWN0IG91ciBtZXNzYWdlIDxFTlRST1BZPjxBRERSRVNTPjxVVUlEPlxuICBjb25zdCBtZXNzYWdlID0gY3JlYXRlUGF5R29BdHRlc3RhdGlvbkJ1ZmZlcihhZGRyZXNzRnJvbU91dHB1dCwgZW50cm9weSwgcHNidC5uZXR3b3JrKTtcblxuICAvLyBiaXAzMnV0aWxzLnZlcmlmeU1lc3NhZ2Ugbm93IHRha2VzIGluIG1lc3NhZ2UgYXMgYSBCdWZmZXJcbiAgaWYgKCF2ZXJpZnlNZXNzYWdlKG1lc3NhZ2UsIHZlcmlmaWNhdGlvblB1YmtleSwgc2lnbmF0dXJlLCB1dHhvbGliLm5ldHdvcmtzLmJpdGNvaW4pKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yUGF5R29BZGRyZXNzUHJvb2ZGYWlsZWRWZXJpZmljYXRpb24oKTtcbiAgfVxufVxuXG4vKiogR2V0IHRoZSBvdXRwdXQgaW5kZXggb2YgdGhlIHBheWdvIG91dHB1dCBpZiB0aGVyZSBpcyBvbmUuIEl0IGRvZXMgdGhpcyBieVxuICogY2hlY2tpbmcgaWYgdGhlIG1ldGFkYXRhIGlzIG9uIG9uZSBvZiB0aGUgb3V0cHV0cyBvZiB0aGUgUFNCVC4gSWYgdGhlcmUgaXNcbiAqIG5vIHBheWdvIG91dHB1dCwgcmV0dXJuIHVuZGVmaW5lZFxuICpcbiAqIEBwYXJhbSBwc2J0XG4gKiBAcmV0dXJucyBudW1iZXIgLSB0aGUgaW5kZXggb2YgdGhlIG91dHB1dCBhZGRyZXNzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRQYXlHb0FkZHJlc3NQcm9vZk91dHB1dEluZGV4KHBzYnQ6IHV0eG9saWIuYml0Z28uVXR4b1BzYnQpOiBudW1iZXIgfCB1bmRlZmluZWQge1xuICBjb25zdCByZXMgPSBwc2J0LmRhdGEub3V0cHV0cy5mbGF0TWFwKChvdXRwdXQsIG91dHB1dEluZGV4KSA9PiB7XG4gICAgY29uc3QgcHJvcHJpZXRhcnlLZXlWYWxzID0gdXR4b2xpYi5iaXRnby5nZXRQc2J0T3V0cHV0UHJvcHJpZXRhcnlLZXlWYWxzKG91dHB1dCwge1xuICAgICAgaWRlbnRpZmllcjogdXR4b2xpYi5iaXRnby5QU0JUX1BST1BSSUVUQVJZX0lERU5USUZJRVIsXG4gICAgICBzdWJ0eXBlOiB1dHhvbGliLmJpdGdvLlByb3ByaWV0YXJ5S2V5U3VidHlwZS5QQVlHT19BRERSRVNTX0FUVEVTVEFUSU9OX1BST09GLFxuICAgIH0pO1xuXG4gICAgaWYgKHByb3ByaWV0YXJ5S2V5VmFscy5sZW5ndGggPiAxKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3JNdWx0aXBsZVBheUdvUHJvb2ZBdFBzYnRJbmRleChvdXRwdXRJbmRleCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHByb3ByaWV0YXJ5S2V5VmFscy5sZW5ndGggPT09IDAgPyBbXSA6IFtvdXRwdXRJbmRleF07XG4gIH0pO1xuXG4gIHJldHVybiByZXMubGVuZ3RoID09PSAwID8gdW5kZWZpbmVkIDogcmVzWzBdO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcHNidE91dHB1dEluY2x1ZGVzUGF5Z29BZGRyZXNzUHJvb2YocHNidDogdXR4b2xpYi5iaXRnby5VdHhvUHNidCk6IGJvb2xlYW4ge1xuICByZXR1cm4gZ2V0UGF5R29BZGRyZXNzUHJvb2ZPdXRwdXRJbmRleChwc2J0KSAhPT0gdW5kZWZpbmVkO1xufVxuIl19

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


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