jsii-release
Version:
Release jsii modules to multiple package managers
612 lines (611 loc) • 75.1 kB
JavaScript
"use strict";
var __addDisposableResource = (this && this.__addDisposableResource) || function (env, value, async) {
if (value !== null && value !== void 0) {
if (typeof value !== "object" && typeof value !== "function") throw new TypeError("Object expected.");
var dispose, inner;
if (async) {
if (!Symbol.asyncDispose) throw new TypeError("Symbol.asyncDispose is not defined.");
dispose = value[Symbol.asyncDispose];
}
if (dispose === void 0) {
if (!Symbol.dispose) throw new TypeError("Symbol.dispose is not defined.");
dispose = value[Symbol.dispose];
if (async) inner = dispose;
}
if (typeof dispose !== "function") throw new TypeError("Object not disposable.");
if (inner) dispose = function() { try { inner.call(this); } catch (e) { return Promise.reject(e); } };
env.stack.push({ value: value, dispose: dispose, async: async });
}
else if (async) {
env.stack.push({ async: true });
}
return value;
};
var __disposeResources = (this && this.__disposeResources) || (function (SuppressedError) {
return function (env) {
function fail(e) {
env.error = env.hasError ? new SuppressedError(e, env.error, "An error was suppressed during disposal.") : e;
env.hasError = true;
}
var r, s = 0;
function next() {
while (r = env.stack.pop()) {
try {
if (!r.async && s === 1) return s = 0, env.stack.push(r), Promise.resolve().then(next);
if (r.dispose) {
var result = r.dispose.call(r.value);
if (r.async) return s |= 2, Promise.resolve(result).then(next, function(e) { fail(e); return next(); });
}
else s |= 1;
}
catch (e) {
fail(e);
}
}
if (s === 1) return env.hasError ? Promise.reject(env.error) : Promise.resolve();
if (env.hasError) throw env.error;
}
return next();
};
})(typeof SuppressedError === "function" ? SuppressedError : function (error, suppressed, message) {
var e = new Error(message);
return e.name = "SuppressedError", e.error = error, e.suppressed = suppressed, e;
});
Object.defineProperty(exports, "__esModule", { value: true });
exports.autoCleanDir = autoCleanDir;
/**
* Legacy vs compatibility endpoint
* ==================================
*
* This script talks about legacy vs compatibility endpoint a bunch.
*
* Both implement the Nexus 2 protocol, however:
*
* - The "legacy" endpoint is the Sonatype OSSRH endpoint that was deprecated June 2025.
* - The "compatibility" endpoint is a Nexus 2-compatible endpoint stood up for the new
* publishing service, Central Publishing.
*
* It should be the same protocol but of course there are subtle differences
* between them (in how they signal error when you try to republish an already
* published version, and even in the wire protocol accepted) so we have to
* treat them differently here.
*
* There is also an "official new" protocol for the new publishing mechanism,
* but the Maven plugin they supply for that doesn't support that. In their words:
*
* > it appears that you are using the "two stage"[1] deployment process, which is
* > not yet supported by the new plugin. You are the first publisher to have
* > requested this, so we were not familiar with this specific usecase when
* > building the new plugin
* > [1] <https://github.com/sonatype/nexus-maven-plugins/blob/43a9940b134c3f87ebe4daa82552e844d9c578b8/staging/maven-plugin/WORKFLOWS.md#two-shots>
*/
const fs_1 = require("fs");
const zx_1 = require("zx");
const LEGACY_OSSRH_SERVER_ID = 'ossrh';
const COMPAT_CENTRAL_SERVER_ID = 'central-ossrh';
const NEXUS_MAVEN_STAGING_PLUGIN = 'org.sonatype.plugins:nexus-staging-maven-plugin:1.7.0';
async function main() {
const args = process.argv.slice(2);
const javaRoot = args[0] ?? 'dist/java';
(0, zx_1.cd)(javaRoot);
const poms = await (0, zx_1.glob)('**/*.pom');
if (poms.length === 0) {
throw new SimpleError(`No JARS to publish: no .pom files found under ${process.cwd()}`);
}
(0, zx_1.echo)(`POMs found: ${poms}`);
const sharedOptions = {
username: envVar('MAVEN_USERNAME'),
password: envVar('MAVEN_PASSWORD'),
dryRun: process.env.MAVEN_DRYRUN === 'true',
verbose: process.env.MAVEN_VERBOSE === 'true',
poms,
};
let options;
const serverId = process.env.MAVEN_SERVER_ID ?? LEGACY_OSSRH_SERVER_ID;
switch (serverId) {
case LEGACY_OSSRH_SERVER_ID:
options = {
type: 'legacy-ossrh',
...sharedOptions,
stagingProfileId: envVar('MAVEN_STAGING_PROFILE_ID'),
endpoint: process.env.MAVEN_ENDPOINT,
privateKey: parsePrivateKeyFromEnv(),
};
break;
case COMPAT_CENTRAL_SERVER_ID:
options = {
type: 'compat-ossrh',
...sharedOptions,
// Not required by the new endpoint: can be any value (maybe never was required to begin with?)
stagingProfileId: 'publib',
endpoint: process.env.MAVEN_ENDPOINT,
privateKey: parsePrivateKeyFromEnv(),
};
break;
default:
// We haven't implemented signing for this, so fail loudly if people try to use it anyway
if (process.env.MAVEN_GPG_PRIVATE_KEY || process.env.MAVEN_GPG_PRIVATE_KEY_FILE) {
throw new SimpleError('MAVEN_GPG_PRIVATE_KEY[_FILE] is only supported for OSSRH publishing');
}
options = {
type: 'custom-nexus',
...sharedOptions,
serverId,
repositoryUrl: envVar('MAVEN_REPOSITORY_URL'),
};
break;
}
await mavenPublish(options);
(0, zx_1.echo)('~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~');
(0, zx_1.echo)('✅ All Done!');
}
function parsePrivateKeyFromEnv() {
if (process.env.MAVEN_GPG_PRIVATE_KEY_FILE) {
return { type: 'file', fileName: process.env.MAVEN_GPG_PRIVATE_KEY_FILE, passPhrase: envVar('MAVEN_GPG_PRIVATE_KEY_PASSPHRASE') };
}
if (process.env.MAVEN_GPG_PRIVATE_KEY) {
return { type: 'material', keyMaterial: process.env.MAVEN_GPG_PRIVATE_KEY, passPhrase: envVar('MAVEN_GPG_PRIVATE_KEY_PASSPHRASE') };
}
throw new SimpleError('MAVEN_GPG_PRIVATE_KEY[_FILE] is required');
}
//----------------------------------------------------------------------
// Publishing functions
//
async function mavenPublish(options) {
const env_1 = { stack: [], error: void 0, hasError: false };
try {
const workDir = __addDisposableResource(env_1, autoCleanDir(), true);
const maven = new Maven(workDir.dir, options.verbose, options.dryRun);
const x = options.type;
switch (x) {
case 'legacy-ossrh':
await deployLegacyOssrh(maven, options);
break;
case 'compat-ossrh':
await deployCompatOssrh(maven, options);
break;
case 'custom-nexus':
await deployCustomNexus(maven, options);
break;
default:
assertNever(x);
}
}
catch (e_1) {
env_1.error = e_1;
env_1.hasError = true;
}
finally {
const result_1 = __disposeResources(env_1);
if (result_1)
await result_1;
}
}
/**
* Deploy to the legacy OSSRH Nexus server
*/
async function deployLegacyOssrh(maven, options) {
(0, zx_1.echo)('📦 Publishing to Maven Central');
const defaultEndpoint = 'https://oss.sonatype.org';
const staged = await deployStagedRepository(maven, {
...options,
endpoint: options.endpoint ?? defaultEndpoint,
republishWill400: false,
});
if (staged.type !== 'success') {
return;
}
/*
const released = await releaseRepo({
serverUrl: options.endpoint ?? defaultEndpoint,
username: options.username,
password: options.password,
repositoryId: staged.repositoryId,
});
if (released) {
return;
}
*/
// Send a remote release command to the repository
const releaseOutput = await maven.exec(`${NEXUS_MAVEN_STAGING_PLUGIN}:rc-release`, {
properties: {
nexusUrl: options.endpoint ?? defaultEndpoint,
serverId: staged.serverId,
stagingRepositoryId: staged.repositoryId,
},
nothrow: true,
});
if (!releaseOutput) {
(0, zx_1.echo)('🏜️ Stopped here because of dry-run');
return;
}
if (releaseOutput.exitCode !== 0) {
// If release failed, check if this was caused because we are trying to publish
// the same version again, which is not an error. The magic string "does not
// allow updating artifact" for a ".pom" file indicates that we are trying to
// override an existing version. Otherwise, fail!
const looksLikeDuplicatePublish = releaseOutput.lines()
.filter(l => l.includes('does not allow updating artifact'))
.filter(l => l.includes('.pom'))
.length > 0;
if (!looksLikeDuplicatePublish) {
throw new SimpleError('Release failed');
}
(0, zx_1.echo)('⚠️ Artifact already published. Skipping');
}
}
/**
* Deploy to the compat OSSRH Nexus server
*/
async function deployCompatOssrh(maven, options) {
(0, zx_1.echo)('📦 Publishing to Maven Central (Compat endpoint)');
const defaultEndpoint = 'https://ossrh-staging-api.central.sonatype.com/';
const endpoint = options.endpoint ?? defaultEndpoint;
const staged = await deployStagedRepository(maven, {
...options,
endpoint,
republishWill400: true,
});
if (staged.type !== 'success') {
return;
}
const released = await releaseRepo({
serverUrl: endpoint,
username: options.username,
password: options.password,
repositoryId: staged.repositoryId,
});
if (released === 'duplicate-version') {
(0, zx_1.echo)('⚠️ Version(s) already published. Skipping');
}
}
/**
* Deploy to a custom Nexus server
*/
async function deployCustomNexus(maven, options) {
(0, zx_1.echo)(`📦 Publishing to ${options.serverId}`);
await maven.writeSettingsFile(options.serverId, false);
for (const pom of options.poms) {
const deployOutput = await maven.exec('deploy:deploy-file', {
properties: {
url: options.repositoryUrl,
repositoryId: options.serverId,
pomFile: pom,
...jarsFromPom(pom),
},
nothrow: true,
});
if (deployOutput?.exitCode && deployOutput.exitCode > 0) {
if (deployOutput.stdout.includes('409 Conflict')) {
(0, zx_1.echo)('⚠️ Artifact already published. Skipping');
}
else {
throw new SimpleError('Release failed');
}
}
}
}
/**
* Create the staging repository. This is the same between the legacy and compat endpoints.
*
* `republishWill400`: in the compatibility endpoint, staging a version that has already
* been published will lead to a `400 Bad Request`, with no further info. We needed to have
* run the Maven command with `--debug` logging to see the actual error message in the output.
*
* In the legacy endpoint, staging takes multiple minutes, and Maven will poll
* every 3s and print the output. We therefore don't use verbose mode to avoid
* stdout spam. In the legacy endpoint, duplicate version publishing gets
* reported during the 'release' step anyway, not during staging.
*/
async function deployStagedRepository(maven, options) {
const serverId = 'ossrh';
await maven.writeSettingsFile(serverId, true);
// First -- sign artifacts
const signedDir = zx_1.path.join(maven.workDir, 'signed');
await fs_1.promises.mkdir(signedDir, { recursive: true });
await signJars(maven, options.poms, signedDir, serverId, options.privateKey);
(0, zx_1.echo)('~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~');
(0, zx_1.echo)(' Deploying and closing repository...');
(0, zx_1.echo)('~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~');
const nexusMavenStaging = 'org.sonatype.plugins:nexus-staging-maven-plugin:1.7.0';
const stageOutput = await maven.exec(`${nexusMavenStaging}:deploy-staged-repository`, {
properties: {
repositoryDirectory: signedDir,
nexusUrl: options.endpoint,
serverId,
stagingProgressTimeoutMinutes: '30',
stagingProfileId: options.stagingProfileId,
},
nothrow: true,
verbose: options.republishWill400,
});
// FIXME: New staging API throws an error here on duplicate versions
if (stageOutput === undefined) {
(0, zx_1.echo)('🏜️ Stopped here because of dry-run');
return { type: 'dry-run' };
}
if (options.republishWill400 && stageOutput.text().match(/Component with package url.*already exists/)) {
// We've seen this fail with the above error message on the OSSRH compatibility API when trying to republish
// an already-published version.
//
// This is potentially a problem if there are multiple versions in the source directory at once, because nothing
// will be published if there's one package in there that has already been published previously. But c'est la vie.
(0, zx_1.echo)('⚠️ Version(s) already published. Skipping');
return { type: 'already-published' };
}
else if (stageOutput.exitCode && stageOutput.exitCode > 0) {
throw stageOutput;
}
// Grep the repository ID out of the printed output
// [INFO] * Closing staging repository with ID "XXXXXXXXXX".
const m = stageOutput.stdout.match(/Closing staging repository with ID "([^"]+)"/);
if (!m) {
throw new SimpleError('Unable to extract repository ID from deploy-staged-repository output. This means it failed to close or there was an unexpected problem. At any rate, we can\'t release it. Sorry.');
}
const repositoryId = m[1];
return { type: 'success', repositoryId, serverId };
}
/**
* Send a custom "release" command to the new Sonatype backwards compatibility endpoint
*
* When using the `nexus-staging-maven-plugin`'s `release` or `rc-release` commands, we get the following
* error:
*
* Sending:
*
* ```
* <stagingActionRequest><data><stagedRepositoryIds class="java.util.Arrays$ArrayList"><a class="string-array"><string>io.github.rix0rrr--63edbcbe-f058-44eb-85f4-fd9dce1aef40</string></a></stagedRepositoryIds><description>unknown</description><autoDropAfterRelease>true</autoDropAfterRelease></data></stagingActionRequest>
* ```
*
* Error:
*
* ```
* Failed to process request: Got unexpected XML element when reading stagedRepositoryIds: Got unexpected element StartElement(a, {"": "", "xml": "http://www.w3.org/XML/1998/namespace", "xmlns": "http://www.w3.org/2000/xmlns/"}, [class -> string-array]), expected one of: string
* ```
*
* Doing a manual slightly modified version of the above request succeeds, so we'll just proceed with doing
* that.
*
* The endpoint also supports JSON, which we prefer over XML.
*
* This endpoint will never return an error, even if the publish didn't happen because
* of duplicate version publishing (not for the legacy, nor for the compatibility endpoint).
*
* @see https://central.sonatype.org/publish/publish-portal-ossrh-staging-api/
* @see https://support.sonatype.com/hc/en-us/articles/213465448-Automatically-dropping-old-staging-repositories
*/
async function releaseRepo(options) {
(0, zx_1.echo)(`🚀 Releasing repository ${options.repositoryId} at ${options.serverUrl}`);
const url = new URL('service/local/staging/bulk/promote', options.serverUrl);
const response = await fetch(url, {
method: 'POST',
body: JSON.stringify({
data: {
stagedRepositoryIds: [options.repositoryId],
description: options.description ?? 'Deployment',
autoDropAfterRelease: true,
},
}),
headers: {
'Content-Type': 'application/json',
'Accept': 'application/json',
'Authorization': `Basic ${btoa(`${options.username}:${options.password}`)}`,
},
});
const body = await response.text();
// It's possible that (due to some replication latency?) the initial "stage" succeeds,
// but the "release" fails, if we try to publish an already-published version again.
//
// Unfortunately there is no great error code to detect this right now, but I've observed
// this error when it happened.
// I have a support request out to Sonatype for this, we'll see if this changes over time.
if (response.status === 400 && body === 'Failed to process request: service error') {
return 'duplicate-version';
}
if (!response.ok) {
throw new Error(`HTTP ${response.status} ${response.statusText}: ${body}`);
}
console.log(body);
return 'ok';
}
/**
* Sign and stage our artifacts into a local directory
*/
async function signJars(maven, poms, targetDir, serverId, key) {
const env_2 = { stack: [], error: void 0, hasError: false };
try {
const gpg = __addDisposableResource(env_2, await importGpgKey(key), true);
// on a mac, --pinentry-mode to "loopback" are required and I couldn't find a
// way to do so via -Dgpg.gpgArguments or the settings file, so here we are.
let gpgBin = 'gpg';
if (zx_1.os.platform() === 'darwin') {
gpgBin = zx_1.path.join(maven.workDir, 'publib-gpg.sh');
await fs_1.promises.writeFile(gpgBin, [
'#!/bin/bash',
'exec gpg --pinentry-mode loopback "\$@"',
].join('\n'), 'utf-8');
await (0, zx_1.$) `chmod +x ${gpgBin}`;
}
for (const pom of poms) {
await maven.exec('gpg:sign-and-deploy-file', {
properties: {
'url': `file://${targetDir}`,
'repositoryId': serverId, // Most likely not necessary
'gpg.homedir': gpg.home,
'gpg.keyname': gpg.keyId,
'gpg.executable': gpgBin,
'pomFile': pom,
...jarsFromPom(pom),
},
});
}
}
catch (e_2) {
env_2.error = e_2;
env_2.hasError = true;
}
finally {
const result_2 = __disposeResources(env_2);
if (result_2)
await result_2;
}
}
/**
* Based on the path of a .pom file, return the paths of the corresponding jars
*/
function jarsFromPom(pom) {
return {
file: pom.replace(/\.pom$/, '.jar'),
sources: pom.replace(/\.pom$/, '-sources.jar'),
javadoc: pom.replace(/\.pom$/, '-javadoc.jar'),
};
}
/**
* Create a temporary directory and import the GPG key material into a new keychain there
*/
async function importGpgKey(key) {
// GnuPG will occasionally bail out with "gpg: <whatever> failed: Inappropriate ioctl for device", the following attempts to fix
const gpgHome = autoCleanDir();
try {
// In CI environments there will be no tty, and we don't want this to stop the script.
// The variable will be populated with a nonsensical string ("not a tty") but that doesn't seem to matter.
const tty = (await (0, zx_1.$)({ stdio: ['inherit', 'pipe', 'pipe'], nothrow: true }) `tty`).stdout;
let privateKeyFile;
const type = key.type;
switch (type) {
case 'file':
privateKeyFile = key.fileName;
break;
case 'material':
privateKeyFile = zx_1.path.join(gpgHome.dir, 'private.pem');
await (0, zx_1.$) `echo -e ${key.keyMaterial} > ${privateKeyFile}`;
break;
default:
assertNever(type);
}
const env = {
GNUPGHOME: gpgHome.dir,
GPG_TTY: tty,
};
const $$ = (0, zx_1.$)({ env: { ...process.env, ...env } });
await $$ `gpg --allow-secret-key-import --batch --yes --no-tty --import ${privateKeyFile}`;
const gpgKeyId = (await $$ `gpg --list-keys --with-colons | grep pub | cut -d: -f5`).stdout.trim();
(0, zx_1.echo)(`gpg_key_id=${gpgKeyId}`);
return {
keyId: gpgKeyId,
env,
home: gpgHome.dir,
[Symbol.asyncDispose]: async () => gpgHome[Symbol.asyncDispose](),
};
}
catch (e) {
await gpgHome[Symbol.asyncDispose]();
throw e;
}
}
class Maven {
constructor(workDir, verbose, dryRun) {
this.workDir = workDir;
this.verbose = verbose;
this.dryRun = dryRun;
this.settingsFile = zx_1.path.join(workDir, 'mvn-settings.xml');
}
/**
* Create a settings.xml file with the user+password for maven
*/
async writeSettingsFile(serverId, signedArtifacts) {
const lines = [];
lines.push('<?xml version="1.0" encoding="UTF-8" ?>', '<settings xmlns="http://maven.apache.org/SETTINGS/1.0.0"', ' xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"', ' xsi:schemaLocation="http://maven.apache.org/SETTINGS/1.0.0', ' http://maven.apache.org/xsd/settings-1.0.0.xsd">', ' <servers>', ' <server>', ` <id>${serverId}</id>`,
// Maven will read these, surrounding code has already made sure they are set
' <username>${env.MAVEN_USERNAME}</username>', ' <password>${env.MAVEN_PASSWORD}</password>', ' </server>');
if (signedArtifacts) {
lines.push(' <server>', ' <id>gpg.passphrase</id>',
// Maven will read these, surrounding code has already made sure they are set
' <passphrase>${env.MAVEN_GPG_PRIVATE_KEY_PASSPHRASE}</passphrase>', ' </server>');
}
lines.push(' </servers>', '</settings>');
await fs_1.promises.writeFile(this.settingsFile, lines.join('\n'), 'utf-8');
}
async exec(mojo, options) {
const args = [
`--settings=${this.settingsFile}`,
'--batch-mode',
...(options.verbose ?? this.verbose ? ['-X'] : []),
mojo,
...Object.entries(options.properties).map(([k, v]) => `-D${k}=${v}`),
];
if (this.dryRun) {
(0, zx_1.echo)(`[DRY-RUN] mvn ${args.map(zx_1.quote).join(' ')}`);
return undefined;
}
const env = {
...process.env,
JDK_JAVA_OPTIONS: [
// If we don't add this, we'll get an error like the following during the nexus-staging-maven-plugin:rc-release mojo.
// Don't know where this is coming from all of a sudden.
//
// [ERROR] message : No converter available
// [ERROR] type : java.util.Arrays$ArrayList
// [ERROR] converter : com.thoughtworks.xstream.converters.reflection.ReflectionConverter
// [ERROR] message[1] : Unable to make field protected transient int java.util.AbstractList.modCount accessible: module java.base does not "opens java.util" to unnamed module @7c8d5312
'--add-opens=java.base/java.util=ALL-UNNAMED',
'--add-opens=java.base/java.lang=ALL-UNNAMED',
'--add-opens=java.base/java.lang.invoke=ALL-UNNAMED',
].join(' '),
};
return (0, zx_1.$)({ verbose: true, nothrow: options.nothrow, env }) `mvn ${args}`;
}
}
/**
* A expected error that doesn't need a stack trace
*/
class SimpleError extends Error {
}
/**
* Require an environment variable
*/
function envVar(name) {
const ret = process.env[name];
if (!ret) {
throw new SimpleError(`${name} is required`);
}
return ret;
}
/**
* A temporary directory that cleans when it goes out of scope
*
* Use with `using`. Could have been async but it's depending
* on an already-sync API, so why not sync?
*/
function autoCleanDir() {
const dir = (0, zx_1.tmpdir)();
return {
dir,
[Symbol.asyncDispose]: async () => {
await fs_1.promises.rm(dir, { force: true, recursive: true });
},
};
}
function assertNever(value) {
throw new Error('Unexpected value: ' + value);
}
// A 'zx' primer
//
// - By default: stderr is printed to the terminal if it is captured.
// - { verbose: true }: print the command, stdout and stderr to the terminal
// - { quiet: true }: print neither stdout nor stderr
//
// .text(): return everything that's captured (stderr and stdout together if stderr is captured)
main().catch(e => {
if (e instanceof zx_1.ProcessOutput) {
(0, zx_1.echo)(`❌ Subprocess failed with exit code ${e.exitCode}`);
}
else if (e instanceof SimpleError) {
(0, zx_1.echo)('❌', e.message);
}
else {
console.error(e);
}
process.exitCode = 1;
});
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGliLW1hdmVuLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Jpbi9wdWJsaWItbWF2ZW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBNm5CQSxvQ0FRQztBQXJvQkQ7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0F5Qkc7QUFDSCwyQkFBb0M7QUFDcEMsMkJBQStFO0FBRS9FLE1BQU0sc0JBQXNCLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sd0JBQXdCLEdBQUcsZUFBZSxDQUFDO0FBRWpELE1BQU0sMEJBQTBCLEdBQUcsdURBQXVELENBQUM7QUFFM0YsS0FBSyxVQUFVLElBQUk7SUFDakIsTUFBTSxJQUFJLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDbkMsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLFdBQVcsQ0FBQztJQUV4QyxJQUFBLE9BQUUsRUFBQyxRQUFRLENBQUMsQ0FBQztJQUViLE1BQU0sSUFBSSxHQUFHLE1BQU0sSUFBQSxTQUFJLEVBQUMsVUFBVSxDQUFDLENBQUM7SUFDcEMsSUFBSSxJQUFJLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO1FBQ3RCLE1BQU0sSUFBSSxXQUFXLENBQUMsaURBQWlELE9BQU8sQ0FBQyxHQUFHLEVBQUUsRUFBRSxDQUFDLENBQUM7SUFDMUYsQ0FBQztJQUVELElBQUEsU0FBSSxFQUFDLGVBQWUsSUFBSSxFQUFFLENBQUMsQ0FBQztJQUU1QixNQUFNLGFBQWEsR0FBeUI7UUFDMUMsUUFBUSxFQUFFLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQztRQUNsQyxRQUFRLEVBQUUsTUFBTSxDQUFDLGdCQUFnQixDQUFDO1FBQ2xDLE1BQU0sRUFBRSxPQUFPLENBQUMsR0FBRyxDQUFDLFlBQVksS0FBSyxN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RSxJQUFJLEVBQUUsU0FBUyxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7UUFDckQsQ0FBQztLQUNGLENBQUM7QUFDSixDQUFDO0FBRUQsU0FBUyxXQUFXLENBQUMsS0FBWTtJQUMvQixNQUFNLElBQUksS0FBSyxDQUFDLG9CQUFvQixHQUFHLEtBQUssQ0FBQyxDQUFDO0FBQ2hELENBQUM7QUFFRCxnQkFBZ0I7QUFDaEIsRUFBRTtBQUNGLHFFQUFxRTtBQUNyRSw0RUFBNEU7QUFDNUUscURBQXFEO0FBQ3JELEVBQUU7QUFDRixnR0FBZ0c7QUFFaEcsSUFBSSxFQUFFLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxFQUFFO0lBQ2YsSUFBSSxDQUFDLFlBQVksa0JBQWEsRUFBRSxDQUFDO1FBQy9CLElBQUEsU0FBSSxFQUFDLHNDQUFzQyxDQUFDLENBQUMsUUFBUSxFQUFFLENBQUMsQ0FBQztJQUMzRCxDQUFDO1NBQU0sSUFBSSxDQUFDLFlBQVksV0FBVyxFQUFFLENBQUM7UUFDcEMsSUFBQSxTQUFJLEVBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUN2QixDQUFDO1NBQU0sQ0FBQztRQUNOLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDbkIsQ0FBQztJQUNELE9BQU8sQ0FBQyxRQUFRLEdBQUcsQ0FBQyxDQUFDO0FBQ3ZCLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBMZWdhY3kgdnMgY29tcGF0aWJpbGl0eSBlbmRwb2ludFxuICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuICpcbiAqIFRoaXMgc2NyaXB0IHRhbGtzIGFib3V0IGxlZ2FjeSB2cyBjb21wYXRpYmlsaXR5IGVuZHBvaW50IGEgYnVuY2guXG4gKlxuICogQm90aCBpbXBsZW1lbnQgdGhlIE5leHVzIDIgcHJvdG9jb2wsIGhvd2V2ZXI6XG4gKlxuICogLSBUaGUgXCJsZWdhY3lcIiBlbmRwb2ludCBpcyB0aGUgU29uYXR5cGUgT1NTUkggZW5kcG9pbnQgdGhhdCB3YXMgZGVwcmVjYXRlZCBKdW5lIDIwMjUuXG4gKiAtIFRoZSBcImNvbXBhdGliaWxpdHlcIiBlbmRwb2ludCBpcyBhIE5leHVzIDItY29tcGF0aWJsZSBlbmRwb2ludCBzdG9vZCB1cCBmb3IgdGhlIG5ld1xuICogICBwdWJsaXNoaW5nIHNlcnZpY2UsIENlbnRyYWwgUHVibGlzaGluZy5cbiAqXG4gKiBJdCBzaG91bGQgYmUgdGhlIHNhbWUgcHJvdG9jb2wgYnV0IG9mIGNvdXJzZSB0aGVyZSBhcmUgc3VidGxlIGRpZmZlcmVuY2VzXG4gKiBiZXR3ZWVuIHRoZW0gKGluIGhvdyB0aGV5IHNpZ25hbCBlcnJvciB3aGVuIHlvdSB0cnkgdG8gcmVwdWJsaXNoIGFuIGFscmVhZHlcbiAqIHB1Ymxpc2hlZCB2ZXJzaW9uLCBhbmQgZXZlbiBpbiB0aGUgd2lyZSBwcm90b2NvbCBhY2NlcHRlZCkgc28gd2UgaGF2ZSB0b1xuICogdHJlYXQgdGhlbSBkaWZmZXJlbnRseSBoZXJlLlxuICpcbiAqIFRoZXJlIGlzIGFsc28gYW4gXCJvZmZpY2lhbCBuZXdcIiBwcm90b2NvbCBmb3IgdGhlIG5ldyBwdWJsaXNoaW5nIG1lY2hhbmlzbSxcbiAqIGJ1dCB0aGUgTWF2ZW4gcGx1Z2luIHRoZXkgc3VwcGx5IGZvciB0aGF0IGRvZXNuJ3Qgc3VwcG9ydCB0aGF0LiBJbiB0aGVpciB3b3JkczpcbiAqXG4gKiA+IGl0IGFwcGVhcnMgdGhhdCB5b3UgYXJlIHVzaW5nIHRoZSBcInR3byBzdGFnZVwiWzFdIGRlcGxveW1lbnQgcHJvY2Vzcywgd2hpY2ggaXNcbiAqID4gbm90IHlldCBzdXBwb3J0ZWQgYnkgdGhlIG5ldyBwbHVnaW4uIFlvdSBhcmUgdGhlIGZpcnN0IHB1Ymxpc2hlciB0byBoYXZlXG4gKiA+IHJlcXVlc3RlZCB0aGlzLCBzbyB3ZSB3ZXJlIG5vdCBmYW1pbGlhciB3aXRoIHRoaXMgc3BlY2lmaWMgdXNlY2FzZSB3aGVuXG4gKiA+IGJ1aWxkaW5nIHRoZSBuZXcgcGx1Z2luXG4gKiA+IFsxXSA8aHR0cHM6Ly9naXRodWIuY29tL3NvbmF0eXBlL25leHVzLW1hdmVuLXBsdWdpbnMvYmxvYi80M2E5OTQwYjEzNGMzZjg3ZWJlNGRhYTgyNTUyZTg0NGQ5YzU3OGI4L3N0YWdpbmcvbWF2ZW4tcGx1Z2luL1dPUktGTE9XUy5tZCN0d28tc2hvdHM+XG4gKi9cbmltcG9ydCB7IHByb21pc2VzIGFzIGZzIH0gZnJvbSAnZnMnO1xuaW1wb3J0IHsgJCwgY2QsIGVjaG8sIGdsb2IsIG9zLCBwYXRoLCBQcm9jZXNzT3V0cHV0LCBxdW90ZSwgdG1wZGlyIH0gZnJvbSAnengnO1xuXG5jb25zdCBMRUdBQ1lfT1NTUkhfU0VSVkVSX0lEID0gJ29zc3JoJztcbmNvbnN0IENPTVBBVF9DRU5UUkFMX1NFUlZFUl9JRCA9ICdjZW50cmFsLW9zc3JoJztcblxuY29uc3QgTkVYVVNfTUFWRU5fU1RBR0lOR19QTFVHSU4gPSAnb3JnLnNvbmF0eXBlLnBsdWdpbnM6bmV4dXMtc3RhZ2luZy1tYXZlbi1wbHVnaW46MS43LjAnO1xuXG5hc3luYyBmdW5jdGlvbiBtYWluKCkge1xuICBjb25zdCBhcmdzID0gcHJvY2Vzcy5hcmd2LnNsaWNlKDIpO1xuICBjb25zdCBqYXZhUm9vdCA9IGFyZ3NbMF0gPz8gJ2Rpc3QvamF2YSc7XG5cbiAgY2QoamF2YVJvb3QpO1xuXG4gIGNvbnN0IHBvbXMgPSBhd2FpdCBnbG9iKCcqKi8qLnBvbScpO1xuICBpZiAocG9tcy5sZW5ndGggPT09IDApIHtcbiAgICB0aHJvdyBuZXcgU2ltcGxlRXJyb3IoYE5vIEpBUlMgdG8gcHVibGlzaDogbm8gLnBvbSBmaWxlcyBmb3VuZCB1bmRlciAke3Byb2Nlc3MuY3dkKCl9YCk7XG4gIH1cblxuICBlY2hvKGBQT01zIGZvdW5kOiAke3BvbXN9YCk7XG5cbiAgY29uc3Qgc2hhcmVkT3B0aW9uczogU2hhcmVkUHVibGlzaE9wdGlvbnMgPSB7XG4gICAgdXNlcm5hbWU6IGVudlZhcignTUFWRU5fVVNFUk5BTUUnKSxcbiAgICBwYXNzd29yZDogZW52VmFyKCdNQVZFTl9QQVNTV09SRCcpLFxuICAgIGRyeVJ1bjogcHJvY2Vzcy5lbnYuTUFWRU5fRFJZUlVOID09PSAndHJ1ZScsXG4gICAgdmVyYm9zZTogcHJvY2Vzcy5lbnYuTUFWRU5fVkVSQk9TRSA9PT0gJ3RydWUnLFxuICAgIHBvbXMsXG4gIH07XG5cbiAgbGV0IG9wdGlvbnM6IE1hdmVuUHVibGlzaE9wdGlvbnM7XG4gIGNvbnN0IHNlcnZlcklkID0gcHJvY2Vzcy5lbnYuTUFWRU5fU0VSVkVSX0lEID8/IExFR0FDWV9PU1NSSF9TRVJWRVJfSUQ7XG5cbiAgc3dpdGNoIChzZXJ2ZXJJZCkge1xuICAgIGNhc2UgTEVHQUNZX09TU1JIX1NFUlZFUl9JRDpcbiAgICAgIG9wdGlvbnMgPSB7XG4gICAgICAgIHR5cGU6ICdsZWdhY3ktb3NzcmgnLFxuICAgICAgICAuLi5zaGFyZWRPcHRpb25zLFxuICAgICAgICBzdGFnaW5nUHJvZmlsZUlkOiBlbnZWYXIoJ01BVkVOX1NUQUdJTkdfUFJPRklMRV9JRCcpLFxuICAgICAgICBlbmRwb2ludDogcHJvY2Vzcy5lbnYuTUFWRU5fRU5EUE9JTlQsXG4gICAgICAgIHByaXZhdGVLZXk6IHBhcnNlUHJpdmF0ZUtleUZyb21FbnYoKSxcbiAgICAgIH07XG4gICAgICBicmVhaztcbiAgICBjYXNlIENPTVBBVF9DRU5UUkFMX1NFUlZFUl9JRDpcbiAgICAgIG9wdGlvbnMgPSB7XG4gICAgICAgIHR5cGU6ICdjb21wYXQtb3NzcmgnLFxuICAgICAgICAuLi5zaGFyZWRPcHRpb25zLFxuICAgICAgICAvLyBOb3QgcmVxdWlyZWQgYnkgdGhlIG5ldyBlbmRwb2ludDogY2FuIGJlIGFueSB2YWx1ZSAobWF5YmUgbmV2ZXIgd2FzIHJlcXVpcmVkIHRvIGJlZ2luIHdpdGg/KVxuICAgICAgICBzdGFnaW5nUHJvZmlsZUlkOiAncHVibGliJyxcbiAgICAgICAgZW5kcG9pbnQ6IHByb2Nlc3MuZW52Lk1BVkVOX0VORFBPSU5ULFxuICAgICAgICBwcml2YXRlS2V5OiBwYXJzZVByaXZhdGVLZXlGcm9tRW52KCksXG4gICAgICB9O1xuICAgICAgYnJlYWs7XG4gICAgZGVmYXVsdDpcbiAgICAgIC8vIFdlIGhhdmVuJ3QgaW1wbGVtZW50ZWQgc2lnbmluZyBmb3IgdGhpcywgc28gZmFpbCBsb3VkbHkgaWYgcGVvcGxlIHRyeSB0byB1c2UgaXQgYW55d2F5XG4gICAgICBpZiAocHJvY2Vzcy5lbnYuTUFWRU5fR1BHX1BSSVZBVEVfS0VZIHx8IHByb2Nlc3MuZW52Lk1BVkVOX0dQR19QUklWQVRFX0tFWV9GSUxFKSB7XG4gICAgICAgIHRocm93IG5ldyBTaW1wbGVFcnJvcignTUFWRU5fR1BHX1BSSVZBVEVfS0VZW19GSUxFXSBpcyBvbmx5IHN1cHBvcnRlZCBmb3IgT1NTUkggcHVibGlzaGluZycpO1xuICAgICAgfVxuXG4gICAgICBvcHRpb25zID0ge1xuICAgICAgICB0eXBlOiAnY3VzdG9tLW5leHVzJyxcbiAgICAgICAgLi4uc2hhcmVkT3B0aW9ucyxcbiAgICAgICAgc2VydmVySWQsXG4gICAgICAgIHJlcG9zaXRvcnlVcmw6IGVudlZhcignTUFWRU5fUkVQT1NJVE9SWV9VUkwnKSxcbiAgICAgIH07XG4gICAgICBicmVhaztcbiAgfVxuXG4gIGF3YWl0IG1hdmVuUHVibGlzaChvcHRpb25zKTtcblxuICBlY2hvKCd+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+Jyk7XG4gIGVjaG8oJ+KchSBBbGwgRG9uZSEnKTtcbn1cblxuZnVuY3Rpb24gcGFyc2VQcml2YXRlS2V5RnJvbUVudigpOiBQcml2YXRlS2V5IHtcbiAgaWYgKHByb2Nlc3MuZW52Lk1BVkVOX0dQR19QUklWQVRFX0tFWV9GSUxFKSB7XG4gICAgcmV0dXJuIHsgdHlwZTogJ2ZpbGUnLCBmaWxlTmFtZTogcHJvY2Vzcy5lbnYuTUFWRU5fR1BHX1BSSVZBVEVfS0VZX0ZJTEUsIHBhc3NQaHJhc2U6IGVudlZhcignTUFWRU5fR1BHX1BSSVZBVEVfS0VZX1BBU1NQSFJBU0UnKSB9O1xuICB9XG4gIGlmIChwcm9jZXNzLmVudi5NQVZFTl9HUEdfUFJJVkFURV9LRVkpIHtcbiAgICByZXR1cm4geyB0eXBlOiAnbWF0ZXJpYWwnLCBrZXlNYXRlcmlhbDogcHJvY2Vzcy5lbnYuTUFWRU5fR1BHX1BSSVZBVEVfS0VZLCBwYXNzUGhyYXNlOiBlbnZWYXIoJ01BVkVOX0dQR19QUklWQVRFX0tFWV9QQVNTUEhSQVNFJykgfTtcbiAgfVxuICB0aHJvdyBuZXcgU2ltcGxlRXJyb3IoJ01BVkVOX0dQR19QUklWQVRFX0tFWVtfRklMRV0gaXMgcmVxdWlyZWQnKTtcbn1cblxuLy8tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4vLyAgTW9kZWxpbmcgYWxsIHRoZSBwYXJhbWV0ZXJzIHNvIHRoYXQgd2UgY2FuIHB1bGwgdGhlIGVudmlyb25tZW50IHBhcnNpbmcgZm9yd2FyZFxuLy9cblxuaW50ZXJmYWNlIFNoYXJlZFB1Ymxpc2hPcHRpb25zIHtcbiAgcmVhZG9ubHkgdXNlcm5hbWU6IHN0cmluZztcbiAgcmVhZG9ubHkgcGFzc3dvcmQ6IHN0cmluZztcbiAgcmVhZG9ubHkgZHJ5UnVuOiBib29sZWFuO1xuICByZWFkb25seSB2ZXJib3NlOiBib29sZWFuO1xuICByZWFkb25seSBwb21zOiBzdHJpbmdbXTtcbn1cblxuaW50ZXJmYWNlIExlZ2FjeU9zc3JoUHVibGlzaE9wdGlvbnMgZXh0ZW5kcyBTaGFyZWRQdWJsaXNoT3B0aW9ucyB7XG4gIHR5cGU6ICdsZWdhY3ktb3NzcmgnO1xuICByZWFkb25seSBwcml2YXRlS2V5OiBQcml2YXRlS2V5O1xuICByZWFkb25seSBlbmRwb2ludD86IHN0cmluZztcbiAgcmVhZG9ubHkgc3RhZ2luZ1Byb2ZpbGVJZDogc3RyaW5nO1xufVxuXG5pbnRlcmZhY2UgQ29tcGF0T3NzcmhQdWJsaXNoT3B0aW9ucyBleHRlbmRzIFNoYXJlZFB1Ymxpc2hPcHRpb25zIHtcbiAgdHlwZTogJ2NvbXBhdC1vc3NyaCc7XG4gIHJlYWRvbmx5IHByaXZhdGVLZXk6IFByaXZhdGVLZXk7XG4gIHJlYWRvbmx5IGVuZHBvaW50Pzogc3Rya