text-to-map
Version:
Text To Map usiluje o lepší, strojově zpracovatelné využití částí vyhlášek s výčtem ulic a dalších lokací. Jde o rozšiřitelnou sadu konceptů a nástrojů, které zajistí hladký převod výčtu ulic a jejich rozsahů v lidsky srozumitelném jazyce do strojově zpra
191 lines • 29 kB
JavaScript
import { configDotenv } from "dotenv";
import knex from "knex";
import { dirname, join } from "path";
import { fileURLToPath } from "url";
const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);
export var SupportedDbType;
(function (SupportedDbType) {
SupportedDbType["sqlite"] = "sqlite";
SupportedDbType["postgres"] = "postgres";
SupportedDbType["mysql"] = "mysql";
})(SupportedDbType || (SupportedDbType = {}));
const mysqlConnectionStringPattern = /^([^:]+):(\d+):([^:]+):([^:]+):([^:]+)$/;
const DbClient = {
[SupportedDbType.sqlite]: "better-sqlite3",
[SupportedDbType.postgres]: "pg",
[SupportedDbType.mysql]: "mysql",
};
const defaults = {
dbType: SupportedDbType.sqlite,
filePath: "address_points.db",
initFilePath: join(__dirname, "..", "address_points_init.db"),
verbose: false,
};
let _knexDb = undefined;
let _knexDbConfig = undefined;
export const getKnexDb = (config = {}) => {
if (_knexDb === undefined) {
_knexDbConfig = { ...defaults, ...getEnvConfig(), ...config };
if (_knexDbConfig.dbType === SupportedDbType.sqlite) {
_knexDb = knex({
client: DbClient[SupportedDbType.sqlite],
connection: {
filename: _knexDbConfig.filePath,
},
pool: {
afterCreate: (db, done) => {
db.pragma("journal_mode = WAL;");
done(false, done);
},
},
useNullAsDefault: true,
debug: _knexDbConfig.verbose,
});
}
else if (_knexDbConfig.dbType === SupportedDbType.postgres) {
_knexDb = knex({
client: DbClient[SupportedDbType.postgres],
connection: _knexDbConfig.pgConnectionString,
useNullAsDefault: true,
debug: _knexDbConfig.verbose,
});
}
else if (_knexDbConfig.dbType === SupportedDbType.mysql) {
const [host, port, user, password, database] = _knexDbConfig.mysqlConnectionString.split(":");
_knexDb = knex({
client: DbClient[SupportedDbType.mysql],
connection: {
host,
port: Number(port),
user,
password,
database,
},
useNullAsDefault: true,
debug: _knexDbConfig.verbose,
});
}
else {
throw new Error(`Unsupported DB type: ${_knexDbConfig.dbType}`);
}
}
return _knexDb;
};
export const isPostgres = (knex) => {
return knex.client.config.client === DbClient[SupportedDbType.postgres];
};
export const isSqlite = (knex) => {
return knex.client.config.client === DbClient[SupportedDbType.sqlite];
};
export const isMysql = (knex) => {
return knex.client.config.client === DbClient[SupportedDbType.mysql];
};
const getMigrationConfig = () => {
return { directory: join(__dirname, "migrations") };
};
export const initDb = async (config = {}) => {
const knex = getKnexDb(config);
await knex.migrate.latest(getMigrationConfig());
return knex;
};
export const clearDb = async (config = {}) => {
const knex = getKnexDb(config);
await knex.migrate.rollback(getMigrationConfig(), true);
await knex.migrate.latest(getMigrationConfig());
return knex;
};
export const disconnectKnex = async () => {
if (_knexDb) {
await _knexDb.destroy();
_knexDb = undefined;
}
};
const getEnvConfig = () => {
configDotenv({ path: ".env.local" });
const envConfig = {};
if (process.env.TEXTTOMAP_DB_TYPE &&
Object.values(SupportedDbType).includes(process.env.TEXTTOMAP_DB_TYPE)) {
envConfig.dbType = process.env.TEXTTOMAP_DB_TYPE;
if (envConfig.dbType === SupportedDbType.postgres) {
if (!process.env.TEXTTOMAP_PG_CONNECTION_STRING) {
throw new Error("Environmental variable 'TEXTTOMAP_PG_CONNECTION_STRING' must be set for PostgreSQL.");
}
envConfig.pgConnectionString = process.env.TEXTTOMAP_PG_CONNECTION_STRING;
}
else if (envConfig.dbType === SupportedDbType.mysql) {
if (!process.env.TEXTTOMAP_MYSQL_CONNECTION_DATA) {
throw new Error("Environmental variable 'TEXTTOMAP_MYSQL_CONNECTION_DATA' must be set for MySQL or MariaDB.");
}
if (!mysqlConnectionStringPattern.test(process.env.TEXTTOMAP_MYSQL_CONNECTION_DATA)) {
throw new Error("Environmental variable 'TEXTTOMAP_MYSQL_CONNECTION_DATA' must be in format 'host:port:user:password:database'.");
}
envConfig.mysqlConnectionString =
process.env.TEXTTOMAP_MYSQL_CONNECTION_DATA;
}
else if (envConfig.dbType === SupportedDbType.sqlite) {
if (process.env.TEXTTOMAP_SQLITE_PATH) {
envConfig.filePath = process.env.TEXTTOMAP_SQLITE_PATH;
}
}
}
return envConfig;
};
export const nonEmptyOrNull = (value) => {
return value ? value : null;
};
export const rawQuery = async (query, ...bindings) => {
const knex = getKnexDb();
const result = await knex.raw(query, ...bindings);
return isMysql(knex) ? result[0] : result;
};
/**
* Efficiently insert multiple rows. If preventDuplicatesByFirstColumn is true, the first
* column should be unique (PK or UNIQUE).
*/
export const insertMultipleRows = async (rows, table, columnNames, preventDuplicates = true) => {
if (rows.length === 0) {
return 0;
}
const insertPlaceholders = generate2DPlaceholders(columnNames.length, rows.length);
const knex = getKnexDb();
const onConfict = preventDuplicates && !isMysql(knex) ? `ON CONFLICT DO NOTHING` : "";
await rawQuery(`INSERT ${isMysql(knex) ? "IGNORE" : ""} INTO ${table} (${columnNames.join(",")}) VALUES ${insertPlaceholders} ${onConfict}`, rows.flat());
return rows.length;
};
/**
* Insert a single row and return the autoincremented ID.
*/
export const insertAutoincrementRow = async (row, table, columnNames) => {
const data = columnNames.reduce((obj, columnName, index) => {
obj[columnName] = row[index];
return obj;
}, {});
const result = await getKnexDb().insert(data).into(table);
return result ? Number(result) : null;
};
export const deleteMultipleRowsKnex = async (keys, table, keyColumnName) => {
if (keys.length === 0) {
return;
}
await getKnexDb().from(table).whereIn(keyColumnName, keys).del();
};
const generateRepetitiveString = (value, glue, n) => {
return new Array(n).fill(value).join(glue);
};
export const generatePlaceholders = (n) => {
return generateRepetitiveString("?", ",", n);
};
export const generate2DPlaceholders = (inner, outer) => {
return generateRepetitiveString(`(${generatePlaceholders(inner)})`, ",", outer);
};
export const extractKeyValuesPairs = (array, keyIndex, valuesIndices) => {
const object = array.reduce((acc, data) => {
if (data[keyIndex]) {
acc[data[keyIndex]] = valuesIndices.map((index) => nonEmptyOrNull(data[index]));
}
return acc;
}, {});
return Object.keys(object).map((key) => [key, ...object[key]]);
};
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZGIvZGIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLFFBQVEsQ0FBQztBQUN0QyxPQUFPLElBQWMsTUFBTSxNQUFNLENBQUM7QUFDbEMsT0FBTyxFQUFFLE9BQU8sRUFBRSxJQUFJLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFDckMsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLEtBQUssQ0FBQztBQVdwQyxNQUFNLFVBQVUsR0FBRyxhQUFhLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztBQUNsRCxNQUFNLFNBQVMsR0FBRyxPQUFPLENBQUMsVUFBVSxDQUFDLENBQUM7QUFFdEMsTUFBTSxDQUFOLElBQVksZUFJWDtBQUpELFdBQVksZUFBZTtJQUN6QixvQ0FBaUIsQ0FBQTtJQUNqQix3Q0FBcUIsQ0FBQTtJQUNyQixrQ0FBZSxDQUFBO0FBQ2pCLENBQUMsRUFKVyxlQUFlLEtBQWYsZUFBZSxRQUkxQjtBQUVELE1BQU0sNEJBQTRCLEdBQUcseUNBQXlDLENBQUM7QUFFL0UsTUFBTSxRQUFRLEdBQUc7SUFDZixDQUFDLGVBQWUsQ0FBQyxNQUFNLENBQUMsRUFBRSxnQkFBZ0I7SUFDMUMsQ0FBQyxlQUFlLENBQUMsUUFBUSxDQUFDLEVBQUUsSUFBSTtJQUNoQyxDQUFDLGVBQWUsQ0FBQyxLQUFLLENBQUMsRUFBRSxPQUFPO0NBQ2pDLENBQUM7QUFFRixNQUFNLFFBQVEsR0FBYTtJQUN6QixNQUFNLEVBQUUsZUFBZSxDQUFDLE1BQU07SUFDOUIsUUFBUSxFQUFFLG1CQUFtQjtJQUM3QixZQUFZLEVBQUUsSUFBSSxDQUFDLFNBQVMsRUFBRSxJQUFJLEVBQUUsd0JBQXdCLENBQUM7SUFDN0QsT0FBTyxFQUFFLEtBQUs7Q0FDZixDQUFDO0FBRUYsSUFBSSxPQUFPLEdBQVMsU0FBUyxDQUFDO0FBQzlCLElBQUksYUFBYSxHQUFhLFNBQVMsQ0FBQztBQUV4QyxNQUFNLENBQUMsTUFBTSxTQUFTLEdBQUcsQ0FBQyxTQUE0QixFQUFFLEVBQVEsRUFBRTtJQUNoRSxJQUFJLE9BQU8sS0FBSyxTQUFTLEVBQUU7UUFDekIsYUFBYSxHQUFHLEVBQUUsR0FBRyxRQUFRLEVBQUUsR0FBRyxZQUFZLEVBQUUsRUFBRSxHQUFHLE1BQU0sRUFBRSxDQUFDO1FBRTlELElBQUksYUFBYSxDQUFDLE1BQU0sS0FBSyxlQUFlLENBQUMsTUFBTSxFQUFFO1lBQ25ELE9BQU8sR0FBRyxJQUFJLENBQUM7Z0JBQ2IsTUFBTSxFQUFFLFFBQVEsQ0FBQyxlQUFlLENBQUMsTUFBTSxDQUFDO2dCQUN4QyxVQUFVLEVBQUU7b0JBQ1YsUUFBUSxFQUFFLGFBQWEsQ0FBQyxRQUFRO2lCQUNqQztnQkFDRCxJQUFJLEVBQUU7b0JBQ0osV0FBVyxFQUFFLENBQUMsRUFBTyxFQUFFLElBQWMsRUFBRSxFQUFFO3dCQUN2QyxFQUFFLENBQUMsTUFBTSxDQUFDLHFCQUFxQixDQUFDLENBQUM7d0JBQ2pDLElBQUksQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLENBQUM7b0JBQ3BCLENBQUM7aUJBQ0Y7Z0JBQ0QsZ0JBQWdCLEVBQUUsSUFBSTtnQkFDdEIsS0FBSyxFQUFFLGFBQWEsQ0FBQyxPQUFPO2FBQzdCLENBQUMsQ0FBQztTQUNKO2FBQU0sSUFBSSxhQUFhLENBQUMsTUFBTSxLQUFLLGVBQWUsQ0FBQyxRQUFRLEVBQUU7WUFDNUQsT0FBTyxHQUFHLElBQUksQ0FBQztnQkFDYixNQUFNLEVBQUUsUUFBUSxDQUFDLGVBQWUsQ0FBQyxRQUFRLENBQUM7Z0JBQzFDLFVBQVUsRUFBRSxhQUFhLENBQUMsa0JBQWtCO2dCQUM1QyxnQkFBZ0IsRUFBRSxJQUFJO2dCQUN0QixLQUFLLEVBQUUsYUFBYSxDQUFDLE9BQU87YUFDN0IsQ0FBQyxDQUFDO1NBQ0o7YUFBTSxJQUFJLGFBQWEsQ0FBQyxNQUFNLEtBQUssZUFBZSxDQUFDLEtBQUssRUFBRTtZQUN6RCxNQUFNLENBQUMsSUFBSSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsUUFBUSxFQUFFLFFBQVEsQ0FBQyxHQUMxQyxhQUFhLENBQUMscUJBQXFCLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO1lBQ2pELE9BQU8sR0FBRyxJQUFJLENBQUM7Z0JBQ2IsTUFBTSxFQUFFLFFBQVEsQ0FBQyxlQUFlLENBQUMsS0FBSyxDQUFDO2dCQUN2QyxVQUFVLEVBQUU7b0JBQ1YsSUFBSTtvQkFDSixJQUFJLEVBQUUsTUFBTSxDQUFDLElBQUksQ0FBQztvQkFDbEIsSUFBSTtvQkFDSixRQUFRO29CQUNSLFFBQVE7aUJBQ1Q7Z0JBQ0QsZ0JBQWdCLEVBQUUsSUFBSTtnQkFDdEIsS0FBSyxFQUFFLGFBQWEsQ0FBQyxPQUFPO2FBQzdCLENBQUMsQ0FBQztTQUNKO2FBQU07WUFDTCxNQUFNLElBQUksS0FBSyxDQUFDLHdCQUF3QixhQUFhLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztTQUNqRTtLQUNGO0lBRUQsT0FBTyxPQUFPLENBQUM7QUFDakIsQ0FBQyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0sVUFBVSxHQUFHLENBQUMsSUFBVSxFQUFXLEVBQUU7SUFDaEQsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEtBQUssUUFBUSxDQUFDLGVBQWUsQ0FBQyxRQUFRLENBQUMsQ0FBQztBQUMxRSxDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxRQUFRLEdBQUcsQ0FBQyxJQUFVLEVBQVcsRUFBRTtJQUM5QyxPQUFPLElBQUksQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLE1BQU0sS0FBSyxRQUFRLENBQUMsZUFBZSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0FBQ3hFLENBQUMsQ0FBQztBQUVGLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxDQUFDLElBQVUsRUFBVyxFQUFFO0lBQzdDLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsTUFBTSxLQUFLLFFBQVEsQ0FBQyxlQUFlLENBQUMsS0FBSyxDQUFDLENBQUM7QUFDdkUsQ0FBQyxDQUFDO0FBRUYsTUFBTSxrQkFBa0IsR0FBRyxHQUF3QixFQUFFO0lBQ25ELE9BQU8sRUFBRSxTQUFTLEVBQUUsSUFBSSxDQUFDLFNBQVMsRUFBRSxZQUFZLENBQUMsRUFBRSxDQUFDO0FBQ3RELENBQUMsQ0FBQztBQUVGLE1BQU0sQ0FBQyxNQUFNLE1BQU0sR0FBRyxLQUFLLEVBQUUsU0FBNEIsRUFBRSxFQUFpQixFQUFFO0lBQzVFLE1BQU0sSUFBSSxHQUFHLFNBQVMsQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUMvQixNQUFNLElBQUksQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLGtCQUFrQixFQUFFLENBQUMsQ0FBQztJQUNoRCxPQUFPLElBQUksQ0FBQztBQUNkLENBQUMsQ0FBQztBQUVGLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxLQUFLLEVBQzFCLFNBQTRCLEVBQUUsRUFDZixFQUFFO0lBQ2pCLE1BQU0sSUFBSSxHQUFHLFNBQVMsQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUMvQixNQUFNLElBQUksQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLGtCQUFrQixFQUFFLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDeEQsTUFBTSxJQUFJLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxrQkFBa0IsRUFBRSxDQUFDLENBQUM7SUFDaEQsT0FBTyxJQUFJLENBQUM7QUFDZCxDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxjQUFjLEdBQUcsS0FBSyxJQUFtQixFQUFFO0lBQ3RELElBQUksT0FBTyxFQUFFO1FBQ1gsTUFBTSxPQUFPLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDeEIsT0FBTyxHQUFHLFNBQVMsQ0FBQztLQUNyQjtBQUNILENBQUMsQ0FBQztBQUVGLE1BQU0sWUFBWSxHQUFHLEdBQXNCLEVBQUU7SUFDM0MsWUFBWSxDQUFDLEVBQUUsSUFBSSxFQUFFLFlBQVksRUFBRSxDQUFDLENBQUM7SUFDckMsTUFBTSxTQUFTLEdBQXNCLEVBQUUsQ0FBQztJQUN4QyxJQUNFLE9BQU8sQ0FBQyxHQUFHLENBQUMsaUJBQWlCO1FBQzdCLE1BQU0sQ0FBQyxNQUFNLENBQUMsZUFBZSxDQUFDLENBQUMsUUFBUSxDQUNyQyxPQUFPLENBQUMsR0FBRyxDQUFDLGlCQUFvQyxDQUNqRCxFQUNEO1FBQ0EsU0FBUyxDQUFDLE1BQU0sR0FBRyxPQUFPLENBQUMsR0FBRyxDQUFDLGlCQUFvQyxDQUFDO1FBQ3BFLElBQUksU0FBUyxDQUFDLE1BQU0sS0FBSyxlQUFlLENBQUMsUUFBUSxFQUFFO1lBQ2pELElBQUksQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLDhCQUE4QixFQUFFO2dCQUMvQyxNQUFNLElBQUksS0FBSyxDQUNiLHFGQUFxRixDQUN0RixDQUFDO2FBQ0g7WUFDRCxTQUFTLENBQUMsa0JBQWtCLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyw4QkFBOEIsQ0FBQztTQUMzRTthQUFNLElBQUksU0FBUyxDQUFDLE1BQU0sS0FBSyxlQUFlLENBQUMsS0FBSyxFQUFFO1lBQ3JELElBQUksQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLCtCQUErQixFQUFFO2dCQUNoRCxNQUFNLElBQUksS0FBSyxDQUNiLDRGQUE0RixDQUM3RixDQUFDO2FBQ0g7WUFFRCxJQUNFLENBQUMsNEJBQTRCLENBQUMsSUFBSSxDQUNoQyxPQUFPLENBQUMsR0FBRyxDQUFDLCtCQUErQixDQUM1QyxFQUNEO2dCQUNBLE1BQU0sSUFBSSxLQUFLLENBQ2IsZ0hBQWdILENBQ2pILENBQUM7YUFDSDtZQUNELFNBQVMsQ0FBQyxxQkFBcUI7Z0JBQzdCLE9BQU8sQ0FBQyxHQUFHLENBQUMsK0JBQStCLENBQUM7U0FDL0M7YUFBTSxJQUFJLFNBQVMsQ0FBQyxNQUFNLEtBQUssZUFBZSxDQUFDLE1BQU0sRUFBRTtZQUN0RCxJQUFJLE9BQU8sQ0FBQyxHQUFHLENBQUMscUJBQXFCLEVBQUU7Z0JBQ3JDLFNBQVMsQ0FBQyxRQUFRLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxxQkFBcUIsQ0FBQzthQUN4RDtTQUNGO0tBQ0Y7SUFDRCxPQUFPLFNBQVMsQ0FBQztBQUNuQixDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxjQUFjLEdBQUcsQ0FBQyxLQUFhLEVBQWlCLEVBQUU7SUFDN0QsT0FBTyxLQUFLLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDO0FBQzlCLENBQUMsQ0FBQztBQUVGLE1BQU0sQ0FBQyxNQUFNLFFBQVEsR0FBRyxLQUFLLEVBQzNCLEtBQWEsRUFDYixHQUFHLFFBQW9DLEVBQ3ZCLEVBQUU7SUFDbEIsTUFBTSxJQUFJLEdBQUcsU0FBUyxFQUFFLENBQUM7SUFDekIsTUFBTSxNQUFNLEdBQUcsTUFBTSxJQUFJLENBQUMsR0FBRyxDQUFDLEtBQUssRUFBRSxHQUFHLFFBQVEsQ0FBQyxDQUFDO0lBQ2xELE9BQU8sT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQztBQUM1QyxDQUFDLENBQUM7QUFFRjs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSxrQkFBa0IsR0FBRyxLQUFLLEVBQ3JDLElBQWdCLEVBQ2hCLEtBQWEsRUFDYixXQUFxQixFQUNyQixvQkFBNkIsSUFBSSxFQUNoQixFQUFFO0lBQ25CLElBQUksSUFBSSxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUU7UUFDckIsT0FBTyxDQUFDLENBQUM7S0FDVjtJQUVELE1BQU0sa0JBQWtCLEdBQUcsc0JBQXNCLENBQy9DLFdBQVcsQ0FBQyxNQUFNLEVBQ2xCLElBQUksQ0FBQyxNQUFNLENBQ1osQ0FBQztJQUVGLE1BQU0sSUFBSSxHQUFHLFNBQVMsRUFBRSxDQUFDO0lBRXpCLE1BQU0sU0FBUyxHQUNiLGlCQUFpQixJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyx3QkFBd0IsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO0lBRXRFLE1BQU0sUUFBUSxDQUNaLFVBQVUsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLEVBQUUsU0FBUyxLQUFLLEtBQUssV0FBVyxDQUFDLElBQUksQ0FDeEUsR0FBRyxDQUNKLFlBQVksa0JBQWtCLElBQUksU0FBUyxFQUFFLEVBQzlDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FDWixDQUFDO0lBRUYsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDO0FBQ3JCLENBQUMsQ0FBQztBQUVGOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sc0JBQXNCLEdBQUcsS0FBSyxFQUN6QyxHQUFVLEVBQ1YsS0FBYSxFQUNiLFdBQXFCLEVBQ0csRUFBRTtJQUMxQixNQUFNLElBQUksR0FBRyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUMsR0FBRyxFQUFFLFVBQVUsRUFBRSxLQUFLLEVBQUUsRUFBRTtRQUN6RCxHQUFHLENBQUMsVUFBVSxDQUFDLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzdCLE9BQU8sR0FBRyxDQUFDO0lBQ2IsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDO0lBRVAsTUFBTSxNQUFNLEdBQUcsTUFBTSxTQUFTLEVBQUUsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBRTFELE9BQU8sTUFBTSxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQztBQUN4QyxDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxzQkFBc0IsR0FBRyxLQUFLLEVBQ3pDLElBQWMsRUFDZCxLQUFhLEVBQ2IsYUFBcUIsRUFDTixFQUFFO0lBQ2pCLElBQUksSUFBSSxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUU7UUFDckIsT0FBTztLQUNSO0lBQ0QsTUFBTSxTQUFTLEVBQUUsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsT0FBTyxDQUFDLGFBQWEsRUFBRSxJQUFJLENBQUMsQ0FBQyxHQUFHLEVBQUUsQ0FBQztBQUNuRSxDQUFDLENBQUM7QUFFRixNQUFNLHdCQUF3QixHQUFHLENBQy9CLEtBQWEsRUFDYixJQUFZLEVBQ1osQ0FBUyxFQUNELEVBQUU7SUFDVixPQUFPLElBQUksS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7QUFDN0MsQ0FBQyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0sb0JBQW9CLEdBQUcsQ0FBQyxDQUFTLEVBQVUsRUFBRTtJQUN4RCxPQUFPLHdCQUF3QixDQUFDLEdBQUcsRUFBRSxHQUFHLEVBQUUsQ0FBQyxDQUFDLENBQUM7QUFDL0MsQ0FBQyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0sc0JBQXNCLEdBQUcsQ0FDcEMsS0FBYSxFQUNiLEtBQWEsRUFDTCxFQUFFO0lBQ1YsT0FBTyx3QkFBd0IsQ0FDN0IsSUFBSSxvQkFBb0IsQ0FBQyxLQUFLLENBQUMsR0FBRyxFQUNsQyxHQUFHLEVBQ0gsS0FBSyxDQUNOLENBQUM7QUFDSixDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxxQkFBcUIsR0FBRyxDQUNuQyxLQUFpQixFQUNqQixRQUFnQixFQUNoQixhQUF1QixFQUNYLEVBQUU7SUFDZCxNQUFNLE1BQU0sR0FBRyxLQUFLLENBQUMsTUFBTSxDQUFDLENBQUMsR0FBVyxFQUFFLElBQWMsRUFBRSxFQUFFO1FBQzFELElBQUksSUFBSSxDQUFDLFFBQVEsQ0FBQyxFQUFFO1lBQ2xCLEdBQUcsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUMsR0FBRyxhQUFhLENBQUMsR0FBRyxDQUFDLENBQUMsS0FBYSxFQUFFLEVBQUUsQ0FDeEQsY0FBYyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUM1QixDQUFDO1NBQ0g7UUFDRCxPQUFPLEdBQUcsQ0FBQztJQUNiLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztJQUVQLE9BQU8sTUFBTSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxHQUFHLEVBQUUsRUFBRSxDQUFDLENBQUMsR0FBRyxFQUFFLEdBQUcsTUFBTSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUNqRSxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjb25maWdEb3RlbnYgfSBmcm9tIFwiZG90ZW52XCI7XHJcbmltcG9ydCBrbmV4LCB7IEtuZXggfSBmcm9tIFwia25leFwiO1xyXG5pbXBvcnQgeyBkaXJuYW1lLCBqb2luIH0gZnJvbSBcInBhdGhcIjtcclxuaW1wb3J0IHsgZmlsZVVSTFRvUGF0aCB9IGZyb20gXCJ1cmxcIjtcclxuXHJcbmludGVyZmFjZSBEYkNvbmZpZyB7XHJcbiAgZGJUeXBlOiBTdXBwb3J0ZWREYlR5cGU7XHJcbiAgcGdDb25uZWN0aW9uU3RyaW5nPzogc3RyaW5nO1xyXG4gIG15c3FsQ29ubmVjdGlvblN0cmluZz86IHN0cmluZztcclxuICBpbml0RmlsZVBhdGg6IHN0cmluZztcclxuICBmaWxlUGF0aD86IHN0cmluZztcclxuICB2ZXJib3NlOiBib29sZWFuO1xyXG59XHJcblxyXG5jb25zdCBfX2ZpbGVuYW1lID0gZmlsZVVSTFRvUGF0aChpbXBvcnQubWV0YS51cmwpO1xyXG5jb25zdCBfX2Rpcm5hbWUgPSBkaXJuYW1lKF9fZmlsZW5hbWUpO1xyXG5cclxuZXhwb3J0IGVudW0gU3VwcG9ydGVkRGJUeXBlIHtcclxuICBzcWxpdGUgPSBcInNxbGl0ZVwiLFxyXG4gIHBvc3RncmVzID0gXCJwb3N0Z3Jlc1wiLFxyXG4gIG15c3FsID0gXCJteXNxbFwiLFxyXG59XHJcblxyXG5jb25zdCBteXNxbENvbm5lY3Rpb25TdHJpbmdQYXR0ZXJuID0gL14oW146XSspOihcXGQrKTooW146XSspOihbXjpdKyk6KFteOl0rKSQvO1xyXG5cclxuY29uc3QgRGJDbGllbnQgPSB7XHJcbiAgW1N1cHBvcnRlZERiVHlwZS5zcWxpdGVdOiBcImJldHRlci1zcWxpdGUzXCIsXHJcbiAgW1N1cHBvcnRlZERiVHlwZS5wb3N0Z3Jlc106IFwicGdcIixcclxuICBbU3VwcG9ydGVkRGJUeXBlLm15c3FsXTogXCJteXNxbFwiLFxyXG59O1xyXG5cclxuY29uc3QgZGVmYXVsdHM6IERiQ29uZmlnID0ge1xyXG4gIGRiVHlwZTogU3VwcG9ydGVkRGJUeXBlLnNxbGl0ZSxcclxuICBmaWxlUGF0aDogXCJhZGRyZXNzX3BvaW50cy5kYlwiLFxyXG4gIGluaXRGaWxlUGF0aDogam9pbihfX2Rpcm5hbWUsIFwiLi5cIiwgXCJhZGRyZXNzX3BvaW50c19pbml0LmRiXCIpLFxyXG4gIHZlcmJvc2U6IGZhbHNlLFxyXG59O1xyXG5cclxubGV0IF9rbmV4RGI6IEtuZXggPSB1bmRlZmluZWQ7XHJcbmxldCBfa25leERiQ29uZmlnOiBEYkNvbmZpZyA9IHVuZGVmaW5lZDtcclxuXHJcbmV4cG9ydCBjb25zdCBnZXRLbmV4RGIgPSAoY29uZmlnOiBQYXJ0aWFsPERiQ29uZmlnPiA9IHt9KTogS25leCA9PiB7XHJcbiAgaWYgKF9rbmV4RGIgPT09IHVuZGVmaW5lZCkge1xyXG4gICAgX2tuZXhEYkNvbmZpZyA9IHsgLi4uZGVmYXVsdHMsIC4uLmdldEVudkNvbmZpZygpLCAuLi5jb25maWcgfTtcclxuXHJcbiAgICBpZiAoX2tuZXhEYkNvbmZpZy5kYlR5cGUgPT09IFN1cHBvcnRlZERiVHlwZS5zcWxpdGUpIHtcclxuICAgICAgX2tuZXhEYiA9IGtuZXgoe1xyXG4gICAgICAgIGNsaWVudDogRGJDbGllbnRbU3VwcG9ydGVkRGJUeXBlLnNxbGl0ZV0sXHJcbiAgICAgICAgY29ubmVjdGlvbjoge1xyXG4gICAgICAgICAgZmlsZW5hbWU6IF9rbmV4RGJDb25maWcuZmlsZVBhdGgsXHJcbiAgICAgICAgfSxcclxuICAgICAgICBwb29sOiB7XHJcbiAgICAgICAgICBhZnRlckNyZWF0ZTogKGRiOiBhbnksIGRvbmU6IEZ1bmN0aW9uKSA9PiB7XHJcbiAgICAgICAgICAgIGRiLnByYWdtYShcImpvdXJuYWxfbW9kZSA9IFdBTDtcIik7XHJcbiAgICAgICAgICAgIGRvbmUoZmFsc2UsIGRvbmUpO1xyXG4gICAgICAgICAgfSxcclxuICAgICAgICB9LFxyXG4gICAgICAgIHVzZU51bGxBc0RlZmF1bHQ6IHRydWUsXHJcbiAgICAgICAgZGVidWc6IF9rbmV4RGJDb25maWcudmVyYm9zZSxcclxuICAgICAgfSk7XHJcbiAgICB9IGVsc2UgaWYgKF9rbmV4RGJDb25maWcuZGJUeXBlID09PSBTdXBwb3J0ZWREYlR5cGUucG9zdGdyZXMpIHtcclxuICAgICAgX2tuZXhEYiA9IGtuZXgoe1xyXG4gICAgICAgIGNsaWVudDogRGJDbGllbnRbU3VwcG9ydGVkRGJUeXBlLnBvc3RncmVzXSxcclxuICAgICAgICBjb25uZWN0aW9uOiBfa25leERiQ29uZmlnLnBnQ29ubmVjdGlvblN0cmluZyxcclxuICAgICAgICB1c2VOdWxsQXNEZWZhdWx0OiB0cnVlLFxyXG4gICAgICAgIGRlYnVnOiBfa25leERiQ29uZmlnLnZlcmJvc2UsXHJcbiAgICAgIH0pO1xyXG4gICAgfSBlbHNlIGlmIChfa25leERiQ29uZmlnLmRiVHlwZSA9PT0gU3VwcG9ydGVkRGJUeXBlLm15c3FsKSB7XHJcbiAgICAgIGNvbnN0IFtob3N0LCBwb3J0LCB1c2VyLCBwYXNzd29yZCwgZGF0YWJhc2VdID1cclxuICAgICAgICBfa25leERiQ29uZmlnLm15c3FsQ29ubmVjdGlvblN0cmluZy5zcGxpdChcIjpcIik7XHJcbiAgICAgIF9rbmV4RGIgPSBrbmV4KHtcclxuICAgICAgICBjbGllbnQ6IERiQ2xpZW50W1N1cHBvcnRlZERiVHlwZS5teXNxbF0sXHJcbiAgICAgICAgY29ubmVjdGlvbjoge1xyXG4gICAgICAgICAgaG9zdCxcclxuICAgICAgICAgIHBvcnQ6IE51bWJlcihwb3J0KSxcclxuICAgICAgICAgIHVzZXIsXHJcbiAgICAgICAgICBwYXNzd29yZCxcclxuICAgICAgICAgIGRhdGFiYXNlLFxyXG4gICAgICAgIH0sXHJcbiAgICAgICAgdXNlTnVsbEFzRGVmYXVsdDogdHJ1ZSxcclxuICAgICAgICBkZWJ1ZzogX2tuZXhEYkNvbmZpZy52ZXJib3NlLFxyXG4gICAgICB9KTtcclxuICAgIH0gZWxzZSB7XHJcbiAgICAgIHRocm93IG5ldyBFcnJvcihgVW5zdXBwb3J0ZWQgREIgdHlwZTogJHtfa25leERiQ29uZmlnLmRiVHlwZX1gKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIHJldHVybiBfa25leERiO1xyXG59O1xyXG5cclxuZXhwb3J0IGNvbnN0IGlzUG9zdGdyZXMgPSAoa25leDogS25leCk6IGJvb2xlYW4gPT4ge1xyXG4gIHJldHVybiBrbmV4LmNsaWVudC5jb25maWcuY2xpZW50ID09PSBEYkNsaWVudFtTdXBwb3J0ZWREYlR5cGUucG9zdGdyZXNdO1xyXG59O1xyXG5cclxuZXhwb3J0IGNvbnN0IGlzU3FsaXRlID0gKGtuZXg6IEtuZXgpOiBib29sZWFuID0+IHtcclxuICByZXR1cm4ga25leC5jbGllbnQuY29uZmlnLmNsaWVudCA9PT0gRGJDbGllbnRbU3VwcG9ydGVkRGJUeXBlLnNxbGl0ZV07XHJcbn07XHJcblxyXG5leHBvcnQgY29uc3QgaXNNeXNxbCA9IChrbmV4OiBLbmV4KTogYm9vbGVhbiA9PiB7XHJcbiAgcmV0dXJuIGtuZXguY2xpZW50LmNvbmZpZy5jbGllbnQgPT09IERiQ2xpZW50W1N1cHBvcnRlZERiVHlwZS5teXNxbF07XHJcbn07XHJcblxyXG5jb25zdCBnZXRNaWdyYXRpb25Db25maWcgPSAoKTogS25leC5NaWdyYXRvckNvbmZpZyA9PiB7XHJcbiAgcmV0dXJuIHsgZGlyZWN0b3J5OiBqb2luKF9fZGlybmFtZSwgXCJtaWdyYXRpb25zXCIpIH07XHJcbn07XHJcblxyXG5leHBvcnQgY29uc3QgaW5pdERiID0gYXN5bmMgKGNvbmZpZzogUGFydGlhbDxEYkNvbmZpZz4gPSB7fSk6IFByb21pc2U8S25leD4gPT4ge1xyXG4gIGNvbnN0IGtuZXggPSBnZXRLbmV4RGIoY29uZmlnKTtcclxuICBhd2FpdCBrbmV4Lm1pZ3JhdGUubGF0ZXN0KGdldE1pZ3JhdGlvbkNvbmZpZygpKTtcclxuICByZXR1cm4ga25leDtcclxufTtcclxuXHJcbmV4cG9ydCBjb25zdCBjbGVhckRiID0gYXN5bmMgKFxyXG4gIGNvbmZpZzogUGFydGlhbDxEYkNvbmZpZz4gPSB7fVxyXG4pOiBQcm9taXNlPEtuZXg+ID0+IHtcclxuICBjb25zdCBrbmV4ID0gZ2V0S25leERiKGNvbmZpZyk7XHJcbiAgYXdhaXQga25leC5taWdyYXRlLnJvbGxiYWNrKGdldE1pZ3JhdGlvbkNvbmZpZygpLCB0cnVlKTtcclxuICBhd2FpdCBrbmV4Lm1pZ3JhdGUubGF0ZXN0KGdldE1pZ3JhdGlvbkNvbmZpZygpKTtcclxuICByZXR1cm4ga25leDtcclxufTtcclxuXHJcbmV4cG9ydCBjb25zdCBkaXNjb25uZWN0S25leCA9IGFzeW5jICgpOiBQcm9taXNlPHZvaWQ+ID0+IHtcclxuICBpZiAoX2tuZXhEYikge1xyXG4gICAgYXdhaXQgX2tuZXhEYi5kZXN0cm95KCk7XHJcbiAgICBfa25leERiID0gdW5kZWZpbmVkO1xyXG4gIH1cclxufTtcclxuXHJcbmNvbnN0IGdldEVudkNvbmZpZyA9ICgpOiBQYXJ0aWFsPERiQ29uZmlnPiA9PiB7XHJcbiAgY29uZmlnRG90ZW52KHsgcGF0aDogXCIuZW52LmxvY2FsXCIgfSk7XHJcbiAgY29uc3QgZW52Q29uZmlnOiBQYXJ0aWFsPERiQ29uZmlnPiA9IHt9O1xyXG4gIGlmIChcclxuICAgIHByb2Nlc3MuZW52LlRFWFRUT01BUF9EQl9UWVBFICYmXHJcbiAgICBPYmplY3QudmFsdWVzKFN1cHBvcnRlZERiVHlwZSkuaW5jbHVkZXMoXHJcbiAgICAgIHByb2Nlc3MuZW52LlRFWFRUT01BUF9EQl9UWVBFIGFzIFN1cHBvcnRlZERiVHlwZVxyXG4gICAgKVxyXG4gICkge1xyXG4gICAgZW52Q29uZmlnLmRiVHlwZSA9IHByb2Nlc3MuZW52LlRFWFRUT01BUF9EQl9UWVBFIGFzIFN1cHBvcnRlZERiVHlwZTtcclxuICAgIGlmIChlbnZDb25maWcuZGJUeXBlID09PSBTdXBwb3J0ZWREYlR5cGUucG9zdGdyZXMpIHtcclxuICAgICAgaWYgKCFwcm9jZXNzLmVudi5URVhUVE9NQVBfUEdfQ09OTkVDVElPTl9TVFJJTkcpIHtcclxuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXHJcbiAgICAgICAgICBcIkVudmlyb25tZW50YWwgdmFyaWFibGUgJ1RFWFRUT01BUF9QR19DT05ORUNUSU9OX1NUUklORycgbXVzdCBiZSBzZXQgZm9yIFBvc3RncmVTUUwuXCJcclxuICAgICAgICApO1xyXG4gICAgICB9XHJcbiAgICAgIGVudkNvbmZpZy5wZ0Nvbm5lY3Rpb25TdHJpbmcgPSBwcm9jZXNzLmVudi5URVhUVE9NQVBfUEdfQ09OTkVDVElPTl9TVFJJTkc7XHJcbiAgICB9IGVsc2UgaWYgKGVudkNvbmZpZy5kYlR5cGUgPT09IFN1cHBvcnRlZERiVHlwZS5teXNxbCkge1xyXG4gICAgICBpZiAoIXByb2Nlc3MuZW52LlRFWFRUT01BUF9NWVNRTF9DT05ORUNUSU9OX0RBVEEpIHtcclxuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXHJcbiAgICAgICAgICBcIkVudmlyb25tZW50YWwgdmFyaWFibGUgJ1RFWFRUT01BUF9NWVNRTF9DT05ORUNUSU9OX0RBVEEnIG11c3QgYmUgc2V0IGZvciBNeVNRTCBvciBNYXJpYURCLlwiXHJcbiAgICAgICAgKTtcclxuICAgICAgfVxyXG5cclxuICAgICAgaWYgKFxyXG4gICAgICAgICFteXNxbENvbm5lY3Rpb25TdHJpbmdQYXR0ZXJuLnRlc3QoXHJcbiAgICAgICAgICBwcm9jZXNzLmVudi5URVhUVE9NQVBfTVlTUUxfQ09OTkVDVElPTl9EQVRBXHJcbiAgICAgICAgKVxyXG4gICAgICApIHtcclxuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXHJcbiAgICAgICAgICBcIkVudmlyb25tZW50YWwgdmFyaWFibGUgJ1RFWFRUT01BUF9NWVNRTF9DT05ORUNUSU9OX0RBVEEnIG11c3QgYmUgaW4gZm9ybWF0ICdob3N0OnBvcnQ6dXNlcjpwYXNzd29yZDpkYXRhYmFzZScuXCJcclxuICAgICAgICApO1xyXG4gICAgICB9XHJcbiAgICAgIGVudkNvbmZpZy5teXNxbENvbm5lY3Rpb25TdHJpbmcgPVxyXG4gICAgICAgIHByb2Nlc3MuZW52LlRFWFRUT01BUF9NWVNRTF9DT05ORUNUSU9OX0RBVEE7XHJcbiAgICB9IGVsc2UgaWYgKGVudkNvbmZpZy5kYlR5cGUgPT09IFN1cHBvcnRlZERiVHlwZS5zcWxpdGUpIHtcclxuICAgICAgaWYgKHByb2Nlc3MuZW52LlRFWFRUT01BUF9TUUxJVEVfUEFUSCkge1xyXG4gICAgICAgIGVudkNvbmZpZy5maWxlUGF0aCA9IHByb2Nlc3MuZW52LlRFWFRUT01BUF9TUUxJVEVfUEFUSDtcclxuICAgICAgfVxyXG4gICAgfVxyXG4gIH1cclxuICByZXR1cm4gZW52Q29uZmlnO1xyXG59O1xyXG5cclxuZXhwb3J0IGNvbnN0IG5vbkVtcHR5T3JOdWxsID0gKHZhbHVlOiBzdHJpbmcpOiBzdHJpbmcgfCBudWxsID0+IHtcclxuICByZXR1cm4gdmFsdWUgPyB2YWx1ZSA6IG51bGw7XHJcbn07XHJcblxyXG5leHBvcnQgY29uc3QgcmF3UXVlcnkgPSBhc3luYyAoXHJcbiAgcXVlcnk6IHN0cmluZyxcclxuICAuLi5iaW5kaW5nczogcmVhZG9ubHkgS25leC5SYXdCaW5kaW5nW11cclxuKTogUHJvbWlzZTxhbnlbXT4gPT4ge1xyXG4gIGNvbnN0IGtuZXggPSBnZXRLbmV4RGIoKTtcclxuICBjb25zdCByZXN1bHQgPSBhd2FpdCBrbmV4LnJhdyhxdWVyeSwgLi4uYmluZGluZ3MpO1xyXG4gIHJldHVybiBpc015c3FsKGtuZXgpID8gcmVzdWx0WzBdIDogcmVzdWx0O1xyXG59O1xyXG5cclxuLyoqXHJcbiAqIEVmZmljaWVudGx5IGluc2VydCBtdWx0aXBsZSByb3dzLiBJZiBwcmV2ZW50RHVwbGljYXRlc0J5Rmlyc3RDb2x1bW4gaXMgdHJ1ZSwgdGhlIGZpcnN0XHJcbiAqIGNvbHVtbiBzaG91bGQgYmUgdW5pcXVlIChQSyBvciBVTklRVUUpLlxyXG4gKi9cclxuZXhwb3J0IGNvbnN0IGluc2VydE11bHRpcGxlUm93cyA9IGFzeW5jIChcclxuICByb3dzOiBzdHJpbmdbXVtdLFxyXG4gIHRhYmxlOiBzdHJpbmcsXHJcbiAgY29sdW1uTmFtZXM6IHN0cmluZ1tdLFxyXG4gIHByZXZlbnREdXBsaWNhdGVzOiBib29sZWFuID0gdHJ1ZVxyXG4pOiBQcm9taXNlPG51bWJlcj4gPT4ge1xyXG4gIGlmIChyb3dzLmxlbmd0aCA9PT0gMCkge1xyXG4gICAgcmV0dXJuIDA7XHJcbiAgfVxyXG5cclxuICBjb25zdCBpbnNlcnRQbGFjZWhvbGRlcnMgPSBnZW5lcmF0ZTJEUGxhY2Vob2xkZXJzKFxyXG4gICAgY29sdW1uTmFtZXMubGVuZ3RoLFxyXG4gICAgcm93cy5sZW5ndGhcclxuICApO1xyXG5cclxuICBjb25zdCBrbmV4ID0gZ2V0S25leERiKCk7XHJcblxyXG4gIGNvbnN0IG9uQ29uZmljdCA9XHJcbiAgICBwcmV2ZW50RHVwbGljYXRlcyAmJiAhaXNNeXNxbChrbmV4KSA/IGBPTiBDT05GTElDVCBETyBOT1RISU5HYCA6IFwiXCI7XHJcblxyXG4gIGF3YWl0IHJhd1F1ZXJ5KFxyXG4gICAgYElOU0VSVCAke2lzTXlzcWwoa25leCkgPyBcIklHTk9SRVwiIDogXCJcIn0gSU5UTyAke3RhYmxlfSAoJHtjb2x1bW5OYW1lcy5qb2luKFxyXG4gICAgICBcIixcIlxyXG4gICAgKX0pIFZBTFVFUyAke2luc2VydFBsYWNlaG9sZGVyc30gJHtvbkNvbmZpY3R9YCxcclxuICAgIHJvd3MuZmxhdCgpXHJcbiAgKTtcclxuXHJcbiAgcmV0dXJuIHJvd3MubGVuZ3RoO1xyXG59O1xyXG5cclxuLyoqXHJcbiAqIEluc2VydCBhIHNpbmdsZSByb3cgYW5kIHJldHVybiB0aGUgYXV0b2luY3JlbWVudGVkIElELlxyXG4gKi9cclxuZXhwb3J0IGNvbnN0IGluc2VydEF1dG9pbmNyZW1lbnRSb3cgPSBhc3luYyAoXHJcbiAgcm93OiBhbnlbXSxcclxuICB0YWJsZTogc3RyaW5nLFxyXG4gIGNvbHVtbk5hbWVzOiBzdHJpbmdbXVxyXG4pOiBQcm9taXNlPG51bWJlciB8IG51bGw+ID0+IHtcclxuICBjb25zdCBkYXRhID0gY29sdW1uTmFtZXMucmVkdWNlKChvYmosIGNvbHVtbk5hbWUsIGluZGV4KSA9PiB7XHJcbiAgICBvYmpbY29sdW1uTmFtZV0gPSByb3dbaW5kZXhdO1xyXG4gICAgcmV0dXJuIG9iajtcclxuICB9LCB7fSk7XHJcblxyXG4gIGNvbnN0IHJlc3VsdCA9IGF3YWl0IGdldEtuZXhEYigpLmluc2VydChkYXRhKS5pbnRvKHRhYmxlKTtcclxuXHJcbiAgcmV0dXJuIHJlc3VsdCA/IE51bWJlcihyZXN1bHQpIDogbnVsbDtcclxufTtcclxuXHJcbmV4cG9ydCBjb25zdCBkZWxldGVNdWx0aXBsZVJvd3NLbmV4ID0gYXN5bmMgKFxyXG4gIGtleXM6IHN0cmluZ1tdLFxyXG4gIHRhYmxlOiBzdHJpbmcsXHJcbiAga2V5Q29sdW1uTmFtZTogc3RyaW5nXHJcbik6IFByb21pc2U8dm9pZD4gPT4ge1xyXG4gIGlmIChrZXlzLmxlbmd0aCA9PT0gMCkge1xyXG4gICAgcmV0dXJuO1xyXG4gIH1cclxuICBhd2FpdCBnZXRLbmV4RGIoKS5mcm9tKHRhYmxlKS53aGVyZUluKGtleUNvbHVtbk5hbWUsIGtleXMpLmRlbCgpO1xyXG59O1xyXG5cclxuY29uc3QgZ2VuZXJhdGVSZXBldGl0aXZlU3RyaW5nID0gKFxyXG4gIHZhbHVlOiBzdHJpbmcsXHJcbiAgZ2x1ZTogc3RyaW5nLFxyXG4gIG46IG51bWJlclxyXG4pOiBzdHJpbmcgPT4ge1xyXG4gIHJldHVybiBuZXcgQXJyYXkobikuZmlsbCh2YWx1ZSkuam9pbihnbHVlKTtcclxufTtcclxuXHJcbmV4cG9ydCBjb25zdCBnZW5lcmF0ZVBsYWNlaG9sZGVycyA9IChuOiBudW1iZXIpOiBzdHJpbmcgPT4ge1xyXG4gIHJldHVybiBnZW5lcmF0ZVJlcGV0aXRpdmVTdHJpbmcoXCI/XCIsIFwiLFwiLCBuKTtcclxufTtcclxuXHJcbmV4cG9ydCBjb25zdCBnZW5lcmF0ZTJEUGxhY2Vob2xkZXJzID0gKFxyXG4gIGlubmVyOiBudW1iZXIsXHJcbiAgb3V0ZXI6IG51bWJlclxyXG4pOiBzdHJpbmcgPT4ge1xyXG4gIHJldHVybiBnZW5lcmF0ZVJlcGV0aXRpdmVTdHJpbmcoXHJcbiAgICBgKCR7Z2VuZXJhdGVQbGFjZWhvbGRlcnMoaW5uZXIpfSlgLFxyXG4gICAgXCIsXCIsXHJcbiAgICBvdXRlclxyXG4gICk7XHJcbn07XHJcblxyXG5leHBvcnQgY29uc3QgZXh0cmFjdEtleVZhbHVlc1BhaXJzID0gKFxyXG4gIGFycmF5OiBzdHJpbmdbXVtdLFxyXG4gIGtleUluZGV4OiBudW1iZXIsXHJcbiAgdmFsdWVzSW5kaWNlczogbnVtYmVyW11cclxuKTogc3RyaW5nW11bXSA9PiB7XHJcbiAgY29uc3Qgb2JqZWN0ID0gYXJyYXkucmVkdWNlKChhY2M6IG9iamVjdCwgZGF0YTogc3RyaW5nW10pID0+IHtcclxuICAgIGlmIChkYXRhW2tleUluZGV4XSkge1xyXG4gICAgICBhY2NbZGF0YVtrZXlJbmRleF1dID0gdmFsdWVzSW5kaWNlcy5tYXAoKGluZGV4OiBudW1iZXIpID0+XHJcbiAgICAgICAgbm9uRW1wdHlPck51bGwoZGF0YVtpbmRleF0pXHJcbiAgICAgICk7XHJcbiAgICB9XHJcbiAgICByZXR1cm4gYWNjO1xyXG4gIH0sIHt9KTtcclxuXHJcbiAgcmV0dXJuIE9iamVjdC5rZXlzKG9iamVjdCkubWFwKChrZXkpID0+IFtrZXksIC4uLm9iamVjdFtrZXldXSk7XHJcbn07XHJcbiJdfQ==