UNPKG

@aws-cdk/integ-runner

Version:

CDK Integration Testing Tool

266 lines 35.4 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.IntegrationTests = exports.IntegTest = void 0; const path = require("path"); const fs = require("fs-extra"); const CDK_OUTDIR_PREFIX = 'cdk-integ.out'; /** * Derived information for IntegTests */ class IntegTest { info; /** * The name of the file to run * * Path is relative to the current working directory. */ fileName; /** * Relative path to the file to run * * Relative from the "discovery root". */ discoveryRelativeFileName; /** * The absolute path to the file */ absoluteFileName; /** * The normalized name of the test. This name * will be the same regardless of what directory the tool * is run from. */ normalizedTestName; /** * Directory the test is in */ directory; /** * Display name for the test * * Depends on the discovery directory. * * Looks like `integ.mytest` or `package/test/integ.mytest`. */ testName; /** * Path of the snapshot directory for this test */ snapshotDir; /** * Path to the temporary output directory for this test */ temporaryOutputDir; /** * The CLI command used to run this test. * If it contains {filePath}, the test file names will be substituted at that place in the command for each run. * * @default - test run command will be `node {filePath}` */ appCommand; constructor(info) { this.info = info; this.appCommand = info.appCommand ?? 'node {filePath}'; // for consistency, always run the CDK apps under test from the CWD // this is especially important for languages that use the CWD to discover assets // @see https://github.com/aws/aws-cdk-cli/issues/638 this.directory = process.cwd(); this.absoluteFileName = path.resolve(info.fileName); this.fileName = path.relative(this.directory, info.fileName); this.discoveryRelativeFileName = path.relative(info.discoveryRoot, info.fileName); // We treat the discovery root as the base for display names // Looks either like `integ.mytest` or `package/test/integ.mytest` const parsed = path.parse(this.fileName); this.testName = path.join(path.relative(this.info.discoveryRoot, parsed.dir), parsed.name); this.normalizedTestName = parsed.name; this.snapshotDir = path.join(parsed.dir, `${parsed.base}.snapshot`); this.temporaryOutputDir = path.join(parsed.dir, `${CDK_OUTDIR_PREFIX}.${parsed.base}.snapshot`); } /** * Whether this test matches the user-given name * * We are very lenient here. A name matches if it matches: * * - The CWD-relative filename * - The discovery root-relative filename * - The suite name * - The absolute filename */ matches(name) { return [ this.fileName, this.discoveryRelativeFileName, this.testName, this.absoluteFileName, ].includes(name); } } exports.IntegTest = IntegTest; /** * Returns the name of the Python executable for the current OS */ function pythonExecutable() { let python = 'python3'; if (process.platform === 'win32') { python = 'python'; } return python; } /** * Discover integration tests */ class IntegrationTests { directory; constructor(directory) { this.directory = directory; } /** * Get integration tests discovery options from CLI options */ async fromCliOptions(options) { const baseOptions = { tests: options.tests, exclude: options.exclude, strict: options.strict, }; // Explicitly set both, app and test-regex if (options.app && options.testRegex) { return this.discover({ testCases: { [options.app]: options.testRegex, }, ...baseOptions, }); } // Use the selected presets if (!options.app && !options.testRegex) { // Only case with multiple languages, i.e. the only time we need to check the special case const ignoreUncompiledTypeScript = options.language?.includes('javascript') && options.language?.includes('typescript'); return this.discover({ testCases: this.getLanguagePresets(options.language), ...baseOptions, }, ignoreUncompiledTypeScript); } // Only one of app or test-regex is set, with a single preset selected // => override either app or test-regex if (options.language?.length === 1) { const [presetApp, presetTestRegex] = this.getLanguagePreset(options.language[0]); return this.discover({ testCases: { [options.app ?? presetApp]: options.testRegex ?? presetTestRegex, }, ...baseOptions, }); } // Only one of app or test-regex is set, with multiple presets // => impossible to resolve const option = options.app ? '--app' : '--test-regex'; throw new Error(`Only a single "--language" can be used with "${option}". Alternatively provide both "--app" and "--test-regex" to fully customize the configuration.`); } /** * Get the default configuration for a language */ getLanguagePreset(language) { const languagePresets = { javascript: ['node {filePath}', ['^integ\\..*\\.js$']], typescript: ['node -r ts-node/register {filePath}', ['^integ\\.(?!.*\\.d\\.ts$).*\\.ts$']], python: [`${pythonExecutable()} {filePath}`, ['^integ_.*\\.py$']], go: ['go run {filePath}', ['^integ_.*\\.go$']], }; return languagePresets[language]; } /** * Get the config for all selected languages */ getLanguagePresets(languages = []) { return Object.fromEntries(languages .map(language => this.getLanguagePreset(language)) .filter(Boolean)); } /** * If the user provides a list of tests, these can either be a list of tests to include or a list of tests to exclude. * * - If it is a list of tests to include then we discover all available tests and check whether they have provided valid tests. * If they have provided a test name that we don't find, then we write out that error message. * - If it is a list of tests to exclude, then we discover all available tests and filter out the tests that were provided by the user. */ filterTests(discoveredTests, requestedTests, exclude, strict) { if (!requestedTests) { return discoveredTests; } const allTests = discoveredTests.filter(t => { const matches = requestedTests.some(pattern => t.matches(pattern)); return matches !== !!exclude; // Looks weird but is equal to (matches && !exclude) || (!matches && exclude) }); // If not excluding, all patterns must have matched at least one test if (!exclude) { const unmatchedPatterns = requestedTests.filter(pattern => !discoveredTests.some(t => t.matches(pattern))); for (const unmatched of unmatchedPatterns) { process.stderr.write(`No such integ test: ${unmatched}\n`); } if (unmatchedPatterns.length > 0) { process.stderr.write(`Available tests: ${discoveredTests.map(t => t.discoveryRelativeFileName).join(' ')}\n`); if (strict) { throw new Error(`Strict mode: ${unmatchedPatterns.length} test(s) not found: ${unmatchedPatterns.join(', ')}`); } return []; } } return allTests; } /** * Takes an optional list of tests to look for, otherwise * it will look for all tests from the directory * * @param tests - Tests to include or exclude, undefined means include all tests. * @param exclude - Whether the 'tests' list is inclusive or exclusive (inclusive by default). */ async discover(options, ignoreUncompiledTypeScript = false) { const files = await this.readTree(); const testCases = Object.entries(options.testCases) .flatMap(([appCommand, patterns]) => files .filter(fileName => patterns.some((pattern) => { const regex = new RegExp(pattern); return regex.test(fileName) || regex.test(path.basename(fileName)); })) .map(fileName => new IntegTest({ discoveryRoot: this.directory, fileName, appCommand, }))); const discoveredTests = ignoreUncompiledTypeScript ? this.filterUncompiledTypeScript(testCases) : testCases; return this.filterTests(discoveredTests, options.tests, options.exclude, options.strict); } filterUncompiledTypeScript(testCases) { const jsTestCases = testCases.filter(t => t.fileName.endsWith('.js')); return testCases // Remove all TypeScript test cases (ending in .ts) // for which a compiled version is present (same name, ending in .js) .filter((tsCandidate) => { if (!tsCandidate.fileName.endsWith('.ts')) { return true; } return jsTestCases.findIndex(jsTest => jsTest.testName === tsCandidate.testName) === -1; }); } async readTree() { const ret = new Array(); async function recurse(dir) { const files = await fs.readdir(dir); for (const file of files) { const fullPath = path.join(dir, file); const statf = await fs.stat(fullPath); if (statf.isFile()) { ret.push(fullPath); } if (statf.isDirectory()) { await recurse(fullPath); } } } await recurse(this.directory); return ret; } } exports.IntegrationTests = IntegrationTests; //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZWdyYXRpb24tdGVzdHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbnRlZ3JhdGlvbi10ZXN0cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw2QkFBNkI7QUFDN0IsK0JBQStCO0FBRS9CLE1BQU0saUJBQWlCLEdBQUcsZUFBZSxDQUFDO0FBdUMxQzs7R0FFRztBQUNILE1BQWEsU0FBUztJQTJEUTtJQTFENUI7Ozs7T0FJRztJQUNhLFFBQVEsQ0FBUztJQUVqQzs7OztPQUlHO0lBQ2EseUJBQXlCLENBQVM7SUFFbEQ7O09BRUc7SUFDYSxnQkFBZ0IsQ0FBUztJQUV6Qzs7OztPQUlHO0lBQ2Esa0JBQWtCLENBQVM7SUFFM0M7O09BRUc7SUFDYSxTQUFTLENBQVM7SUFFbEM7Ozs7OztPQU1HO0lBQ2EsUUFBUSxDQUFTO0lBRWpDOztPQUVHO0lBQ2EsV0FBVyxDQUFTO0lBRXBDOztPQUVHO0lBQ2Esa0JBQWtCLENBQVM7SUFFM0M7Ozs7O09BS0c7SUFDTSxVQUFVLENBQVM7SUFFNUIsWUFBNEIsSUFBbUI7UUFBbkIsU0FBSSxHQUFKLElBQUksQ0FBZTtRQUM3QyxJQUFJLENBQUMsVUFBVSxHQUFHLElBQUksQ0FBQyxVQUFVLElBQUksaUJBQWlCLENBQUM7UUFFdkQsbUVBQW1FO1FBQ25FLGlGQUFpRjtRQUNqRixxREFBcUQ7UUFDckQsSUFBSSxDQUFDLFNBQVMsR0FBRyxPQUFPLENBQUMsR0FBRyxFQUFFLENBQUM7UUFDL0IsSUFBSSxDQUFDLGdCQUFnQixHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQ3BELElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLElBQUksQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUM3RCxJQUFJLENBQUMseUJBQXlCLEdBQUcsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsYUFBYSxFQUFF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QyxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUM7Z0JBQ3RDLE1BQU0sS0FBSyxHQUFHLE1BQU0sRUFBRSxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsQ0FBQztnQkFDdEMsSUFBSSxLQUFLLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQztvQkFDbkIsR0FBRyxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsQ0FBQztnQkFDckIsQ0FBQztnQkFDRCxJQUFJLEtBQUssQ0FBQyxXQUFXLEVBQUUsRUFBRSxDQUFDO29CQUN4QixNQUFNLE9BQU8sQ0FBQyxRQUFRLENBQUMsQ0FBQztnQkFDMUIsQ0FBQztZQUNILENBQUM7UUFDSCxDQUFDO1FBRUQsTUFBTSxPQUFPLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBQzlCLE9BQU8sR0FBRyxDQUFDO0lBQ2IsQ0FBQztDQUNGO0FBeExELDRDQXdMQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIHBhdGggZnJvbSAncGF0aCc7XG5pbXBvcnQgKiBhcyBmcyBmcm9tICdmcy1leHRyYSc7XG5cbmNvbnN0IENES19PVVRESVJfUFJFRklYID0gJ2Nkay1pbnRlZy5vdXQnO1xuXG4vKipcbiAqIFJlcHJlc2VudHMgYSBzaW5nbGUgaW50ZWdyYXRpb24gdGVzdFxuICpcbiAqIFRoaXMgdHlwZSBpcyBhIGRhdGEtb25seSBzdHJ1Y3R1cmUsIHNvIGl0IGNhbiB0cml2aWFsbHkgYmUgcGFzc2VkIHRvIHdvcmtlcnMuXG4gKiBEZXJpdmVkIGF0dHJpYnV0ZXMgYXJlIGNhbGN1bGF0ZWQgdXNpbmcgdGhlIGBJbnRlZ1Rlc3RgIGNsYXNzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEludGVnVGVzdEluZm8ge1xuICAvKipcbiAgICogUGF0aCB0byB0aGUgZmlsZSB0byBydW5cbiAgICpcbiAgICogUGF0aCBpcyByZWxhdGl2ZSB0byB0aGUgY3VycmVudCB3b3JraW5nIGRpcmVjdG9yeS5cbiAgICovXG4gIHJlYWRvbmx5IGZpbGVOYW1lOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFRoZSByb290IGRpcmVjdG9yeSB3ZSBkaXNjb3ZlcmVkIHRoaXMgdGVzdCBmcm9tXG4gICAqXG4gICAqIFBhdGggaXMgcmVsYXRpdmUgdG8gdGhlIGN1cnJlbnQgd29ya2luZyBkaXJlY3RvcnkuXG4gICAqL1xuICByZWFkb25seSBkaXNjb3ZlcnlSb290OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFRoZSBDTEkgY29tbWFuZCB1c2VkIHRvIHJ1biB0aGlzIHRlc3QuXG4gICAqIElmIGl0IGNvbnRhaW5zIHtmaWxlUGF0aH0sIHRoZSB0ZXN0IGZpbGUgbmFtZXMgd2lsbCBiZSBzdWJzdGl0dXRlZCBhdCB0aGF0IHBsYWNlIGluIHRoZSBjb21tYW5kIGZvciBlYWNoIHJ1bi5cbiAgICpcbiAgICogQGRlZmF1bHQgLSB0ZXN0IHJ1biBjb21tYW5kIHdpbGwgYmUgYG5vZGUge2ZpbGVQYXRofWBcbiAgICovXG4gIHJlYWRvbmx5IGFwcENvbW1hbmQ/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIHRydWUgaWYgdGhpcyB0ZXN0IGlzIHJ1bm5pbmcgaW4gd2F0Y2ggbW9kZVxuICAgKlxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKi9cbiAgcmVhZG9ubHkgd2F0Y2g/OiBib29sZWFuO1xufVxuXG4vKipcbiAqIERlcml2ZWQgaW5mb3JtYXRpb24gZm9yIEludGVnVGVzdHNcbiAqL1xuZXhwb3J0IGNsYXNzIEludGVnVGVzdCB7XG4gIC8qKlxuICAgKiBUaGUgbmFtZSBvZiB0aGUgZmlsZSB0byBydW5cbiAgICpcbiAgICogUGF0aCBpcyByZWxhdGl2ZSB0byB0aGUgY3VycmVudCB3b3JraW5nIGRpcmVjdG9yeS5cbiAgICovXG4gIHB1YmxpYyByZWFkb25seSBmaWxlTmFtZTogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBSZWxhdGl2ZSBwYXRoIHRvIHRoZSBmaWxlIHRvIHJ1blxuICAgKlxuICAgKiBSZWxhdGl2ZSBmcm9tIHRoZSBcImRpc2NvdmVyeSByb290XCIuXG4gICAqL1xuICBwdWJsaWMgcmVhZG9ubHkgZGlzY292ZXJ5UmVsYXRpdmVGaWxlTmFtZTogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBUaGUgYWJzb2x1dGUgcGF0aCB0byB0aGUgZmlsZVxuICAgKi9cbiAgcHVibGljIHJlYWRvbmx5IGFic29sdXRlRmlsZU5hbWU6IHN0cmluZztcblxuICAvKipcbiAgICogVGhlIG5vcm1hbGl6ZWQgbmFtZSBvZiB0aGUgdGVzdC4gVGhpcyBuYW1lXG4gICAqIHdpbGwgYmUgdGhlIHNhbWUgcmVnYXJkbGVzcyBvZiB3aGF0IGRpcmVjdG9yeSB0aGUgdG9vbFxuICAgKiBpcyBydW4gZnJvbS5cbiAgICovXG4gIHB1YmxpYyByZWFkb25seSBub3JtYWxpemVkVGVzdE5hbWU6IHN0cmluZztcblxuICAvKipcbiAgICogRGlyZWN0b3J5IHRoZSB0ZXN0IGlzIGluXG4gICAqL1xuICBwdWJsaWMgcmVhZG9ubHkgZGlyZWN0b3J5OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIERpc3BsYXkgbmFtZSBmb3IgdGhlIHRlc3RcbiAgICpcbiAgICogRGVwZW5kcyBvbiB0aGUgZGlzY292ZXJ5IGRpcmVjdG9yeS5cbiAgICpcbiAgICogTG9va3MgbGlrZSBgaW50ZWcubXl0ZXN0YCBvciBgcGFja2FnZS90ZXN0L2ludGVnLm15dGVzdGAuXG4gICAqL1xuICBwdWJsaWMgcmVhZG9ubHkgdGVzdE5hbWU6IHN0cmluZztcblxuICAvKipcbiAgICogUGF0aCBvZiB0aGUgc25hcHNob3QgZGlyZWN0b3J5IGZvciB0aGlzIHRlc3RcbiAgICovXG4gIHB1YmxpYyByZWFkb25seSBzbmFwc2hvdERpcjogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBQYXRoIHRvIHRoZSB0ZW1wb3Jhcnkgb3V0cHV0IGRpcmVjdG9yeSBmb3IgdGhpcyB0ZXN0XG4gICAqL1xuICBwdWJsaWMgcmVhZG9ubHkgdGVtcG9yYXJ5T3V0cHV0RGlyOiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFRoZSBDTEkgY29tbWFuZCB1c2VkIHRvIHJ1biB0aGlzIHRlc3QuXG4gICAqIElmIGl0IGNvbnRhaW5zIHtmaWxlUGF0aH0sIHRoZSB0ZXN0IGZpbGUgbmFtZXMgd2lsbCBiZSBzdWJzdGl0dXRlZCBhdCB0aGF0IHBsYWNlIGluIHRoZSBjb21tYW5kIGZvciBlYWNoIHJ1bi5cbiAgICpcbiAgICogQGRlZmF1bHQgLSB0ZXN0IHJ1biBjb21tYW5kIHdpbGwgYmUgYG5vZGUge2ZpbGVQYXRofWBcbiAgICovXG4gIHJlYWRvbmx5IGFwcENvbW1hbmQ6IHN0cmluZztcblxuICBjb25zdHJ1Y3RvcihwdWJsaWMgcmVhZG9ubHkgaW5mbzogSW50ZWdUZXN0SW5mbykge1xuICAgIHRoaXMuYXBwQ29tbWFuZCA9IGluZm8uYXBwQ29tbWFuZCA/PyAnbm9kZSB7ZmlsZVBhdGh9JztcblxuICAgIC8vIGZvciBjb25zaXN0ZW5jeSwgYWx3YXlzIHJ1biB0aGUgQ0RLIGFwcHMgdW5kZXIgdGVzdCBmcm9tIHRoZSBDV0RcbiAgICAvLyB0aGlzIGlzIGVzcGVjaWFsbHkgaW1wb3J0YW50IGZvciBsYW5ndWFnZXMgdGhhdCB1c2UgdGhlIENXRCB0byBkaXNjb3ZlciBhc3NldHNcbiAgICAvLyBAc2VlIGh0dHBzOi8vZ2l0aHViLmNvbS9hd3MvYXdzLWNkay1jbGkvaXNzdWVzLzYzOFxuICAgIHRoaXMuZGlyZWN0b3J5ID0gcHJvY2Vzcy5jd2QoKTtcbiAgICB0aGlzLmFic29sdXRlRmlsZU5hbWUgPSBwYXRoLnJlc29sdmUoaW5mby5maWxlTmFtZSk7XG4gICAgdGhpcy5maWxlTmFtZSA9IHBhdGgucmVsYXRpdmUodGhpcy5kaXJlY3RvcnksIGluZm8uZmlsZU5hbWUpO1xuICAgIHRoaXMuZGlzY292ZXJ5UmVsYXRpdmVGaWxlTmFtZSA9IHBhdGgucmVsYXRpdmUoaW5mby5kaXNjb3ZlcnlSb290LCBpbmZvLmZpbGVOYW1lKTtcblxuICAgIC8vIFdlIHRyZWF0IHRoZSBkaXNjb3Zlcnkgcm9vdCBhcyB0aGUgYmFzZSBmb3IgZGlzcGxheSBuYW1lc1xuICAgIC8vIExvb2tzIGVpdGhlciBsaWtlIGBpbnRlZy5teXRlc3RgIG9yIGBwYWNrYWdlL3Rlc3QvaW50ZWcubXl0ZXN0YFxuICAgIGNvbnN0IHBhcnNlZCA9IHBhdGgucGFyc2UodGhpcy5maWxlTmFtZSk7XG4gICAgdGhpcy50ZXN0TmFtZSA9IHBhdGguam9pbihwYXRoLnJlbGF0aXZlKHRoaXMuaW5mby5kaXNjb3ZlcnlSb290LCBwYXJzZWQuZGlyKSwgcGFyc2VkLm5hbWUpO1xuICAgIHRoaXMubm9ybWFsaXplZFRlc3ROYW1lID0gcGFyc2VkLm5hbWU7XG4gICAgdGhpcy5zbmFwc2hvdERpciA9IHBhdGguam9pbihwYXJzZWQuZGlyLCBgJHtwYXJzZWQuYmFzZX0uc25hcHNob3RgKTtcbiAgICB0aGlzLnRlbXBvcmFyeU91dHB1dERpciA9IHBhdGguam9pbihwYXJzZWQuZGlyLCBgJHtDREtfT1VURElSX1BSRUZJWH0uJHtwYXJzZWQuYmFzZX0uc25hcHNob3RgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBXaGV0aGVyIHRoaXMgdGVzdCBtYXRjaGVzIHRoZSB1c2VyLWdpdmVuIG5hbWVcbiAgICpcbiAgICogV2UgYXJlIHZlcnkgbGVuaWVudCBoZXJlLiBBIG5hbWUgbWF0Y2hlcyBpZiBpdCBtYXRjaGVzOlxuICAgKlxuICAgKiAtIFRoZSBDV0QtcmVsYXRpdmUgZmlsZW5hbWVcbiAgICogLSBUaGUgZGlzY292ZXJ5IHJvb3QtcmVsYXRpdmUgZmlsZW5hbWVcbiAgICogLSBUaGUgc3VpdGUgbmFtZVxuICAgKiAtIFRoZSBhYnNvbHV0ZSBmaWxlbmFtZVxuICAgKi9cbiAgcHVibGljIG1hdGNoZXMobmFtZTogc3RyaW5nKSB7XG4gICAgcmV0dXJuIFtcbiAgICAgIHRoaXMuZmlsZU5hbWUsXG4gICAgICB0aGlzLmRpc2NvdmVyeVJlbGF0aXZlRmlsZU5hbWUsXG4gICAgICB0aGlzLnRlc3ROYW1lLFxuICAgICAgdGhpcy5hYnNvbHV0ZUZpbGVOYW1lLFxuICAgIF0uaW5jbHVkZXMobmFtZSk7XG4gIH1cbn1cblxuLyoqXG4gKiBDb25maWd1cmF0aW9uIG9wdGlvbnMgaG93IGludGVncmF0aW9uIHRlc3QgZmlsZXMgYXJlIGRpc2NvdmVyZWRcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBJbnRlZ3JhdGlvblRlc3RzRGlzY292ZXJ5T3B0aW9ucyB7XG4gIC8qKlxuICAgKiBJZiB0aGlzIGlzIHNldCB0byB0cnVlIHRoZW4gdGhlIGxpc3Qgb2YgdGVzdHNcbiAgICogcHJvdmlkZWQgd2lsbCBiZSBleGNsdWRlZFxuICAgKlxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKi9cbiAgcmVhZG9ubHkgZXhjbHVkZT86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIElmIHRoaXMgaXMgc2V0IHRvIHRydWUsIHRocm93IGFuIGVycm9yIGlmIGFueSBzcGVjaWZpZWQgdGVzdHMgYXJlIG5vdCBmb3VuZFxuICAgKlxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKi9cbiAgcmVhZG9ubHkgc3RyaWN0PzogYm9vbGVhbjtcblxuICAvKipcbiAgICogTGlzdCBvZiB0ZXN0cyB0byBpbmNsdWRlIChvciBleGNsdWRlIGlmIGBleGNsdWRlPXRydWVgKVxuICAgKlxuICAgKiBAZGVmYXVsdCAtIGFsbCBtYXRjaGVkIGZpbGVzXG4gICAqL1xuICByZWFkb25seSB0ZXN0cz86IHN0cmluZ1tdO1xuXG4gIC8qKlxuICAgKiBBIG1hcCBvZiBvZiB0aGUgYXBwIGNvbW1hbmRzIHRvIHJ1biBpbnRlZ3JhdGlvbiB0ZXN0cyB3aXRoLFxuICAgKiBhbmQgdGhlIHJlZ2V4IHBhdHRlcm5zIG1hdGNoaW5nIHRoZSBpbnRlZ3JhdGlvbiB0ZXN0IGZpbGVzIGVhY2ggYXBwIGNvbW1hbmQuXG4gICAqXG4gICAqIElmIHRoZSBhcHAgY29tbWFuZCBjb250YWlucyB7ZmlsZVBhdGh9LCB0aGUgdGVzdCBmaWxlIG5hbWVzIHdpbGwgYmUgc3Vic3RpdHV0ZWQgYXQgdGhhdCBwbGFjZSBpbiB0aGUgY29tbWFuZCBmb3IgZWFjaCBydW4uXG4gICAqL1xuICByZWFkb25seSB0ZXN0Q2FzZXM6IHtcbiAgICBbYXBwOiBzdHJpbmddOiBzdHJpbmdbXTtcbiAgfTtcbn1cblxuLyoqXG4gKiBSZXR1cm5zIHRoZSBuYW1lIG9mIHRoZSBQeXRob24gZXhlY3V0YWJsZSBmb3IgdGhlIGN1cnJlbnQgT1NcbiAqL1xuZnVuY3Rpb24gcHl0aG9uRXhlY3V0YWJsZSgpIHtcbiAgbGV0IHB5dGhvbiA9ICdweXRob24zJztcbiAgaWYgKHByb2Nlc3MucGxhdGZvcm0gPT09ICd3aW4zMicpIHtcbiAgICBweXRob24gPSAncHl0aG9uJztcbiAgfVxuICByZXR1cm4gcHl0aG9uO1xufVxuXG4vKipcbiAqIERpc2NvdmVyIGludGVncmF0aW9uIHRlc3RzXG4gKi9cbmV4cG9ydCBjbGFzcyBJbnRlZ3JhdGlvblRlc3RzIHtcbiAgY29uc3RydWN0b3IocHJpdmF0ZSByZWFkb25seSBkaXJlY3Rvcnk6IHN0cmluZykge1xuICB9XG5cbiAgLyoqXG4gICAqIEdldCBpbnRlZ3JhdGlvbiB0ZXN0cyBkaXNjb3Zlcnkgb3B0aW9ucyBmcm9tIENMSSBvcHRpb25zXG4gICAqL1xuICBwdWJsaWMgYXN5bmMgZnJvbUNsaU9wdGlvbnMob3B0aW9uczoge1xuICAgIGFwcD86IHN0cmluZztcbiAgICBleGNsdWRlPzogYm9vbGVhbjtcbiAgICBsYW5ndWFnZT86IHN0cmluZ1tdO1xuICAgIHRlc3RSZWdleD86IHN0cmluZ1tdO1xuICAgIHRlc3RzPzogc3RyaW5nW107XG4gICAgc3RyaWN0PzogYm9vbGVhbjtcbiAgfSk6IFByb21pc2U8SW50ZWdUZXN0W10+IHtcbiAgICBjb25zdCBiYXNlT3B0aW9ucyA9IHtcbiAgICAgIHRlc3RzOiBvcHRpb25zLnRlc3RzLFxuICAgICAgZXhjbHVkZTogb3B0aW9ucy5leGNsdWRlLFxuICAgICAgc3RyaWN0OiBvcHRpb25zLnN0cmljdCxcbiAgICB9O1xuXG4gICAgLy8gRXhwbGljaXRseSBzZXQgYm90aCwgYXBwIGFuZCB0ZXN0LXJlZ2V4XG4gICAgaWYgKG9wdGlvbnMuYXBwICYmIG9wdGlvbnMudGVzdFJlZ2V4KSB7XG4gICAgICByZXR1cm4gdGhpcy5kaXNjb3Zlcih7XG4gICAgICAgIHRlc3RDYXNlczoge1xuICAgICAgICAgIFtvcHRpb25zLmFwcF06IG9wdGlvbnMudGVzdFJlZ2V4LFxuICAgICAgICB9LFxuICAgICAgICAuLi5iYXNlT3B0aW9ucyxcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIC8vIFVzZSB0aGUgc2VsZWN0ZWQgcHJlc2V0c1xuICAgIGlmICghb3B0aW9ucy5hcHAgJiYgIW9wdGlvbnMudGVzdFJlZ2V4KSB7XG4gICAgICAvLyBPbmx5IGNhc2Ugd2l0aCBtdWx0aXBsZSBsYW5ndWFnZXMsIGkuZS4gdGhlIG9ubHkgdGltZSB3ZSBuZWVkIHRvIGNoZWNrIHRoZSBzcGVjaWFsIGNhc2VcbiAgICAgIGNvbnN0IGlnbm9yZVVuY29tcGlsZWRUeXBlU2NyaXB0ID0gb3B0aW9ucy5sYW5ndWFnZT8uaW5jbHVkZXMoJ2phdmFzY3JpcHQnKSAmJiBvcHRpb25zLmxhbmd1YWdlPy5pbmNsdWRlcygndHlwZXNjcmlwdCcpO1xuXG4gICAgICByZXR1cm4gdGhpcy5kaXNjb3Zlcih7XG4gICAgICAgIHRlc3RDYXNlczogdGhpcy5nZXRMYW5ndWFnZVByZXNldHMob3B0aW9ucy5sYW5ndWFnZSksXG4gICAgICAgIC4uLmJhc2VPcHRpb25zLFxuICAgICAgfSwgaWdub3JlVW5jb21waWxlZFR5cGVTY3JpcHQpO1xuICAgIH1cblxuICAgIC8vIE9ubHkgb25lIG9mIGFwcCBvciB0ZXN0LXJlZ2V4IGlzIHNldCwgd2l0aCBhIHNpbmdsZSBwcmVzZXQgc2VsZWN0ZWRcbiAgICAvLyA9PiBvdmVycmlkZSBlaXRoZXIgYXBwIG9yIHRlc3QtcmVnZXhcbiAgICBpZiAob3B0aW9ucy5sYW5ndWFnZT8ubGVuZ3RoID09PSAxKSB7XG4gICAgICBjb25zdCBbcHJlc2V0QXBwLCBwcmVzZXRUZXN0UmVnZXhdID0gdGhpcy5nZXRMYW5ndWFnZVByZXNldChvcHRpb25zLmxhbmd1YWdlWzBdKTtcbiAgICAgIHJldHVybiB0aGlzLmRpc2NvdmVyKHtcbiAgICAgICAgdGVzdENhc2VzOiB7XG4gICAgICAgICAgW29wdGlvbnMuYXBwID8/IHByZXNldEFwcF06IG9wdGlvbnMudGVzdFJlZ2V4ID8/IHByZXNldFRlc3RSZWdleCxcbiAgICAgICAgfSxcbiAgICAgICAgLi4uYmFzZU9wdGlvbnMsXG4gICAgICB9KTtcbiAgICB9XG5cbiAgICAvLyBPbmx5IG9uZSBvZiBhcHAgb3IgdGVzdC1yZWdleCBpcyBzZXQsIHdpdGggbXVsdGlwbGUgcHJlc2V0c1xuICAgIC8vID0+IGltcG9zc2libGUgdG8gcmVzb2x2ZVxuICAgIGNvbnN0IG9wdGlvbiA9IG9wdGlvbnMuYXBwID8gJy0tYXBwJyA6ICctLXRlc3QtcmVnZXgnO1xuICAgIHRocm93IG5ldyBFcnJvcihgT25seSBhIHNpbmdsZSBcIi0tbGFuZ3VhZ2VcIiBjYW4gYmUgdXNlZCB3aXRoIFwiJHtvcHRpb259XCIuIEFsdGVybmF0aXZlbHkgcHJvdmlkZSBib3RoIFwiLS1hcHBcIiBhbmQgXCItLXRlc3QtcmVnZXhcIiB0byBmdWxseSBjdXN0b21pemUgdGhlIGNvbmZpZ3VyYXRpb24uYCk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IHRoZSBkZWZhdWx0IGNvbmZpZ3VyYXRpb24gZm9yIGEgbGFuZ3VhZ2VcbiAgICovXG4gIHByaXZhdGUgZ2V0TGFuZ3VhZ2VQcmVzZXQobGFuZ3VhZ2U6IHN0cmluZykge1xuICAgIGNvbnN0IGxhbmd1YWdlUHJlc2V0czoge1xuICAgICAgW2xhbmd1YWdlOiBzdHJpbmddOiBbc3RyaW5nLCBzdHJpbmdbXV07XG4gICAgfSA9IHtcbiAgICAgIGphdmFzY3JpcHQ6IFsnbm9kZSB7ZmlsZVBhdGh9JywgWydeaW50ZWdcXFxcLi4qXFxcXC5qcyQnXV0sXG4gICAgICB0eXBlc2NyaXB0OiBbJ25vZGUgLXIgdHMtbm9kZS9yZWdpc3RlciB7ZmlsZVBhdGh9JywgWydeaW50ZWdcXFxcLig/IS4qXFxcXC5kXFxcXC50cyQpLipcXFxcLnRzJCddXSxcbiAgICAgIHB5dGhvbjogW2Ake3B5dGhvbkV4ZWN1dGFibGUoKX0ge2ZpbGVQYXRofWAsIFsnXmludGVnXy4qXFxcXC5weSQnXV0sXG4gICAgICBnbzogWydnbyBydW4ge2ZpbGVQYXRofScsIFsnXmludGVnXy4qXFxcXC5nbyQnXV0sXG4gICAgfTtcblxuICAgIHJldHVybiBsYW5ndWFnZVByZXNldHNbbGFuZ3VhZ2VdO1xuICB9XG5cbiAgLyoqXG4gICAqIEdldCB0aGUgY29uZmlnIGZvciBhbGwgc2VsZWN0ZWQgbGFuZ3VhZ2VzXG4gICAqL1xuICBwcml2YXRlIGdldExhbmd1YWdlUHJlc2V0cyhsYW5ndWFnZXM6IHN0cmluZ1tdID0gW10pIHtcbiAgICByZXR1cm4gT2JqZWN0LmZyb21FbnRyaWVzKFxuICAgICAgbGFuZ3VhZ2VzXG4gICAgICAgIC5tYXAobGFuZ3VhZ2UgPT4gdGhpcy5nZXRMYW5ndWFnZVByZXNldChsYW5ndWFnZSkpXG4gICAgICAgIC5maWx0ZXIoQm9vbGVhbiksXG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBJZiB0aGUgdXNlciBwcm92aWRlcyBhIGxpc3Qgb2YgdGVzdHMsIHRoZXNlIGNhbiBlaXRoZXIgYmUgYSBsaXN0IG9mIHRlc3RzIHRvIGluY2x1ZGUgb3IgYSBsaXN0IG9mIHRlc3RzIHRvIGV4Y2x1ZGUuXG4gICAqXG4gICAqIC0gSWYgaXQgaXMgYSBsaXN0IG9mIHRlc3RzIHRvIGluY2x1ZGUgdGhlbiB3ZSBkaXNjb3ZlciBhbGwgYXZhaWxhYmxlIHRlc3RzIGFuZCBjaGVjayB3aGV0aGVyIHRoZXkgaGF2ZSBwcm92aWRlZCB2YWxpZCB0ZXN0cy5cbiAgICogICBJZiB0aGV5IGhhdmUgcHJvdmlkZWQgYSB0ZXN0IG5hbWUgdGhhdCB3ZSBkb24ndCBmaW5kLCB0aGVuIHdlIHdyaXRlIG91dCB0aGF0IGVycm9yIG1lc3NhZ2UuXG4gICAqIC0gSWYgaXQgaXMgYSBsaXN0IG9mIHRlc3RzIHRvIGV4Y2x1ZGUsIHRoZW4gd2UgZGlzY292ZXIgYWxsIGF2YWlsYWJsZSB0ZXN0cyBhbmQgZmlsdGVyIG91dCB0aGUgdGVzdHMgdGhhdCB3ZXJlIHByb3ZpZGVkIGJ5IHRoZSB1c2VyLlxuICAgKi9cbiAgcHJpdmF0ZSBmaWx0ZXJUZXN0cyhkaXNjb3ZlcmVkVGVzdHM6IEludGVnVGVzdFtdLCByZXF1ZXN0ZWRUZXN0cz86IHN0cmluZ1tdLCBleGNsdWRlPzogYm9vbGVhbiwgc3RyaWN0PzogYm9vbGVhbik6IEludGVnVGVzdFtdIHtcbiAgICBpZiAoIXJlcXVlc3RlZFRlc3RzKSB7XG4gICAgICByZXR1cm4gZGlzY292ZXJlZFRlc3RzO1xuICAgIH1cblxuICAgIGNvbnN0IGFsbFRlc3RzID0gZGlzY292ZXJlZFRlc3RzLmZpbHRlcih0ID0+IHtcbiAgICAgIGNvbnN0IG1hdGNoZXMgPSByZXF1ZXN0ZWRUZXN0cy5zb21lKHBhdHRlcm4gPT4gdC5tYXRjaGVzKHBhdHRlcm4pKTtcbiAgICAgIHJldHVybiBtYXRjaGVzICE9PSAhIWV4Y2x1ZGU7IC8vIExvb2tzIHdlaXJkIGJ1dCBpcyBlcXVhbCB0byAobWF0Y2hlcyAmJiAhZXhjbHVkZSkgfHwgKCFtYXRjaGVzICYmIGV4Y2x1ZGUpXG4gICAgfSk7XG5cbiAgICAvLyBJZiBub3QgZXhjbHVkaW5nLCBhbGwgcGF0dGVybnMgbXVzdCBoYXZlIG1hdGNoZWQgYXQgbGVhc3Qgb25lIHRlc3RcbiAgICBpZiAoIWV4Y2x1ZGUpIHtcbiAgICAgIGNvbnN0IHVubWF0Y2hlZFBhdHRlcm5zID0gcmVxdWVzdGVkVGVzdHMuZmlsdGVyKHBhdHRlcm4gPT4gIWRpc2NvdmVyZWRUZXN0cy5zb21lKHQgPT4gdC5tYXRjaGVzKHBhdHRlcm4pKSk7XG4gICAgICBmb3IgKGNvbnN0IHVubWF0Y2hlZCBvZiB1bm1hdGNoZWRQYXR0ZXJucykge1xuICAgICAgICBwcm9jZXNzLnN0ZGVyci53cml0ZShgTm8gc3VjaCBpbnRlZyB0ZXN0OiAke3VubWF0Y2hlZH1cXG5gKTtcbiAgICAgIH1cbiAgICAgIGlmICh1bm1hdGNoZWRQYXR0ZXJucy5sZW5ndGggPiAwKSB7XG4gICAgICAgIHByb2Nlc3Muc3RkZXJyLndyaXRlKGBBdmFpbGFibGUgdGVzdHM6ICR7ZGlzY292ZXJlZFRlc3RzLm1hcCh0ID0+IHQuZGlzY292ZXJ5UmVsYXRpdmVGaWxlTmFtZSkuam9pbignICcpfVxcbmApO1xuICAgICAgICBpZiAoc3RyaWN0KSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBTdHJpY3QgbW9kZTogJHt1bm1hdGNoZWRQYXR0ZXJucy5sZW5ndGh9IHRlc3Qocykgbm90IGZvdW5kOiAke3VubWF0Y2hlZFBhdHRlcm5zLmpvaW4oJywgJyl9YCk7XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIFtdO1xuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiBhbGxUZXN0cztcbiAgfVxuXG4gIC8qKlxuICAgKiBUYWtlcyBhbiBvcHRpb25hbCBsaXN0IG9mIHRlc3RzIHRvIGxvb2sgZm9yLCBvdGhlcndpc2VcbiAgICogaXQgd2lsbCBsb29rIGZvciBhbGwgdGVzdHMgZnJvbSB0aGUgZGlyZWN0b3J5XG4gICAqXG4gICAqIEBwYXJhbSB0ZXN0cyAtIFRlc3RzIHRvIGluY2x1ZGUgb3IgZXhjbHVkZSwgdW5kZWZpbmVkIG1lYW5zIGluY2x1ZGUgYWxsIHRlc3RzLlxuICAgKiBAcGFyYW0gZXhjbHVkZSAtIFdoZXRoZXIgdGhlICd0ZXN0cycgbGlzdCBpcyBpbmNsdXNpdmUgb3IgZXhjbHVzaXZlIChpbmNsdXNpdmUgYnkgZGVmYXVsdCkuXG4gICAqL1xuICBwcml2YXRlIGFzeW5jIGRpc2NvdmVyKG9wdGlvbnM6IEludGVncmF0aW9uVGVzdHNEaXNjb3ZlcnlPcHRpb25zLCBpZ25vcmVVbmNvbXBpbGVkVHlwZVNjcmlwdDogYm9vbGVhbiA9IGZhbHNlKTogUHJvbWlzZTxJbnRlZ1Rlc3RbXT4ge1xuICAgIGNvbnN0IGZpbGVzID0gYXdhaXQgdGhpcy5yZWFkVHJlZSgpO1xuXG4gICAgY29uc3QgdGVzdENhc2VzID0gT2JqZWN0LmVudHJpZXMob3B0aW9ucy50ZXN0Q2FzZXMpXG4gICAgICAuZmxhdE1hcCgoW2FwcENvbW1hbmQsIHBhdHRlcm5zXSkgPT4gZmlsZXNcbiAgICAgICAgLmZpbHRlcihmaWxlTmFtZSA9PiBwYXR0ZXJucy5zb21lKChwYXR0ZXJuKSA9PiB7XG4gICAgICAgICAgY29uc3QgcmVnZXggPSBuZXcgUmVnRXhwKHBhdHRlcm4pO1xuICAgICAgICAgIHJldHVybiByZWdleC50ZXN0KGZpbGVOYW1lKSB8fCByZWdleC50ZXN0KHBhdGguYmFzZW5hbWUoZmlsZU5hbWUpKTtcbiAgICAgICAgfSkpXG4gICAgICAgIC5tYXAoZmlsZU5hbWUgPT4gbmV3IEludGVnVGVzdCh7XG4gICAgICAgICAgZGlzY292ZXJ5Um9vdDogdGhpcy5kaXJlY3RvcnksXG4gICAgICAgICAgZmlsZU5hbWUsXG4gICAgICAgICAgYXBwQ29tbWFuZCxcbiAgICAgICAgfSkpLFxuICAgICAgKTtcblxuICAgIGNvbnN0IGRpc2NvdmVyZWRUZXN0cyA9IGlnbm9yZVVuY29tcGlsZWRUeXBlU2NyaXB0ID8gdGhpcy5maWx0ZXJVbmNvbXBpbGVkVHlwZVNjcmlwdCh0ZXN0Q2FzZXMpIDogdGVzdENhc2VzO1xuXG4gICAgcmV0dXJuIHRoaXMuZmlsdGVyVGVzdHMoZGlzY292ZXJlZFRlc3RzLCBvcHRpb25zLnRlc3RzLCBvcHRpb25zLmV4Y2x1ZGUsIG9wdGlvbnMuc3RyaWN0KTtcbiAgfVxuXG4gIHByaXZhdGUgZmlsdGVyVW5jb21waWxlZFR5cGVTY3JpcHQodGVzdENhc2VzOiBJbnRlZ1Rlc3RbXSk6IEludGVnVGVzdFtdIHtcbiAgICBjb25zdCBqc1Rlc3RDYXNlcyA9IHRlc3RDYXNlcy5maWx0ZXIodCA9PiB0LmZpbGVOYW1lLmVuZHNXaXRoKCcuanMnKSk7XG5cbiAgICByZXR1cm4gdGVzdENhc2VzXG4gICAgICAvLyBSZW1vdmUgYWxsIFR5cGVTY3JpcHQgdGVzdCBjYXNlcyAoZW5kaW5nIGluIC50cylcbiAgICAgIC8vIGZvciB3aGljaCBhIGNvbXBpbGVkIHZlcnNpb24gaXMgcHJlc2VudCAoc2FtZSBuYW1lLCBlbmRpbmcgaW4gLmpzKVxuICAgICAgLmZpbHRlcigodHNDYW5kaWRhdGUpID0+IHtcbiAgICAgICAgaWYgKCF0c0NhbmRpZGF0ZS5maWxlTmFtZS5lbmRzV2l0aCgnLnRzJykpIHtcbiAgICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4ganNUZXN0Q2FzZXMuZmluZEluZGV4KGpzVGVzdCA9PiBqc1Rlc3QudGVzdE5hbWUgPT09IHRzQ2FuZGlkYXRlLnRlc3ROYW1lKSA9PT0gLTE7XG4gICAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgYXN5bmMgcmVhZFRyZWUoKTogUHJvbWlzZTxzdHJpbmdbXT4ge1xuICAgIGNvbnN0IHJldCA9IG5ldyBBcnJheTxzdHJpbmc+KCk7XG5cbiAgICBhc3luYyBmdW5jdGlvbiByZWN1cnNlKGRpcjogc3RyaW5nKSB7XG4gICAgICBjb25zdCBmaWxlcyA9IGF3YWl0IGZzLnJlYWRkaXIoZGlyKTtcbiAgICAgIGZvciAoY29uc3QgZmlsZSBvZiBmaWxlcykge1xuICAgICAgICBjb25zdCBmdWxsUGF0aCA9IHBhdGguam9pbihkaXIsIGZpbGUpO1xuICAgICAgICBjb25zdCBzdGF0ZiA9IGF3YWl0IGZzLnN0YXQoZnVsbFBhdGgpO1xuICAgICAgICBpZiAoc3RhdGYuaXNGaWxlKCkpIHtcbiAgICAgICAgICByZXQucHVzaChmdWxsUGF0aCk7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKHN0YXRmLmlzRGlyZWN0b3J5KCkpIHtcbiAgICAgICAgICBhd2FpdCByZWN1cnNlKGZ1bGxQYXRoKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIGF3YWl0IHJlY3Vyc2UodGhpcy5kaXJlY3RvcnkpO1xuICAgIHJldHVybiByZXQ7XG4gIH1cbn1cbiJdfQ==