@bitgo/utxo-lib
Version:
Client-side Bitcoin JavaScript library
248 lines • 44 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.AcidTest = exports.txFormats = exports.outputScriptTypes = exports.inputScriptTypes = exports.signStages = void 0;
exports.toUnspent = toUnspent;
exports.getSigners = getSigners;
exports.signPsbtInput = signPsbtInput;
exports.signAllPsbtInputs = signAllPsbtInputs;
exports.constructPsbt = constructPsbt;
exports.verifyFullySignedSignatures = verifyFullySignedSignatures;
const bitcoinjslib = require("bitcoinjs-lib");
const assert_1 = require("assert");
const outputScripts_1 = require("../bitgo/outputScripts");
const bitgo_1 = require("../bitgo");
const networks_1 = require("../networks");
const mock_1 = require("./mock");
const address_1 = require("../address");
const keys_1 = require("./keys");
exports.signStages = ['unsigned', 'halfsigned', 'fullsigned'];
/**
* array of supported input script types.
* use p2trMusig2 for p2trMusig2 script path.
* use taprootKeyPathSpend for p2trMusig2 key path.
*/
exports.inputScriptTypes = [...outputScripts_1.scriptTypes2Of3, 'taprootKeyPathSpend', outputScripts_1.scriptTypeP2shP2pk];
/**
* array of supported output script types.
*/
exports.outputScriptTypes = outputScripts_1.scriptTypes2Of3;
/**
* create unspent object from input script type, index, network and root wallet key.
*/
function toUnspent(input, index, network, rootWalletKeys, { p2shP2pkKey } = {}) {
if (input.scriptType === 'p2shP2pk') {
return (0, mock_1.mockReplayProtectionUnspent)(network, input.value, {
key: p2shP2pkKey ?? rootWalletKeys['user'],
vout: index,
});
}
else {
const chain = (0, bitgo_1.getInternalChainCode)(input.scriptType === 'taprootKeyPathSpend' ? 'p2trMusig2' : input.scriptType);
return (0, mock_1.mockWalletUnspent)(network, input.value, {
chain,
vout: index,
keys: rootWalletKeys,
index,
});
}
}
/**
* returns signer and cosigner names for InputScriptType.
* user and undefined as signer and cosigner respectively for p2shP2pk.
* user and backup as signer and cosigner respectively for p2trMusig2.
* user and bitgo as signer and cosigner respectively for other input script types.
*/
function getSigners(inputType) {
return {
signerName: 'user',
cosignerName: inputType === 'p2shP2pk' ? undefined : inputType === 'p2trMusig2' ? 'backup' : 'bitgo',
};
}
/**
* signs with first or second signature for single input.
* p2shP2pk is signed only with first sign.
*/
function signPsbtInput(psbt, input, inputIndex, rootWalletKeys, sign, params) {
const { signers, deterministic } = params ?? {};
const { signerName, cosignerName } = signers ? signers : getSigners(input.scriptType);
if (sign === 'halfsigned') {
if (input.scriptType === 'p2shP2pk') {
psbt.signInput(inputIndex, rootWalletKeys[signerName]);
}
else {
psbt.signInputHD(inputIndex, rootWalletKeys[signerName]);
}
}
if (sign === 'fullsigned' && cosignerName && input.scriptType !== 'p2shP2pk') {
psbt.signInputHD(inputIndex, rootWalletKeys[cosignerName], { deterministic });
}
}
/**
* signs with first or second signature for all inputs.
* p2shP2pk is signed only with first sign.
*/
function signAllPsbtInputs(psbt, inputs, rootWalletKeys, sign, params) {
const { signers, deterministic } = params ?? {};
inputs.forEach((input, inputIndex) => {
signPsbtInput(psbt, input, inputIndex, rootWalletKeys, sign, { signers, deterministic });
});
}
/**
* construct psbt for given inputs, outputs, network and root wallet keys.
*/
function constructPsbt(inputs, outputs, network, rootWalletKeys, signStage, params) {
const { signers, deterministic, skipNonWitnessUtxo } = params ?? {};
const totalInputAmount = inputs.reduce((sum, input) => sum + input.value, BigInt(0));
const outputInputAmount = outputs.reduce((sum, output) => sum + output.value, BigInt(0));
(0, assert_1.ok)(totalInputAmount >= outputInputAmount, 'total output can not exceed total input');
const psbt = (0, bitgo_1.createPsbtForNetwork)({ network });
if (params?.addGlobalXPubs) {
(0, bitgo_1.addXpubsToPsbt)(psbt, rootWalletKeys);
}
const unspents = inputs.map((input, i) => toUnspent(input, i, network, rootWalletKeys, { p2shP2pkKey: params?.p2shP2pkKey }));
unspents.forEach((u, i) => {
const { signerName, cosignerName } = signers ? signers : getSigners(inputs[i].scriptType);
if ((0, bitgo_1.isWalletUnspent)(u) && cosignerName) {
(0, bitgo_1.addWalletUnspentToPsbt)(psbt, u, rootWalletKeys, signerName, cosignerName, { skipNonWitnessUtxo });
}
else {
const { redeemScript } = (0, outputScripts_1.createOutputScriptP2shP2pk)(params?.p2shP2pkKey ?? rootWalletKeys.user.publicKey);
(0, assert_1.ok)(redeemScript);
(0, bitgo_1.addReplayProtectionUnspentToPsbt)(psbt, u, redeemScript, { skipNonWitnessUtxo });
}
});
outputs.forEach((output, i) => {
if ('scriptType' in output) {
(0, bitgo_1.addWalletOutputToPsbt)(psbt, output.walletKeys ?? rootWalletKeys, output.isInternalAddress ? (0, bitgo_1.getInternalChainCode)(output.scriptType) : (0, bitgo_1.getExternalChainCode)(output.scriptType), i, output.value, { addDerivationInfo: output.walletKeys !== null });
return;
}
else if ('address' in output) {
const { address, value } = output;
psbt.addOutput({ script: (0, address_1.toOutputScript)(address, network), value });
return;
}
else if ('opReturn' in output) {
const { opReturn, value } = output;
const script = bitcoinjslib.payments.embed({ data: [Buffer.from(opReturn, 'ascii')] }).output;
(0, assert_1.ok)(script, 'script is required');
psbt.addOutput({ script, value });
return;
}
else if ('script' in output) {
const { script, value } = output;
psbt.addOutput({ script: Buffer.from(script, 'hex'), value });
return;
}
throw new Error('invalid output');
});
if (signStage === 'unsigned') {
return psbt;
}
/* use fixed sessionId for deterministic nonce creation */
const sessionId = Buffer.alloc(32);
psbt.setAllInputsMusig2NonceHD(rootWalletKeys['user'], { sessionId });
psbt.setAllInputsMusig2NonceHD(rootWalletKeys['bitgo'], { deterministic });
signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'halfsigned', { signers, skipNonWitnessUtxo });
if (signStage === 'fullsigned') {
signAllPsbtInputs(psbt, inputs, rootWalletKeys, signStage, { signers, deterministic, skipNonWitnessUtxo });
}
return psbt;
}
exports.txFormats = ['psbt', 'psbt-lite'];
/**
* Creates a valid PSBT with as many features as possible.
*
* - Inputs:
* - All wallet script types that are supported by the network.
* - A p2shP2pk input (for replay protection)
* - Outputs:
* - All wallet script types that are supported by the network.
* - A p2sh output with derivation info of a different wallet (not in the global psbt xpubs)
* - A p2sh output with no derivation info (external output)
* - An OP_RETURN output
*/
class AcidTest {
constructor(network, signStage, txFormat, rootWalletKeys, otherWalletKeys, inputs, outputs) {
this.network = network;
this.signStage = signStage;
this.txFormat = txFormat;
this.rootWalletKeys = rootWalletKeys;
this.otherWalletKeys = otherWalletKeys;
this.inputs = inputs;
this.outputs = outputs;
}
static withConfig(network, signStage, txFormat, suiteConfig) {
const rootWalletKeys = (0, keys_1.getDefaultWalletKeys)();
const otherWalletKeys = (0, keys_1.getWalletKeysForSeed)('too many secrets');
const inputs = exports.inputScriptTypes
.filter((scriptType) => scriptType === 'taprootKeyPathSpend'
? (0, outputScripts_1.isSupportedScriptType)(network, 'p2trMusig2')
: (0, outputScripts_1.isSupportedScriptType)(network, scriptType))
.filter((scriptType) => (suiteConfig.includeP2trMusig2ScriptPath ?? true) || scriptType !== 'p2trMusig2')
.map((scriptType) => ({ scriptType, value: BigInt(2000) }));
const outputs = exports.outputScriptTypes
.filter((scriptType) => (0, outputScripts_1.isSupportedScriptType)(network, scriptType))
.map((scriptType) => ({ scriptType, value: BigInt(900) }));
// Test other wallet output (with derivation info)
outputs.push({ scriptType: 'p2sh', value: BigInt(900), walletKeys: otherWalletKeys });
// Tes non-wallet output
outputs.push({ scriptType: 'p2sh', value: BigInt(900), walletKeys: null });
// Test OP_RETURN output
outputs.push({ opReturn: 'setec astronomy', value: BigInt(0) });
return new AcidTest(network, signStage, txFormat, rootWalletKeys, otherWalletKeys, inputs, outputs);
}
get name() {
const networkName = (0, networks_1.getNetworkName)(this.network);
return `${networkName} ${this.signStage} ${this.txFormat}`;
}
getReplayProtectionPublicKey() {
return this.rootWalletKeys.user.publicKey;
}
getReplayProtectionOutputScript() {
const { scriptPubKey } = (0, outputScripts_1.createOutputScriptP2shP2pk)(this.getReplayProtectionPublicKey());
(0, assert_1.ok)(scriptPubKey);
return scriptPubKey;
}
createPsbt() {
const psbt = constructPsbt(this.inputs, this.outputs, this.network, this.rootWalletKeys, this.signStage, {
deterministic: true,
addGlobalXPubs: true,
});
if (this.txFormat === 'psbt-lite') {
return (0, bitgo_1.clonePsbtWithoutNonWitnessUtxo)(psbt);
}
return psbt;
}
static suite(suiteConfig = {}) {
return (0, networks_1.getNetworkList)()
.filter((network) => (0, networks_1.isMainnet)(network) && network !== networks_1.networks.bitcoinsv)
.flatMap((network) => exports.signStages.flatMap((signStage) => exports.txFormats.flatMap((txFormat) => AcidTest.withConfig(network, signStage, txFormat, suiteConfig))));
}
}
exports.AcidTest = AcidTest;
/**
* Verifies signatures of fully signed tx (with taproot key path support).
* NOTE: taproot key path tx can only be built and signed with PSBT.
*/
function verifyFullySignedSignatures(tx, unspents, walletKeys, signer, cosigner) {
const prevOutputs = unspents.map((u) => (0, bitgo_1.toOutput)(u, tx.network));
return unspents.every((u, index) => {
if ((0, bitgo_1.parseSignatureScript2Of3)(tx.ins[index]).scriptType === 'taprootKeyPathSpend') {
const result = (0, bitgo_1.getSignatureVerifications)(tx, index, u.value, undefined, prevOutputs);
return result.length === 1 && result[0].signature;
}
else {
const result = (0, bitgo_1.verifySignatureWithUnspent)(tx, index, unspents, walletKeys);
if ((signer === 'user' && cosigner === 'bitgo') || (signer === 'bitgo' && cosigner === 'user')) {
return result[0] && !result[1] && result[2];
}
else if ((signer === 'user' && cosigner === 'backup') || (signer === 'backup' && cosigner === 'user')) {
return result[0] && result[1] && !result[2];
}
else {
return !result[0] && result[1] && result[2];
}
}
});
}
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHNidC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy90ZXN0dXRpbC9wc2J0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQW9GQSw4QkFxQkM7QUFRRCxnQ0FLQztBQU1ELHNDQXlCQztBQU1ELDhDQWdCQztBQUtELHNDQXNGQztBQTJIRCxrRUF1QkM7QUF4WkQsOENBQThDO0FBQzlDLG1DQUFzQztBQUV0QywwREFPZ0M7QUFDaEMsb0NBbUJrQjtBQUNsQiwwQ0FBMkY7QUFDM0YsaUNBQXdFO0FBQ3hFLHdDQUE0QztBQUM1QyxpQ0FBb0U7QUFtQnZELFFBQUEsVUFBVSxHQUFHLENBQUMsVUFBVSxFQUFFLFlBQVksRUFBRSxZQUFZLENBQVUsQ0FBQztBQWdCNUU7Ozs7R0FJRztBQUNVLFFBQUEsZ0JBQWdCLEdBQUcsQ0FBQyxHQUFHLCtCQUFlLEVBQUUscUJBQXFCLEVBQUUsa0NBQWtCLENBQVUsQ0FBQztBQUV6Rzs7R0FFRztBQUNVLFFBQUEsaUJBQWlCLEdBQUcsK0JBQWUsQ0FBQztBQUVqRDs7R0FFRztBQUNILFNBQWdCLFNBQVMsQ0FDdkIsS0FBWSxFQUNaLEtBQWEsRUFDYixPQUFnQixFQUNoQixjQUE4QixFQUM5QixFQUFFLFdBQVcsS0FBK0IsRUFBRTtJQUU5QyxJQUFJLEtBQUssQ0FBQyxVQUFVLEtBQUssVUFBVSxFQUFFLENBQUM7UUFDcEMsT0FBTyxJQUFBLGtDQUEyQixFQUFDLE9BQU8sRUFBRSxLQUFLLENBQUMsS0FBSyxFQUFFO1lBQ3ZELEdBQUcsRUFBRSxXQUFXLElBQUksY0FBYyxDQUFDLE1BQU0sQ0FBQztZQUMxQyxJQUFJLEVBQUUsS0FBSztTQUNaLENBQUMsQ0FBQztJQUNMLENBQUM7U0FBTSxDQUFDO1FBQ04sTUFBTSxLQUFLLEdBQUcsSUFBQSw0QkFBb0IsRUFBQyxLQUFLLENBQUMsVUFBVSxLQUFLLHFCQUFxQixDQUFDLENBQUMsQ0FBQyxZQUFZLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FB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QU0sR0FBRyxJQUFBLGtDQUEwQixFQUFDLEVBQUUsRUFBRSxLQUFLLEVBQUUsUUFBUSxFQUFFLFVBQVUsQ0FBQyxDQUFDO1lBQzNFLElBQUksQ0FBQyxNQUFNLEtBQUssTUFBTSxJQUFJLFFBQVEsS0FBSyxPQUFPLENBQUMsSUFBSSxDQUFDLE1BQU0sS0FBSyxPQUFPLElBQUksUUFBUSxLQUFLLE1BQU0sQ0FBQyxFQUFFLENBQUM7Z0JBQy9GLE9BQU8sTUFBTSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxJQUFJLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUM5QyxDQUFDO2lCQUFNLElBQUksQ0FBQyxNQUFNLEtBQUssTUFBTSxJQUFJLFFBQVEsS0FBSyxRQUFRLENBQUMsSUFBSSxDQUFDLE1BQU0sS0FBSyxRQUFRLElBQUksUUFBUSxLQUFLLE1BQU0sQ0FBQyxFQUFFLENBQUM7Z0JBQ3hHLE9BQU8sTUFBTSxDQUFDLENBQUMsQ0FBQyxJQUFJLE1BQU0sQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUM5QyxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsSUFBSSxNQUFNLENBQUMsQ0FBQyxDQUFDLElBQUksTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQzlDLENBQUM7UUFDSCxDQUFDO0lBQ0gsQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgYml0Y29pbmpzbGliIGZyb20gJ2JpdGNvaW5qcy1saWInO1xuaW1wb3J0IHsgb2sgYXMgYXNzZXJ0IH0gZnJvbSAnYXNzZXJ0JztcblxuaW1wb3J0IHtcbiAgY3JlYXRlT3V0cHV0U2NyaXB0UDJzaFAycGssXG4gIGlzU3VwcG9ydGVkU2NyaXB0VHlwZSxcbiAgU2NyaXB0VHlwZSxcbiAgU2NyaXB0VHlwZTJPZjMsXG4gIHNjcmlwdFR5cGVQMnNoUDJwayxcbiAgc2NyaXB0VHlwZXMyT2YzLFxufSBmcm9tICcuLi9iaXRnby9vdXRwdXRTY3JpcHRzJztcbmltcG9ydCB7XG4gIGFkZFJlcGxheVByb3RlY3Rpb25VbnNwZW50VG9Qc2J0LFxuICBhZGRXYWxsZXRPdXRwdXRUb1BzYnQsXG4gIGFkZFdhbGxldFVuc3BlbnRUb1BzYnQsXG4gIGNyZWF0ZVBzYnRGb3JOZXR3b3JrLFxuICBnZXRFeHRlcm5hbENoYWluQ29kZSxcbiAgZ2V0SW50ZXJuYWxDaGFpbkNvZGUsXG4gIGdldFNpZ25hdHVyZVZlcmlmaWNhdGlvbnMsXG4gIGlzV2FsbGV0VW5zcGVudCxcbiAgS2V5TmFtZSxcbiAgcGFyc2VTaWduYXR1cmVTY3JpcHQyT2YzLFxuICBSb290V2FsbGV0S2V5cyxcbiAgdG9PdXRwdXQsXG4gIFVuc3BlbnQsXG4gIFV0eG9Qc2J0LFxuICBVdHhvVHJhbnNhY3Rpb24sXG4gIHZlcmlmeVNpZ25hdHVyZVdpdGhVbnNwZW50LFxuICBhZGRYcHVic1RvUHNidCxcbiAgY2xvbmVQc2J0V2l0aG91dE5vbldpdG5lc3NVdHhvLFxufSBmcm9tICcuLi9iaXRnbyc7XG5pbXBvcnQgeyBnZXROZXR3b3JrTGlzdCwgZ2V0TmV0d29ya05hbWUsIGlzTWFpbm5ldCwgTmV0d29yaywgbmV0d29ya3MgfSBmcm9tICcuLi9uZXR3b3Jrcyc7XG5pbXBvcnQgeyBtb2NrUmVwbGF5UHJvdGVjdGlvblVuc3BlbnQsIG1vY2tXYWxsZXRVbnNwZW50IH0gZnJvbSAnLi9tb2NrJztcbmltcG9ydCB7IHRvT3V0cHV0U2NyaXB0IH0gZnJvbSAnLi4vYWRkcmVzcyc7XG5pbXBvcnQgeyBnZXREZWZhdWx0V2FsbGV0S2V5cywgZ2V0V2FsbGV0S2V5c0ZvclNlZWQgfSBmcm9tICcuL2tleXMnO1xuXG4vKipcbiAqIFRoaXMgaXMgYSBiaXQgb2YgYSBtaXNub21lciwgYXMgaXQgYWN0dWFsbHkgc3BlY2lmaWVzIHRoZSBzcGVuZCB0eXBlIG9mIHRoZSBpbnB1dC5cbiAqIFRoaXMgbWFrZXMgYSBkaWZmZXJlbmNlIGZvciBwMnRyTXVzaWcyIGlucHV0cywgYXMgdGhleSBjYW4gYmUgc3BlbnQgZWl0aGVyIGJ5IGtleSBwYXRoIG9yIHNjcmlwdCBwYXRoLlxuICogVGhlIHZhbHVlIHAydHJNdXNpZzIgaXMgdXNlZCBmb3IgcDJ0ck11c2lnMiBzY3JpcHQgcGF0aC5cbiAqIFRoZSB2YWx1ZSB0YXByb290S2V5UGF0aFNwZW5kIGlzIHVzZWQgZm9yIHAydHJNdXNpZzIga2V5IHBhdGguXG4gKi9cbmV4cG9ydCB0eXBlIElucHV0U2NyaXB0VHlwZSA9IFNjcmlwdFR5cGUgfCAndGFwcm9vdEtleVBhdGhTcGVuZCc7XG5leHBvcnQgdHlwZSBPdXRwdXRTY3JpcHRUeXBlID0gU2NyaXB0VHlwZTJPZjM7XG5cbi8qKlxuICogaW5wdXQgc2NyaXB0IHR5cGUgYW5kIHZhbHVlXG4gKi9cbmV4cG9ydCB0eXBlIElucHV0ID0ge1xuICBzY3JpcHRUeXBlOiBJbnB1dFNjcmlwdFR5cGU7XG4gIHZhbHVlOiBiaWdpbnQ7XG59O1xuXG5leHBvcnQgY29uc3Qgc2lnblN0YWdlcyA9IFsndW5zaWduZWQnLCAnaGFsZnNpZ25lZCcsICdmdWxsc2lnbmVkJ10gYXMgY29uc3Q7XG5leHBvcnQgdHlwZSBTaWduU3RhZ2UgPSAodHlwZW9mIHNpZ25TdGFnZXMpW251bWJlcl07XG5cbi8qKlxuICogU2V0IGlzSW50ZXJuYWxBZGRyZXNzPXRydWUgZm9yIGludGVybmFsIG91dHB1dCBhZGRyZXNzXG4gKi9cbi8vIE1ha2Ugc2NyaXB0OiBzdHJpbmcgYXMgaW5zdGVhZCBvZiBzY3JpcHRUeXBlIG9yIGFkZHJlc3NcbmV4cG9ydCB0eXBlIE91dHB1dCA9IHtcbiAgdmFsdWU6IGJpZ2ludDtcbiAgLy8gRGV0ZXJtaW5lcyBjaGFpbiBjb2RlIGZvciB0aGUgb3V0cHV0XG4gIGlzSW50ZXJuYWxBZGRyZXNzPzogYm9vbGVhbjtcbiAgLy8gRGV0ZXJtaW5lcyB0aGUgd2FsbGV0IGtleXMgdG8gdXNlIGZvciB0aGUgb3V0cHV0LiBCeSBkZWZhdWx0IHVzZSByb290IHdhbGxldCBrZXlzIHVzZWQgZm9yIHRoZSBpbnB1dHMuXG4gIC8vIFdoZW4gc2V0IHRvIG51bGwsIG9taXRzIHRoZSBkZXJpdmF0aW9uIGluZm8gYW5kIGVmZmVjdGl2ZWx5IG1ha2VzIHRoZSBvdXRwdXQgbm9uLXdhbGxldCBvdXRwdXQuXG4gIHdhbGxldEtleXM/OiBSb290V2FsbGV0S2V5cyB8IG51bGw7XG59ICYgKHsgc2NyaXB0VHlwZTogT3V0cHV0U2NyaXB0VHlwZSB9IHwgeyBhZGRyZXNzOiBzdHJpbmcgfSB8IHsgc2NyaXB0OiBzdHJpbmcgfSB8IHsgb3BSZXR1cm46IHN0cmluZyB9KTtcblxuLyoqXG4gKiBhcnJheSBvZiBzdXBwb3J0ZWQgaW5wdXQgc2NyaXB0IHR5cGVzLlxuICogdXNlIHAydHJNdXNpZzIgZm9yIHAydHJNdXNpZzIgc2NyaXB0IHBhdGguXG4gKiB1c2UgdGFwcm9vdEtleVBhdGhTcGVuZCBmb3IgcDJ0ck11c2lnMiBrZXkgcGF0aC5cbiAqL1xuZXhwb3J0IGNvbnN0IGlucHV0U2NyaXB0VHlwZXMgPSBbLi4uc2NyaXB0VHlwZXMyT2YzLCAndGFwcm9vdEtleVBhdGhTcGVuZCcsIHNjcmlwdFR5cGVQMnNoUDJwa10gYXMgY29uc3Q7XG5cbi8qKlxuICogYXJyYXkgb2Ygc3VwcG9ydGVkIG91dHB1dCBzY3JpcHQgdHlwZXMuXG4gKi9cbmV4cG9ydCBjb25zdCBvdXRwdXRTY3JpcHRUeXBlcyA9IHNjcmlwdFR5cGVzMk9mMztcblxuLyoqXG4gKiBjcmVhdGUgdW5zcGVudCBvYmplY3QgZnJvbSBpbnB1dCBzY3JpcHQgdHlwZSwgaW5kZXgsIG5ldHdvcmsgYW5kIHJvb3Qgd2FsbGV0IGtleS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHRvVW5zcGVudChcbiAgaW5wdXQ6IElucHV0LFxuICBpbmRleDogbnVtYmVyLFxuICBuZXR3b3JrOiBOZXR3b3JrLFxuICByb290V2FsbGV0S2V5czogUm9vdFdhbGxldEtleXMsXG4gIHsgcDJzaFAycGtLZXkgfTogeyBwMnNoUDJwa0tleT86IEJ1ZmZlciB9ID0ge31cbik6IFVuc3BlbnQ8YmlnaW50PiB7XG4gIGlmIChpbnB1dC5zY3JpcHRUeXBlID09PSAncDJzaFAycGsnKSB7XG4gICAgcmV0dXJuIG1vY2tSZXBsYXlQcm90ZWN0aW9uVW5zcGVudChuZXR3b3JrLCBpbnB1dC52YWx1ZSwge1xuICAgICAga2V5OiBwMnNoUDJwa0tleSA/PyByb290V2FsbGV0S2V5c1sndXNlciddLFxuICAgICAgdm91dDogaW5kZXgsXG4gICAgfSk7XG4gIH0gZWxzZSB7XG4gICAgY29uc3QgY2hhaW4gPSBnZXRJbnRlcm5hbENoYWluQ29kZShpbnB1dC5zY3JpcHRUeXBlID09PSAndGFwcm9vdEtleVBhdGhTcGVuZCcgPyAncDJ0ck11c2lnMicgOiBpbnB1dC5zY3JpcHRUeXBlKTtcbiAgICByZXR1cm4gbW9ja1dhbGxldFVuc3BlbnQobmV0d29yaywgaW5wdXQudmFsdWUsIHtcbiAgICAgIGNoYWluLFxuICAgICAgdm91dDogaW5kZXgsXG4gICAgICBrZXlzOiByb290V2FsbGV0S2V5cyxcbiAgICAgIGluZGV4LFxuICAgIH0pO1xuICB9XG59XG5cbi8qKlxuICogcmV0dXJucyBzaWduZXIgYW5kIGNvc2lnbmVyIG5hbWVzIGZvciBJbnB1dFNjcmlwdFR5cGUuXG4gKiB1c2VyIGFuZCB1bmRlZmluZWQgYXMgc2lnbmVyIGFuZCBjb3NpZ25lciByZXNwZWN0aXZlbHkgZm9yIHAyc2hQMnBrLlxuICogdXNlciBhbmQgYmFja3VwIGFzIHNpZ25lciBhbmQgY29zaWduZXIgcmVzcGVjdGl2ZWx5IGZvciBwMnRyTXVzaWcyLlxuICogdXNlciBhbmQgYml0Z28gYXMgc2lnbmVyIGFuZCBjb3NpZ25lciByZXNwZWN0aXZlbHkgZm9yIG90aGVyIGlucHV0IHNjcmlwdCB0eXBlcy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldFNpZ25lcnMoaW5wdXRUeXBlOiBJbnB1dFNjcmlwdFR5cGUpOiB7IHNpZ25lck5hbWU6IEtleU5hbWU7IGNvc2lnbmVyTmFtZT86IEtleU5hbWUgfSB7XG4gIHJldHVybiB7XG4gICAgc2lnbmVyTmFtZTogJ3VzZXInLFxuICAgIGNvc2lnbmVyTmFtZTogaW5wdXRUeXBlID09PSAncDJzaFAycGsnID8gdW5kZWZpbmVkIDogaW5wdXRUeXBlID09PSAncDJ0ck11c2lnMicgPyAnYmFja3VwJyA6ICdiaXRnbycsXG4gIH07XG59XG5cbi8qKlxuICogc2lnbnMgd2l0aCBmaXJzdCBvciBzZWNvbmQgc2lnbmF0dXJlIGZvciBzaW5nbGUgaW5wdXQuXG4gKiBwMnNoUDJwayBpcyBzaWduZWQgb25seSB3aXRoIGZpcnN0IHNpZ24uXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBzaWduUHNidElucHV0KFxuICBwc2J0OiBVdHhvUHNidCxcbiAgaW5wdXQ6IElucHV0LFxuICBpbnB1dEluZGV4OiBudW1iZXIsXG4gIHJvb3RXYWxsZXRLZXlzOiBSb290V2FsbGV0S2V5cyxcbiAgc2lnbjogJ2hhbGZzaWduZWQnIHwgJ2Z1bGxzaWduZWQnLFxuICBwYXJhbXM/OiB7XG4gICAgc2lnbmVycz86IHsgc2lnbmVyTmFtZTogS2V5TmFtZTsgY29zaWduZXJOYW1lPzogS2V5TmFtZSB9O1xuICAgIGRldGVybWluaXN0aWM/OiBib29sZWFuO1xuICAgIC8vIEZvciBiYWNrd2FyZHMgY29tcGF0aWJpbGl0eSBrZWVwIHRoaXMgaGVyZS5cbiAgICBza2lwTm9uV2l0bmVzc1V0eG8/OiBib29sZWFuO1xuICB9XG4pOiB2b2lkIHtcbiAgY29uc3QgeyBzaWduZXJzLCBkZXRlcm1pbmlzdGljIH0gPSBwYXJhbXMgPz8ge307XG4gIGNvbnN0IHsgc2lnbmVyTmFtZSwgY29zaWduZXJOYW1lIH0gPSBzaWduZXJzID8gc2lnbmVycyA6IGdldFNpZ25lcnMoaW5wdXQuc2NyaXB0VHlwZSk7XG4gIGlmIChzaWduID09PSAnaGFsZnNpZ25lZCcpIHtcbiAgICBpZiAoaW5wdXQuc2NyaXB0VHlwZSA9PT0gJ3Ayc2hQMnBrJykge1xuICAgICAgcHNidC5zaWduSW5wdXQoaW5wdXRJbmRleCwgcm9vdFdhbGxldEtleXNbc2lnbmVyTmFtZV0pO1xuICAgIH0gZWxzZSB7XG4gICAgICBwc2J0LnNpZ25JbnB1dEhEKGlucHV0SW5kZXgsIHJvb3RXYWxsZXRLZXlzW3NpZ25lck5hbWVdKTtcbiAgICB9XG4gIH1cbiAgaWYgKHNpZ24gPT09ICdmdWxsc2lnbmVkJyAmJiBjb3NpZ25lck5hbWUgJiYgaW5wdXQuc2NyaXB0VHlwZSAhPT0gJ3Ayc2hQMnBrJykge1xuICAgIHBzYnQuc2lnbklucHV0SEQoaW5wdXRJbmRleCwgcm9vdFdhbGxldEtleXNbY29zaWduZXJOYW1lXSwgeyBkZXRlcm1pbmlzdGljIH0pO1xuICB9XG59XG5cbi8qKlxuICogc2lnbnMgd2l0aCBmaXJzdCBvciBzZWNvbmQgc2lnbmF0dXJlIGZvciBhbGwgaW5wdXRzLlxuICogcDJzaFAycGsgaXMgc2lnbmVkIG9ubHkgd2l0aCBmaXJzdCBzaWduLlxuICovXG5leHBvcnQgZnVuY3Rpb24gc2lnbkFsbFBzYnRJbnB1dHMoXG4gIHBzYnQ6IFV0eG9Qc2J0LFxuICBpbnB1dHM6IElucHV0W10sXG4gIHJvb3RXYWxsZXRLZXlzOiBSb290V2FsbGV0S2V5cyxcbiAgc2lnbjogJ2hhbGZzaWduZWQnIHwgJ2Z1bGxzaWduZWQnLFxuICBwYXJhbXM/OiB7XG4gICAgc2lnbmVycz86IHsgc2lnbmVyTmFtZTogS2V5TmFtZTsgY29zaWduZXJOYW1lPzogS2V5TmFtZSB9O1xuICAgIGRldGVybWluaXN0aWM/OiBib29sZWFuO1xuICAgIC8vIEZvciBiYWNrd2FyZHMgY29tcGF0aWJpbGl0eSBrZWVwIHRoaXMgaGVyZS5cbiAgICBza2lwTm9uV2l0bmVzc1V0eG8/OiBib29sZWFuO1xuICB9XG4pOiB2b2lkIHtcbiAgY29uc3QgeyBzaWduZXJzLCBkZXRlcm1pbmlzdGljIH0gPSBwYXJhbXMgPz8ge307XG4gIGlucHV0cy5mb3JFYWNoKChpbnB1dCwgaW5wdXRJbmRleCkgPT4ge1xuICAgIHNpZ25Qc2J0SW5wdXQocHNidCwgaW5wdXQsIGlucHV0SW5kZXgsIHJvb3RXYWxsZXRLZXlzLCBzaWduLCB7IHNpZ25lcnMsIGRldGVybWluaXN0aWMgfSk7XG4gIH0pO1xufVxuXG4vKipcbiAqIGNvbnN0cnVjdCBwc2J0IGZvciBnaXZlbiBpbnB1dHMsIG91dHB1dHMsIG5ldHdvcmsgYW5kIHJvb3Qgd2FsbGV0IGtleXMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb25zdHJ1Y3RQc2J0KFxuICBpbnB1dHM6IElucHV0W10sXG4gIG91dHB1dHM6IE91dHB1dFtdLFxuICBuZXR3b3JrOiBOZXR3b3JrLFxuICByb290V2FsbGV0S2V5czogUm9vdFdhbGxldEtleXMsXG4gIHNpZ25TdGFnZTogU2lnblN0YWdlLFxuICBwYXJhbXM/OiB7XG4gICAgcDJzaFAycGtLZXk/OiBCdWZmZXI7XG4gICAgc2lnbmVycz86IHsgc2lnbmVyTmFtZTogS2V5TmFtZTsgY29zaWduZXJOYW1lPzogS2V5TmFtZSB9O1xuICAgIGRldGVybWluaXN0aWM/OiBib29sZWFuO1xuICAgIHNraXBOb25XaXRuZXNzVXR4bz86IGJvb2xlYW47XG4gICAgYWRkR2xvYmFsWFB1YnM/OiBib29sZWFuO1xuICB9XG4pOiBVdHhvUHNidCB7XG4gIGNvbnN0IHsgc2lnbmVycywgZGV0ZXJtaW5pc3RpYywgc2tpcE5vbldpdG5lc3NVdHhvIH0gPSBwYXJhbXMgPz8ge307XG4gIGNvbnN0IHRvdGFsSW5wdXRBbW91bnQgPSBpbnB1dHMucmVkdWNlKChzdW0sIGlucHV0KSA9PiBzdW0gKyBpbnB1dC52YWx1ZSwgQmlnSW50KDApKTtcbiAgY29uc3Qgb3V0cHV0SW5wdXRBbW91bnQgPSBvdXRwdXRzLnJlZHVjZSgoc3VtLCBvdXRwdXQpID0+IHN1bSArIG91dHB1dC52YWx1ZSwgQmlnSW50KDApKTtcbiAgYXNzZXJ0KHRvdGFsSW5wdXRBbW91bnQgPj0gb3V0cHV0SW5wdXRBbW91bnQsICd0b3RhbCBvdXRwdXQgY2FuIG5vdCBleGNlZWQgdG90YWwgaW5wdXQnKTtcblxuICBjb25zdCBwc2J0ID0gY3JlYXRlUHNidEZvck5ldHdvcmsoeyBuZXR3b3JrIH0pO1xuXG4gIGlmIChwYXJhbXM/LmFkZEdsb2JhbFhQdWJzKSB7XG4gICAgYWRkWHB1YnNUb1BzYnQocHNidCwgcm9vdFdhbGxldEtleXMpO1xuICB9XG5cbiAgY29uc3QgdW5zcGVudHMgPSBpbnB1dHMubWFwKChpbnB1dCwgaSkgPT5cbiAgICB0b1Vuc3BlbnQoaW5wdXQsIGksIG5ldHdvcmssIHJvb3RXYWxsZXRLZXlzLCB7IHAyc2hQMnBrS2V5OiBwYXJhbXM/LnAyc2hQMnBrS2V5IH0pXG4gICk7XG5cbiAgdW5zcGVudHMuZm9yRWFjaCgodSwgaSkgPT4ge1xuICAgIGNvbnN0IHsgc2lnbmVyTmFtZSwgY29zaWduZXJOYW1lIH0gPSBzaWduZXJzID8gc2lnbmVycyA6IGdldFNpZ25lcnMoaW5wdXRzW2ldLnNjcmlwdFR5cGUpO1xuICAgIGlmIChpc1dhbGxldFVuc3BlbnQodSkgJiYgY29zaWduZXJOYW1lKSB7XG4gICAgICBhZGRXYWxsZXRVbnNwZW50VG9Qc2J0KHBzYnQsIHUsIHJvb3RXYWxsZXRLZXlzLCBzaWduZXJOYW1lLCBjb3NpZ25lck5hbWUsIHsgc2tpcE5vbldpdG5lc3NVdHhvIH0pO1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCB7IHJlZGVlbVNjcmlwdCB9ID0gY3JlYXRlT3V0cHV0U2NyaXB0UDJzaFAycGsocGFyYW1zPy5wMnNoUDJwa0tleSA/PyByb290V2FsbGV0S2V5cy51c2VyLnB1YmxpY0tleSk7XG4gICAgICBhc3NlcnQocmVkZWVtU2NyaXB0KTtcbiAgICAgIGFkZFJlcGxheVByb3RlY3Rpb25VbnNwZW50VG9Qc2J0KHBzYnQsIHUsIHJlZGVlbVNjcmlwdCwgeyBza2lwTm9uV2l0bmVzc1V0eG8gfSk7XG4gICAgfVxuICB9KTtcblxuICBvdXRwdXRzLmZvckVhY2goKG91dHB1dCwgaSkgPT4ge1xuICAgIGlmICgnc2NyaXB0VHlwZScgaW4gb3V0cHV0KSB7XG4gICAgICBhZGRXYWxsZXRPdXRwdXRUb1BzYnQoXG4gICAgICAgIHBzYnQsXG4gICAgICAgIG91dHB1dC53YWxsZXRLZXlzID8/IHJvb3RXYWxsZXRLZXlzLFxuICAgICAgICBvdXRwdXQuaXNJbnRlcm5hbEFkZHJlc3MgPyBnZXRJbnRlcm5hbENoYWluQ29kZShvdXRwdXQuc2NyaXB0VHlwZSkgOiBnZXRFeHRlcm5hbENoYWluQ29kZShvdXRwdXQuc2NyaXB0VHlwZSksXG4gICAgICAgIGksXG4gICAgICAgIG91dHB1dC52YWx1ZSxcbiAgICAgICAgeyBhZGREZXJpdmF0aW9uSW5mbzogb3V0cHV0LndhbGxldEtleXMgIT09IG51bGwgfVxuICAgICAgKTtcbiAgICAgIHJldHVybjtcbiAgICB9IGVsc2UgaWYgKCdhZGRyZXNzJyBpbiBvdXRwdXQpIHtcbiAgICAgIGNvbnN0IHsgYWRkcmVzcywgdmFsdWUgfSA9IG91dHB1dDtcbiAgICAgIHBzYnQuYWRkT3V0cHV0KHsgc2NyaXB0OiB0b091dHB1dFNjcmlwdChhZGRyZXNzLCBuZXR3b3JrKSwgdmFsdWUgfSk7XG4gICAgICByZXR1cm47XG4gICAgfSBlbHNlIGlmICgnb3BSZXR1cm4nIGluIG91dHB1dCkge1xuICAgICAgY29uc3QgeyBvcFJldHVybiwgdmFsdWUgfSA9IG91dHB1dDtcbiAgICAgIGNvbnN0IHNjcmlwdCA9IGJpdGNvaW5qc2xpYi5wYXltZW50cy5lbWJlZCh7IGRhdGE6IFtCdWZmZXIuZnJvbShvcFJldHVybiwgJ2FzY2lpJyldIH0pLm91dHB1dDtcbiAgICAgIGFzc2VydChzY3JpcHQsICdzY3JpcHQgaXMgcmVxdWlyZWQnKTtcbiAgICAgIHBzYnQuYWRkT3V0cHV0KHsgc2NyaXB0LCB2YWx1ZSB9KTtcbiAgICAgIHJldHVybjtcbiAgICB9IGVsc2UgaWYgKCdzY3JpcHQnIGluIG91dHB1dCkge1xuICAgICAgY29uc3QgeyBzY3JpcHQsIHZhbHVlIH0gPSBvdXRwdXQ7XG4gICAgICBwc2J0LmFkZE91dHB1dCh7IHNjcmlwdDogQnVmZmVyLmZyb20oc2NyaXB0LCAnaGV4JyksIHZhbHVlIH0pO1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIHRocm93IG5ldyBFcnJvcignaW52YWxpZCBvdXRwdXQnKTtcbiAgfSk7XG5cbiAgaWYgKHNpZ25TdGFnZSA9PT0gJ3Vuc2lnbmVkJykge1xuICAgIHJldHVybiBwc2J0O1xuICB9XG5cbiAgLyogdXNlIGZpeGVkIHNlc3Npb25JZCBmb3IgZGV0ZXJtaW5pc3RpYyBub25jZSBjcmVhdGlvbiAqL1xuICBjb25zdCBzZXNzaW9uSWQgPSBCdWZmZXIuYWxsb2MoMzIpO1xuICBwc2J0LnNldEFsbElucHV0c011c2lnMk5vbmNlSEQocm9vdFdhbGxldEtleXNbJ3VzZXInXSwgeyBzZXNzaW9uSWQgfSk7XG4gIHBzYnQuc2V0QWxsSW5wdXRzTXVzaWcyTm9uY2VIRChyb290V2FsbGV0S2V5c1snYml0Z28nXSwgeyBkZXRlcm1pbmlzdGljIH0pO1xuXG4gIHNpZ25BbGxQc2J0SW5wdXRzKHBzYnQsIGlucHV0cywgcm9vdFdhbGxldEtleXMsICdoYWxmc2lnbmVkJywgeyBzaWduZXJzLCBza2lwTm9uV2l0bmVzc1V0eG8gfSk7XG5cbiAgaWYgKHNpZ25TdGFnZSA9PT0gJ2Z1bGxzaWduZWQnKSB7XG4gICAgc2lnbkFsbFBzYnRJbnB1dHMocHNidCwgaW5wdXRzLCByb290V2FsbGV0S2V5cywgc2lnblN0YWdlLCB7IHNpZ25lcnMsIGRldGVybWluaXN0aWMsIHNraXBOb25XaXRuZXNzVXR4byB9KTtcbiAgfVxuXG4gIHJldHVybiBwc2J0O1xufVxuXG5leHBvcnQgY29uc3QgdHhGb3JtYXRzID0gWydwc2J0JywgJ3BzYnQtbGl0ZSddIGFzIGNvbnN0O1xuZXhwb3J0IHR5cGUgVHhGb3JtYXQgPSAodHlwZW9mIHR4Rm9ybWF0cylbbnVtYmVyXTtcblxudHlwZSBTdWl0ZUNvbmZpZyA9IHtcbiAgLyoqXG4gICAqIEJ5IGRlZmF1bHQsIHdlIGluY2x1ZGUgcDJ0ck11c2lnMiBzY3JpcHQgcGF0aCBpbiB0aGUgaW5wdXRzLlxuICAgKiBUaGlzIGlucHV0IGlzIGEgYml0IG9mIGEgd2VpcmRvIGJlY2F1c2UgaXQgaXMgc2lnbmVkIGJ5IHRoZSB1c2VyIGFuZCB0aGVcbiAgICogYmFja3VwIGtleSwgd2hpY2ggdXN1YWxseSBpcyBub3QgbWl4ZWQgd2l0aCBvdGhlciBpbnB1dHMgYW5kIG91dHB1dHMuXG4gICAqXG4gICAqIFRoaXMgb3B0aW9uIGFsbG93cyB0byBleGNsdWRlIHRoaXMgaW5wdXQgZnJvbSB0aGUgaW5wdXRzLlxuICAgKi9cbiAgaW5jbHVkZVAydHJNdXNpZzJTY3JpcHRQYXRoPzogYm9vbGVhbjtcbn07XG5cbi8qKlxuICogQ3JlYXRlcyBhIHZhbGlkIFBTQlQgd2l0aCBhcyBtYW55IGZlYXR1cmVzIGFzIHBvc3NpYmxlLlxuICpcbiAqIC0gSW5wdXRzOlxuICogICAtIEFsbCB3YWxsZXQgc2NyaXB0IHR5cGVzIHRoYXQgYXJlIHN1cHBvcnRlZCBieSB0aGUgbmV0d29yay5cbiAqICAgLSBBIHAyc2hQMnBrIGlucHV0IChmb3IgcmVwbGF5IHByb3RlY3Rpb24pXG4gKiAtIE91dHB1dHM6XG4gKiAgIC0gQWxsIHdhbGxldCBzY3JpcHQgdHlwZXMgdGhhdCBhcmUgc3VwcG9ydGVkIGJ5IHRoZSBuZXR3b3JrLlxuICogICAtIEEgcDJzaCBvdXRwdXQgd2l0aCBkZXJpdmF0aW9uIGluZm8gb2YgYSBkaWZmZXJlbnQgd2FsbGV0IChub3QgaW4gdGhlIGdsb2JhbCBwc2J0IHhwdWJzKVxuICogICAtIEEgcDJzaCBvdXRwdXQgd2l0aCBubyBkZXJpdmF0aW9uIGluZm8gKGV4dGVybmFsIG91dHB1dClcbiAqICAgLSBBbiBPUF9SRVRVUk4gb3V0cHV0XG4gKi9cbmV4cG9ydCBjbGFzcyBBY2lkVGVzdCB7XG4gIHB1YmxpYyByZWFkb25seSBuZXR3b3JrOiBOZXR3b3JrO1xuICBwdWJsaWMgcmVhZG9ubHkgc2lnblN0YWdlOiBTaWduU3RhZ2U7XG4gIHB1YmxpYyByZWFkb25seSB0eEZvcm1hdDogVHhGb3JtYXQ7XG4gIHB1YmxpYyByZWFkb25seSByb290V2FsbGV0S2V5czogUm9vdFdhbGxldEtleXM7XG4gIHB1YmxpYyByZWFkb25seSBvdGhlcldhbGxldEtleXM6IFJvb3RXYWxsZXRLZXlzO1xuICBwdWJsaWMgcmVhZG9ubHkgaW5wdXRzOiBJbnB1dFtdO1xuICBwdWJsaWMgcmVhZG9ubHkgb3V0cHV0czogT3V0cHV0W107XG5cbiAgY29uc3RydWN0b3IoXG4gICAgbmV0d29yazogTmV0d29yayxcbiAgICBzaWduU3RhZ2U6IFNpZ25TdGFnZSxcbiAgICB0eEZvcm1hdDogVHhGb3JtYXQsXG4gICAgcm9vdFdhbGxldEtleXM6IFJvb3RXYWxsZXRLZXlzLFxuICAgIG90aGVyV2FsbGV0S2V5czogUm9vdFdhbGxldEtleXMsXG4gICAgaW5wdXRzOiBJbnB1dFtdLFxuICAgIG91dHB1dHM6IE91dHB1dFtdXG4gICkge1xuICAgIHRoaXMubmV0d29yayA9IG5ldHdvcms7XG4gICAgdGhpcy5zaWduU3RhZ2UgPSBzaWduU3RhZ2U7XG4gICAgdGhpcy50eEZvcm1hdCA9IHR4Rm9ybWF0O1xuICAgIHRoaXMucm9vdFdhbGxldEtleXMgPSByb290V2FsbGV0S2V5cztcbiAgICB0aGlzLm90aGVyV2FsbGV0S2V5cyA9IG90aGVyV2FsbGV0S2V5cztcbiAgICB0aGlzLmlucHV0cyA9IGlucHV0cztcbiAgICB0aGlzLm91dHB1dHMgPSBvdXRwdXRzO1xuICB9XG5cbiAgc3RhdGljIHdpdGhDb25maWcobmV0d29yazogTmV0d29yaywgc2lnblN0YWdlOiBTaWduU3RhZ2UsIHR4Rm9ybWF0OiBUeEZvcm1hdCwgc3VpdGVDb25maWc6IFN1aXRlQ29uZmlnKTogQWNpZFRlc3Qge1xuICAgIGNvbnN0IHJvb3RXYWxsZXRLZXlzID0gZ2V0RGVmYXVsdFdhbGxldEtleXMoKTtcblxuICAgIGNvbnN0IG90aGVyV2FsbGV0S2V5cyA9IGdldFdhbGxldEtleXNGb3JTZWVkKCd0b28gbWFueSBzZWNyZXRzJyk7XG4gICAgY29uc3QgaW5wdXRzOiBJbnB1dFtdID0gaW5wdXRTY3JpcHRUeXBlc1xuICAgICAgLmZpbHRlcigoc2NyaXB0VHlwZSkgPT5cbiAgICAgICAgc2NyaXB0VHlwZSA9PT0gJ3RhcHJvb3RLZXlQYXRoU3BlbmQnXG4gICAgICAgICAgPyBpc1N1cHBvcnRlZFNjcmlwdFR5cGUobmV0d29yaywgJ3AydHJNdXNpZzInKVxuICAgICAgICAgIDogaXNTdXBwb3J0ZWRTY3JpcHRUeXBlKG5ldHdvcmssIHNjcmlwdFR5cGUpXG4gICAgICApXG4gICAgICAuZmlsdGVyKChzY3JpcHRUeXBlKSA9PiAoc3VpdGVDb25maWcuaW5jbHVkZVAydHJNdXNpZzJTY3JpcHRQYXRoID8/IHRydWUpIHx8IHNjcmlwdFR5cGUgIT09ICdwMnRyTXVzaWcyJylcbiAgICAgIC5tYXAoKHNjcmlwdFR5cGUpID0+ICh7IHNjcmlwdFR5cGUsIHZhbHVlOiBCaWdJbnQoMjAwMCkgfSkpO1xuXG4gICAgY29uc3Qgb3V0cHV0czogT3V0cHV0W10gPSBvdXRwdXRTY3JpcHRUeXBlc1xuICAgICAgLmZpbHRlcigoc2NyaXB0VHlwZSkgPT4gaXNTdXBwb3J0ZWRTY3JpcHRUeXBlKG5ldHdvcmssIHNjcmlwdFR5cGUpKVxuICAgICAgLm1hcCgoc2NyaXB0VHlwZSkgPT4gKHsgc2NyaXB0VHlwZSwgdmFsdWU6IEJpZ0ludCg5MDApIH0pKTtcblxuICAgIC8vIFRlc3Qgb3RoZXIgd2FsbGV0IG91dHB1dCAod2l0aCBkZXJpdmF0aW9uIGluZm8pXG4gICAgb3V0cHV0cy5wdXNoKHsgc2NyaXB0VHlwZTogJ3Ayc2gnLCB2YWx1ZTogQmlnSW50KDkwMCksIHdhbGxldEtleXM6IG90aGVyV2FsbGV0S2V5cyB9KTtcbiAgICAvLyBUZXMgbm9uLXdhbGxldCBvdXRwdXRcbiAgICBvdXRwdXRzLnB1c2goeyBzY3JpcHRUeXBlOiAncDJzaCcsIHZhbHVlOiBCaWdJbnQoOTAwKSwgd2FsbGV0S2V5czogbnVsbCB9KTtcbiAgICAvLyBUZXN0IE9QX1JFVFVSTiBvdXRwdXRcbiAgICBvdXRwdXRzLnB1c2goeyBvcFJldHVybjogJ3NldGVjIGFzdHJvbm9teScsIHZhbHVlOiBCaWdJbnQoMCkgfSk7XG5cbiAgICByZXR1cm4gbmV3IEFjaWRUZXN0KG5ldHdvcmssIHNpZ25TdGFnZSwgdHhGb3JtYXQsIHJvb3RXYWxsZXRLZXlzLCBvdGhlcldhbGxldEtleXMsIGlucHV0cywgb3V0cHV0cyk7XG4gIH1cblxuICBnZXQgbmFtZSgpOiBzdHJpbmcge1xuICAgIGNvbnN0IG5ldHdvcmtOYW1lID0gZ2V0TmV0d29ya05hbWUodGhpcy5uZXR3b3JrKTtcbiAgICByZXR1cm4gYCR7bmV0d29ya05hbWV9ICR7dGhpcy5zaWduU3RhZ2V9ICR7dGhpcy50eEZvcm1hdH1gO1xuICB9XG5cbiAgZ2V0UmVwbGF5UHJvdGVjdGlvblB1YmxpY0tleSgpOiBCdWZmZXIge1xuICAgIHJldHVybiB0aGlzLnJvb3RXYWxsZXRLZXlzLnVzZXIucHVibGljS2V5O1xuICB9XG5cbiAgZ2V0UmVwbGF5UHJvdGVjdGlvbk91dHB1dFNjcmlwdCgpOiBCdWZmZXIge1xuICAgIGNvbnN0IHsgc2NyaXB0UHViS2V5IH0gPSBjcmVhdGVPdXRwdXRTY3JpcHRQMnNoUDJwayh0aGlzLmdldFJlcGxheVByb3RlY3Rpb25QdWJsaWNLZXkoKSk7XG4gICAgYXNzZXJ0KHNjcmlwdFB1YktleSk7XG4gICAgcmV0dXJuIHNjcmlwdFB1YktleTtcbiAgfVxuXG4gIGNyZWF0ZVBzYnQoKTogVXR4b1BzYnQge1xuICAgIGNvbnN0IHBzYnQgPSBjb25zdHJ1Y3RQc2J0KHRoaXMuaW5wdXRzLCB0aGlzLm91dHB1dHMsIHRoaXMubmV0d29yaywgdGhpcy5yb290V2FsbGV0S2V5cywgdGhpcy5zaWduU3RhZ2UsIHtcbiAgICAgIGRldGVybWluaXN0aWM6IHRydWUsXG4gICAgICBhZGRHbG9iYWxYUHViczogdHJ1ZSxcbiAgICB9KTtcbiAgICBpZiAodGhpcy50eEZvcm1hdCA9PT0gJ3BzYnQtbGl0ZScpIHtcbiAgICAgIHJldHVybiBjbG9uZVBzYnRXaXRob3V0Tm9uV2l0bmVzc1V0eG8ocHNidCk7XG4gICAgfVxuICAgIHJldHVybiBwc2J0O1xuICB9XG5cbiAgc3RhdGljIHN1aXRlKHN1aXRlQ29uZmlnOiBTdWl0ZUNvbmZpZyA9IHt9KTogQWNpZFRlc3RbXSB7XG4gICAgcmV0dXJuIGdldE5ldHdvcmtMaXN0KClcbiAgICAgIC5maWx0ZXIoKG5ldHdvcmspID0+IGlzTWFpbm5ldChuZXR3b3JrKSAmJiBuZXR3b3JrICE9PSBuZXR3b3Jrcy5iaXRjb2luc3YpXG4gICAgICAuZmxhdE1hcCgobmV0d29yaykgPT5cbiAgICAgICAgc2lnblN0YWdlcy5mbGF0TWFwKChzaWduU3RhZ2UpID0+XG4gICAgICAgICAgdHhGb3JtYXRzLmZsYXRNYXAoKHR4Rm9ybWF0KSA9PiBBY2lkVGVzdC53aXRoQ29uZmlnKG5ldHdvcmssIHNpZ25TdGFnZSwgdHhGb3JtYXQsIHN1aXRlQ29uZmlnKSlcbiAgICAgICAgKVxuICAgICAgKTtcbiAgfVxufVxuXG4vKipcbiAqIFZlcmlmaWVzIHNpZ25hdHVyZXMgb2YgZnVsbHkgc2lnbmVkIHR4ICh3aXRoIHRhcHJvb3Qga2V5IHBhdGggc3VwcG9ydCkuXG4gKiBOT1RFOiB0YXByb290IGtleSBwYXRoIHR4IGNhbiBvbmx5IGJlIGJ1aWx0IGFuZCBzaWduZWQgd2l0aCBQU0JULlxuICovXG5leHBvcnQgZnVuY3Rpb24gdmVyaWZ5RnVsbHlTaWduZWRTaWduYXR1cmVzKFxuICB0eDogVXR4b1RyYW5zYWN0aW9uPGJpZ2ludD4sXG4gIHVuc3BlbnRzOiBVbnNwZW50PGJpZ2ludD5bXSxcbiAgd2FsbGV0S2V5czogUm9vdFdhbGxldEtleXMsXG4gIHNpZ25lcjogS2V5TmFtZSxcbiAgY29zaWduZXI6IEtleU5hbWVcbik6IGJvb2xlYW4ge1xuICBjb25zdCBwcmV2T3V0cHV0cyA9IHVuc3BlbnRzLm1hcCgodSkgPT4gdG9PdXRwdXQodSwgdHgubmV0d29yaykpO1xuICByZXR1cm4gdW5zcGVudHMuZXZlcnkoKHUsIGluZGV4KSA9PiB7XG4gICAgaWYgKHBhcnNlU2lnbmF0dXJlU2NyaXB0Mk9mMyh0eC5pbnNbaW5kZXhdKS5zY3JpcHRUeXBlID09PSAndGFwcm9vdEtleVBhdGhTcGVuZCcpIHtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IGdldFNpZ25hdHVyZVZlcmlmaWNhdGlvbnModHgsIGluZGV4LCB1LnZhbHVlLCB1bmRlZmluZWQsIHByZXZPdXRwdXRzKTtcbiAgICAgIHJldHVybiByZXN1bHQubGVuZ3RoID09PSAxICYmIHJlc3VsdFswXS5zaWduYXR1cmU7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IHZlcmlmeVNpZ25hdHVyZVdpdGhVbnNwZW50KHR4LCBpbmRleCwgdW5zcGVudHMsIHdhbGxldEtleXMpO1xuICAgICAgaWYgKChzaWduZXIgPT09ICd1c2VyJyAmJiBjb3NpZ25lciA9PT0gJ2JpdGdvJykgfHwgKHNpZ25lciA9PT0gJ2JpdGdvJyAmJiBjb3NpZ25lciA9PT0gJ3VzZXInKSkge1xuICAgICAgICByZXR1cm4gcmVzdWx0WzBdICYmICFyZXN1bHRbMV0gJiYgcmVzdWx0WzJdO1xuICAgICAgfSBlbHNlIGlmICgoc2lnbmVyID09PSAndXNlcicgJiYgY29zaWduZXIgPT09ICdiYWNrdXAnKSB8fCAoc2lnbmVyID09PSAnYmFja3VwJyAmJiBjb3NpZ25lciA9PT0gJ3VzZXInKSkge1xuICAgICAgICByZXR1cm4gcmVzdWx0WzBdICYmIHJlc3VsdFsxXSAmJiAhcmVzdWx0WzJdO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgcmV0dXJuICFyZXN1bHRbMF0gJiYgcmVzdWx0WzFdICYmIHJlc3VsdFsyXTtcbiAgICAgIH1cbiAgICB9XG4gIH0pO1xufVxuIl19