@nlabs/lex 1.46.2 → 1.47.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/__mocks__/LexConfig.js +20 -0
- package/__mocks__/boxen.js +7 -0
- package/__mocks__/build.js +16 -0
- package/__mocks__/chalk.js +23 -0
- package/__mocks__/compile.js +8 -0
- package/__mocks__/execa.js +21 -0
- package/__mocks__/ora.js +17 -0
- package/__mocks__/versions.js +12 -0
- package/dist/LexConfig.js +72 -14
- package/dist/commands/ai/ai.js +303 -0
- package/dist/commands/ai/index.js +7 -0
- package/dist/commands/build/build.js +350 -0
- package/dist/commands/clean/clean.js +31 -0
- package/dist/commands/compile/compile.js +195 -0
- package/dist/commands/config/config.js +43 -0
- package/dist/commands/copy/copy.js +38 -0
- package/dist/commands/create/create.js +124 -0
- package/dist/commands/dev/dev.js +70 -0
- package/dist/commands/init/init.js +93 -0
- package/dist/commands/link/link.js +15 -0
- package/dist/commands/lint/lint.js +656 -0
- package/dist/commands/migrate/migrate.js +37 -0
- package/dist/commands/publish/publish.js +104 -0
- package/dist/commands/test/test.js +327 -0
- package/dist/commands/update/update.js +62 -0
- package/dist/commands/upgrade/upgrade.js +47 -0
- package/dist/commands/versions/versions.js +41 -0
- package/dist/create/changelog.js +3 -3
- package/dist/index.js +35 -0
- package/dist/jest.config.lex.d.ts +2 -0
- package/dist/lex.js +25 -22
- package/dist/types.js +1 -0
- package/dist/utils/aiService.js +290 -0
- package/dist/utils/app.js +3 -3
- package/dist/utils/file.js +1 -1
- package/dist/utils/log.js +2 -1
- package/dist/utils/reactShim.js +3 -3
- package/dist/webpack.config.d.ts +2 -0
- package/eslint.config.js +10 -0
- package/index.cjs +20 -0
- package/jest.config.cjs +31 -27
- package/jest.config.lex.js +90 -38
- package/jest.setup.js +5 -0
- package/lex.config.js +50 -0
- package/package.json +69 -52
- package/{.postcssrc.js → postcss.config.js} +21 -9
- package/tsconfig.json +2 -1
- package/webpack.config.js +27 -11
- package/dist/commands/build.js +0 -265
- package/dist/commands/bulid.test.js +0 -317
- package/dist/commands/clean.js +0 -31
- package/dist/commands/clean.test.js +0 -63
- package/dist/commands/compile.js +0 -195
- package/dist/commands/compile.test.js +0 -93
- package/dist/commands/config.js +0 -43
- package/dist/commands/copy.js +0 -38
- package/dist/commands/create.js +0 -120
- package/dist/commands/dev.js +0 -70
- package/dist/commands/init.js +0 -93
- package/dist/commands/link.js +0 -15
- package/dist/commands/lint.js +0 -179
- package/dist/commands/migrate.js +0 -37
- package/dist/commands/publish.js +0 -104
- package/dist/commands/test.js +0 -190
- package/dist/commands/update.js +0 -64
- package/dist/commands/upgrade.js +0 -47
- package/dist/commands/versions.js +0 -41
- package/dist/commands/versions.test.js +0 -49
- package/dist/lint.js +0 -11
- package/jest.setup.ts +0 -3
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { execa } from "execa";
|
|
2
|
+
import { LexConfig } from "../../LexConfig.js";
|
|
3
|
+
import { createSpinner, getPackageJson, removeConflictModules, removeModules } from "../../utils/app.js";
|
|
4
|
+
import { log } from "../../utils/log.js";
|
|
5
|
+
const migrate = async (cmd, callback = process.exit) => {
|
|
6
|
+
const { cliName = "Lex", packageManager: cmdPackageManager, quiet } = cmd;
|
|
7
|
+
const cwd = process.cwd();
|
|
8
|
+
const spinner = createSpinner(quiet);
|
|
9
|
+
spinner.start("Removing node modules...");
|
|
10
|
+
await removeModules();
|
|
11
|
+
const { packageManager: configPackageManager } = LexConfig.config;
|
|
12
|
+
const packageManager = cmdPackageManager || configPackageManager;
|
|
13
|
+
const packagePath = `${cwd}/package.json`;
|
|
14
|
+
const appPackage = getPackageJson(packagePath);
|
|
15
|
+
const { dependencies = {}, devDependencies = {} } = appPackage;
|
|
16
|
+
appPackage.dependencies = removeConflictModules(dependencies);
|
|
17
|
+
appPackage.devDependencies = removeConflictModules(devDependencies);
|
|
18
|
+
try {
|
|
19
|
+
await execa(packageManager, ["install"], {
|
|
20
|
+
encoding: "utf8",
|
|
21
|
+
stdio: "inherit"
|
|
22
|
+
});
|
|
23
|
+
spinner.succeed("Successfully migrated app!");
|
|
24
|
+
callback(0);
|
|
25
|
+
return 0;
|
|
26
|
+
} catch (error) {
|
|
27
|
+
log(`
|
|
28
|
+
${cliName} Error: ${error.message}`, "error", quiet);
|
|
29
|
+
spinner.fail("Failed to remove modules.");
|
|
30
|
+
callback(1);
|
|
31
|
+
return 1;
|
|
32
|
+
}
|
|
33
|
+
};
|
|
34
|
+
export {
|
|
35
|
+
migrate
|
|
36
|
+
};
|
|
37
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2NvbW1hbmRzL21pZ3JhdGUvbWlncmF0ZS50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTgtUHJlc2VudCwgTml0cm9nZW4gTGFicywgSW5jLlxuICogQ29weXJpZ2h0cyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSB0aGUgYWNjb21wYW55aW5nIExJQ0VOU0UgZmlsZSBmb3IgdGVybXMuXG4gKi9cbmltcG9ydCB7ZXhlY2F9IGZyb20gJ2V4ZWNhJztcblxuaW1wb3J0IHtMZXhDb25maWd9IGZyb20gJy4uLy4uL0xleENvbmZpZy5qcyc7XG5pbXBvcnQge2NyZWF0ZVNwaW5uZXIsIGdldFBhY2thZ2VKc29uLCByZW1vdmVDb25mbGljdE1vZHVsZXMsIHJlbW92ZU1vZHVsZXN9IGZyb20gJy4uLy4uL3V0aWxzL2FwcC5qcyc7XG5pbXBvcnQge2xvZ30gZnJvbSAnLi4vLi4vdXRpbHMvbG9nLmpzJztcblxuZXhwb3J0IGludGVyZmFjZSBNaWdyYXRlT3B0aW9ucyB7XG4gIHJlYWRvbmx5IGNsaU5hbWU/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IHBhY2thZ2VNYW5hZ2VyPzogc3RyaW5nO1xuICByZWFkb25seSBxdWlldD86IGJvb2xlYW47XG59XG5cbmV4cG9ydCB0eXBlIE1pZ3JhdGVDYWxsYmFjayA9IHR5cGVvZiBwcm9jZXNzLmV4aXQ7XG5cbmV4cG9ydCBjb25zdCBtaWdyYXRlID0gYXN5bmMgKGNtZDogTWlncmF0ZU9wdGlvbnMsIGNhbGxiYWNrOiBNaWdyYXRlQ2FsbGJhY2sgPSBwcm9jZXNzLmV4aXQpOiBQcm9taXNlPG51bWJlcj4gPT4ge1xuICBjb25zdCB7Y2xpTmFtZSA9ICdMZXgnLCBwYWNrYWdlTWFuYWdlcjogY21kUGFja2FnZU1hbmFnZXIsIHF1aWV0fSA9IGNtZDtcblxuICBjb25zdCBjd2Q6IHN0cmluZyA9IHByb2Nlc3MuY3dkKCk7XG5cbiAgLy8gLy8gRGlzcGxheSBtZXNzYWdlXG4gIC8vIGxvZyhgJHtjbGlOYW1lfSBjb3B5aW5nIFwiJHt0b31cIi4uLmAsICdpbmZvJywgcXVpZXQpO1xuXG4gIC8vIFNwaW5uZXJcbiAgY29uc3Qgc3Bpbm5lciA9IGNyZWF0ZVNwaW5uZXIocXVpZXQpO1xuICBzcGlubmVyLnN0YXJ0KCdSZW1vdmluZyBub2RlIG1vZHVsZXMuLi4nKTtcblxuICAvLyBSZW1vdmUgbm9kZV9tb2R1bGVzXG4gIGF3YWl0IHJlbW92ZU1vZHVsZXMoKTtcblxuICBjb25zdCB7cGFja2FnZU1hbmFnZXI6IGNvbmZpZ1BhY2thZ2VNYW5hZ2VyfSA9IExleENvbmZpZy5jb25maWc7XG4gIGNvbnN0IHBhY2thZ2VNYW5hZ2VyOiBzdHJpbmcgPSBjbWRQYWNrYWdlTWFuYWdlciB8fCBjb25maWdQYWNrYWdlTWFuYWdlcjtcbiAgY29uc3QgcGFja2FnZVBhdGg6IHN0cmluZyA9IGAke2N3ZH0vcGFja2FnZS5qc29uYDtcbiAgY29uc3QgYXBwUGFja2FnZSA9IGdldFBhY2thZ2VKc29uKHBhY2thZ2VQYXRoKTtcbiAgY29uc3Qge2RlcGVuZGVuY2llcyA9IHt9LCBkZXZEZXBlbmRlbmNpZXMgPSB7fX0gPSBhcHBQYWNrYWdlO1xuXG4gIC8vIFJlbW92ZSBFU0J1aWxkLCBKZXN0IGFuZCBXZWJwYWNrIGZyb20gYXBwIHNpbmNlIGl0IHdpbGwgY29uZmxpY3RcbiAgYXBwUGFja2FnZS5kZXBlbmRlbmNpZXMgPSByZW1vdmVDb25mbGljdE1vZHVsZXMoZGVwZW5kZW5jaWVzKTtcbiAgYXBwUGFja2FnZS5kZXZEZXBlbmRlbmNpZXMgPSByZW1vdmVDb25mbGljdE1vZHVsZXMoZGV2RGVwZW5kZW5jaWVzKTtcblxuICAvLyBJbnN0YWxsIG5ldyBsaXN0IG9mIHBhY2thZ2VzXG4gIHRyeSB7XG4gICAgYXdhaXQgZXhlY2EocGFja2FnZU1hbmFnZXIsIFsnaW5zdGFsbCddLCB7XG4gICAgICBlbmNvZGluZzogJ3V0ZjgnLFxuICAgICAgc3RkaW86ICdpbmhlcml0J1xuICAgIH0pO1xuXG4gICAgLy8gU3RvcCBsb2FkZXJcbiAgICBzcGlubmVyLnN1Y2NlZWQoJ1N1Y2Nlc3NmdWxseSBtaWdyYXRlZCBhcHAhJyk7XG5cbiAgICAvLyBLaWxsIHByb2Nlc3NcbiAgICBjYWxsYmFjaygwKTtcbiAgICByZXR1cm4gMDtcbiAgfSBjYXRjaChlcnJvcikge1xuICAgIC8vIERpc3BsYXkgZXJyb3IgbWVzc2FnZVxuICAgIGxvZyhgXFxuJHtjbGlOYW1lfSBFcnJvcjogJHtlcnJvci5tZXNzYWdlfWAsICdlcnJvcicsIHF1aWV0KTtcblxuICAgIC8vIFN0b3Agc3Bpbm5lclxuICAgIHNwaW5uZXIuZmFpbCgnRmFpbGVkIHRvIHJlbW92ZSBtb2R1bGVzLicpO1xuXG4gICAgLy8gS2lsbCBwcm9jZXNzXG4gICAgY2FsbGJhY2soMSk7XG4gICAgcmV0dXJuIDE7XG4gIH1cbn07Il0sCiAgIm1hcHBpbmdzIjogIkFBSUEsU0FBUSxhQUFZO0FBRXBCLFNBQVEsaUJBQWdCO0FBQ3hCLFNBQVEsZUFBZSxnQkFBZ0IsdUJBQXVCLHFCQUFvQjtBQUNsRixTQUFRLFdBQVU7QUFVWCxNQUFNLFVBQVUsT0FBTyxLQUFxQixXQUE0QixRQUFRLFNBQTBCO0FBQy9HLFFBQU0sRUFBQyxVQUFVLE9BQU8sZ0JBQWdCLG1CQUFtQixNQUFLLElBQUk7QUFFcEUsUUFBTSxNQUFjLFFBQVEsSUFBSTtBQU1oQyxRQUFNLFVBQVUsY0FBYyxLQUFLO0FBQ25DLFVBQVEsTUFBTSwwQkFBMEI7QUFHeEMsUUFBTSxjQUFjO0FBRXBCLFFBQU0sRUFBQyxnQkFBZ0IscUJBQW9CLElBQUksVUFBVTtBQUN6RCxRQUFNLGlCQUF5QixxQkFBcUI7QUFDcEQsUUFBTSxjQUFzQixHQUFHLEdBQUc7QUFDbEMsUUFBTSxhQUFhLGVBQWUsV0FBVztBQUM3QyxRQUFNLEVBQUMsZUFBZSxDQUFDLEdBQUcsa0JBQWtCLENBQUMsRUFBQyxJQUFJO0FBR2xELGFBQVcsZUFBZSxzQkFBc0IsWUFBWTtBQUM1RCxhQUFXLGtCQUFrQixzQkFBc0IsZUFBZTtBQUdsRSxNQUFJO0FBQ0YsVUFBTSxNQUFNLGdCQUFnQixDQUFDLFNBQVMsR0FBRztBQUFBLE1BQ3ZDLFVBQVU7QUFBQSxNQUNWLE9BQU87QUFBQSxJQUNULENBQUM7QUFHRCxZQUFRLFFBQVEsNEJBQTRCO0FBRzVDLGFBQVMsQ0FBQztBQUNWLFdBQU87QUFBQSxFQUNULFNBQVEsT0FBTztBQUViLFFBQUk7QUFBQSxFQUFLLE9BQU8sV0FBVyxNQUFNLE9BQU8sSUFBSSxTQUFTLEtBQUs7QUFHMUQsWUFBUSxLQUFLLDJCQUEyQjtBQUd4QyxhQUFTLENBQUM7QUFDVixXQUFPO0FBQUEsRUFDVDtBQUNGOyIsCiAgIm5hbWVzIjogW10KfQo=
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import { execa } from "execa";
|
|
2
|
+
import semver from "semver";
|
|
3
|
+
import { LexConfig } from "../../LexConfig.js";
|
|
4
|
+
import { createSpinner, getPackageJson, setPackageJson } from "../../utils/app.js";
|
|
5
|
+
import { log } from "../../utils/log.js";
|
|
6
|
+
const publish = async (cmd, callback = process.exit) => {
|
|
7
|
+
const { bump, cliName = "Lex", newVersion, otp, packageManager: cmdPackageManager, private: accessPrivate, tag, quiet } = cmd;
|
|
8
|
+
log(`${cliName} publishing npm module...`, "info", quiet);
|
|
9
|
+
const spinner = createSpinner(quiet);
|
|
10
|
+
await LexConfig.parseConfig(cmd);
|
|
11
|
+
const { packageManager: configPackageManager } = LexConfig.config;
|
|
12
|
+
const packageManager = cmdPackageManager || configPackageManager;
|
|
13
|
+
const publishOptions = ["publish"];
|
|
14
|
+
if (accessPrivate) {
|
|
15
|
+
publishOptions.push("--access", "restricted");
|
|
16
|
+
}
|
|
17
|
+
if (otp) {
|
|
18
|
+
publishOptions.push("--otp", otp);
|
|
19
|
+
}
|
|
20
|
+
if (tag) {
|
|
21
|
+
publishOptions.push("--tag", tag);
|
|
22
|
+
}
|
|
23
|
+
let nextVersion;
|
|
24
|
+
const packagePath = `${process.cwd()}/package.json`;
|
|
25
|
+
let packageJson;
|
|
26
|
+
let packageName;
|
|
27
|
+
let prevVersion;
|
|
28
|
+
try {
|
|
29
|
+
packageJson = getPackageJson(packagePath);
|
|
30
|
+
packageName = packageJson.name;
|
|
31
|
+
prevVersion = packageJson.version;
|
|
32
|
+
} catch (error) {
|
|
33
|
+
log(`
|
|
34
|
+
${cliName} Error: The file, ${packagePath}, was not found or is malformed.
|
|
35
|
+
`, "error", quiet);
|
|
36
|
+
log(error.message, "error");
|
|
37
|
+
callback(1);
|
|
38
|
+
return 1;
|
|
39
|
+
}
|
|
40
|
+
if (newVersion) {
|
|
41
|
+
nextVersion = newVersion;
|
|
42
|
+
} else if (bump) {
|
|
43
|
+
const formatBump = bump.toString().trim().toLowerCase();
|
|
44
|
+
if (formatBump) {
|
|
45
|
+
const validReleases = ["major", "minor", "patch"];
|
|
46
|
+
const validPreReleases = ["alpha", "beta", "rc"];
|
|
47
|
+
const packageVersion = semver.coerce(prevVersion);
|
|
48
|
+
if (!semver.valid(packageVersion)) {
|
|
49
|
+
log(`
|
|
50
|
+
${cliName} Error: Version is invalid in package.json`, "error", quiet);
|
|
51
|
+
callback(1);
|
|
52
|
+
return 1;
|
|
53
|
+
}
|
|
54
|
+
if (validReleases.includes(formatBump)) {
|
|
55
|
+
nextVersion = semver.inc(packageVersion, formatBump);
|
|
56
|
+
} else if (validPreReleases.includes(formatBump)) {
|
|
57
|
+
nextVersion = semver.inc(packageVersion, "prerelease", formatBump);
|
|
58
|
+
} else {
|
|
59
|
+
log(`
|
|
60
|
+
${cliName} Error: Bump type is invalid. please make sure it is one of the following: ${validReleases.join(", ")}, ${validPreReleases.join(", ")}`, "error", quiet);
|
|
61
|
+
callback(1);
|
|
62
|
+
return 1;
|
|
63
|
+
}
|
|
64
|
+
} else {
|
|
65
|
+
log(`
|
|
66
|
+
${cliName} Error: Bump type is missing.`, "error", quiet);
|
|
67
|
+
callback(1);
|
|
68
|
+
return 1;
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
if (nextVersion && packageManager === "yarn") {
|
|
72
|
+
publishOptions.push("--new-version", nextVersion);
|
|
73
|
+
} else if (nextVersion && packageJson) {
|
|
74
|
+
try {
|
|
75
|
+
setPackageJson({ ...packageJson, version: nextVersion }, packagePath);
|
|
76
|
+
} catch (error) {
|
|
77
|
+
log(`
|
|
78
|
+
${cliName} Error: The file, ${packagePath}, was not found or is malformed. ${error.message}`, "error", quiet);
|
|
79
|
+
callback(1);
|
|
80
|
+
return 1;
|
|
81
|
+
}
|
|
82
|
+
} else {
|
|
83
|
+
nextVersion = prevVersion;
|
|
84
|
+
}
|
|
85
|
+
try {
|
|
86
|
+
await execa(packageManager, publishOptions, {
|
|
87
|
+
encoding: "utf8",
|
|
88
|
+
stdio: "inherit"
|
|
89
|
+
});
|
|
90
|
+
spinner.succeed(`Successfully published npm package: ${packageName}!`);
|
|
91
|
+
callback(0);
|
|
92
|
+
return 0;
|
|
93
|
+
} catch (error) {
|
|
94
|
+
log(`
|
|
95
|
+
${cliName} Error: ${error.message}`, "error", quiet);
|
|
96
|
+
spinner.fail("Publishing to npm has failed.");
|
|
97
|
+
callback(1);
|
|
98
|
+
return 1;
|
|
99
|
+
}
|
|
100
|
+
};
|
|
101
|
+
export {
|
|
102
|
+
publish
|
|
103
|
+
};
|
|
104
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/commands/publish/publish.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2018-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {execa} from 'execa';\nimport semver, {ReleaseType} from 'semver';\n\nimport {LexConfig} from '../../LexConfig.js';\nimport {createSpinner, getPackageJson, setPackageJson} from '../../utils/app.js';\nimport {log} from '../../utils/log.js';\n\nexport interface PublishOptions {\n  readonly bump?: string;\n  readonly cliName?: string;\n  readonly newVersion?: string;\n  readonly otp?: string;\n  readonly packageManager?: string;\n  readonly private?: boolean;\n  readonly quiet?: boolean;\n  readonly tag?: string;\n}\n\nexport type PublishCallback = typeof process.exit;\n\nexport const publish = async (cmd: PublishOptions, callback: PublishCallback = process.exit): Promise<number> => {\n  const {bump, cliName = 'Lex', newVersion, otp, packageManager: cmdPackageManager, private: accessPrivate, tag, quiet} = cmd;\n  log(`${cliName} publishing npm module...`, 'info', quiet);\n\n  // Spinner\n  const spinner = createSpinner(quiet);\n\n  // Get custom configuration\n  await LexConfig.parseConfig(cmd);\n\n  const {packageManager: configPackageManager} = LexConfig.config;\n  const packageManager: string = cmdPackageManager || configPackageManager;\n  const publishOptions: string[] = ['publish'];\n\n  if(accessPrivate) {\n    publishOptions.push('--access', 'restricted');\n  }\n\n  if(otp) {\n    publishOptions.push('--otp', otp);\n  }\n\n  if(tag) {\n    publishOptions.push('--tag', tag);\n  }\n\n  // Get next version number\n  let nextVersion: string;\n  const packagePath: string = `${process.cwd()}/package.json`;\n  let packageJson;\n  let packageName: string;\n  let prevVersion: string;\n\n  // If not using yarn, we'll use npm and manually update the version number\n  try {\n    packageJson = getPackageJson(packagePath);\n    packageName = packageJson.name;\n    prevVersion = packageJson.version;\n  } catch(error) {\n    log(`\\n${cliName} Error: The file, ${packagePath}, was not found or is malformed.\\n`, 'error', quiet);\n    log(error.message, 'error');\n    callback(1);\n    return 1;\n  }\n\n  // Update package.json with the latest version\n  if(newVersion) {\n    // If using a specific version, we don't need to determine the next bump\n    nextVersion = newVersion;\n  } else if(bump) {\n    // Determine next version\n    const formatBump = bump.toString()\n      .trim()\n      .toLowerCase();\n\n    if(formatBump) {\n      const validReleases: string[] = ['major', 'minor', 'patch'];\n      const validPreReleases: string[] = ['alpha', 'beta', 'rc'];\n\n      // Make sure the version in package.json is valid\n      const packageVersion = semver.coerce(prevVersion);\n\n      if(!semver.valid(packageVersion)) {\n        log(`\\n${cliName} Error: Version is invalid in package.json`, 'error', quiet);\n        callback(1);\n        return 1;\n      }\n\n      if(validReleases.includes(formatBump)) {\n        nextVersion = semver.inc(packageVersion, formatBump as ReleaseType);\n      } else if(validPreReleases.includes(formatBump)) {\n        nextVersion = semver.inc(packageVersion, 'prerelease', formatBump);\n      } else {\n        log(`\\n${cliName} Error: Bump type is invalid. please make sure it is one of the following: ${validReleases.join(', ')}, ${validPreReleases.join(', ')}`, 'error', quiet);\n        callback(1);\n        return 1;\n      }\n    } else {\n      log(`\\n${cliName} Error: Bump type is missing.`, 'error', quiet);\n      callback(1);\n      return 1;\n    }\n  }\n\n  if(nextVersion && packageManager === 'yarn') {\n    publishOptions.push('--new-version', nextVersion);\n  } else if(nextVersion && packageJson) {\n    try {\n      // Save updated version\n      setPackageJson({...packageJson, version: nextVersion}, packagePath);\n    } catch(error) {\n      log(`\\n${cliName} Error: The file, ${packagePath}, was not found or is malformed. ${error.message}`, 'error', quiet);\n      callback(1);\n      return 1;\n    }\n  } else {\n    nextVersion = prevVersion;\n  }\n\n  try {\n    await execa(packageManager, publishOptions, {\n      encoding: 'utf8',\n      stdio: 'inherit'\n    });\n\n    spinner.succeed(`Successfully published npm package: ${packageName}!`);\n\n    // Kill process\n    callback(0);\n    return 0;\n  } catch(error) {\n    // Display error message\n    log(`\\n${cliName} Error: ${error.message}`, 'error', quiet);\n\n    // Stop spinner\n    spinner.fail('Publishing to npm has failed.');\n\n    // Kill process\n    callback(1);\n    return 1;\n  }\n};"],
  "mappings": "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",
  "names": []
}

|
|
@@ -0,0 +1,327 @@
|
|
|
1
|
+
import { execa } from "execa";
|
|
2
|
+
import { readFileSync } from "fs";
|
|
3
|
+
import { sync as globSync } from "glob";
|
|
4
|
+
import { resolve as pathResolve } from "path";
|
|
5
|
+
import { URL } from "url";
|
|
6
|
+
import { LexConfig } from "../../LexConfig.js";
|
|
7
|
+
import { createSpinner } from "../../utils/app.js";
|
|
8
|
+
import { relativeNodePath } from "../../utils/file.js";
|
|
9
|
+
import { log } from "../../utils/log.js";
|
|
10
|
+
import { aiFunction } from "../ai/ai.js";
|
|
11
|
+
const getTestFilePatterns = (testPathPattern) => {
|
|
12
|
+
const defaultPatterns = ["**/*.test.*", "**/*.spec.*"];
|
|
13
|
+
if (!testPathPattern) {
|
|
14
|
+
return defaultPatterns;
|
|
15
|
+
}
|
|
16
|
+
return [testPathPattern];
|
|
17
|
+
};
|
|
18
|
+
const findUncoveredSourceFiles = () => {
|
|
19
|
+
const sourceFiles = globSync("src/**/*.{ts,tsx,js,jsx}", {
|
|
20
|
+
cwd: process.cwd(),
|
|
21
|
+
ignore: ["**/node_modules/**", "**/dist/**", "**/*.test.*", "**/*.spec.*"]
|
|
22
|
+
});
|
|
23
|
+
const testFiles = globSync("**/*.{test,spec}.{ts,tsx,js,jsx}", {
|
|
24
|
+
cwd: process.cwd(),
|
|
25
|
+
ignore: ["**/node_modules/**", "**/dist/**"]
|
|
26
|
+
});
|
|
27
|
+
return sourceFiles.filter((sourceFile) => {
|
|
28
|
+
const baseName = sourceFile.replace(/\.[^/.]+$/, "");
|
|
29
|
+
return !testFiles.some((testFile) => testFile.includes(baseName));
|
|
30
|
+
});
|
|
31
|
+
};
|
|
32
|
+
const processTestResults = (outputFile) => {
|
|
33
|
+
if (!outputFile) {
|
|
34
|
+
return null;
|
|
35
|
+
}
|
|
36
|
+
try {
|
|
37
|
+
const content = readFileSync(outputFile, "utf-8");
|
|
38
|
+
return JSON.parse(content);
|
|
39
|
+
} catch (_error) {
|
|
40
|
+
return null;
|
|
41
|
+
}
|
|
42
|
+
};
|
|
43
|
+
const test = async (options, args, callback = process.exit) => {
|
|
44
|
+
const {
|
|
45
|
+
analyze = false,
|
|
46
|
+
aiAnalyze = false,
|
|
47
|
+
aiDebug = false,
|
|
48
|
+
aiGenerate = false,
|
|
49
|
+
bail,
|
|
50
|
+
changedFilesWithAncestor,
|
|
51
|
+
changedSince,
|
|
52
|
+
ci,
|
|
53
|
+
cliName = "Lex",
|
|
54
|
+
collectCoverageFrom,
|
|
55
|
+
colors,
|
|
56
|
+
config,
|
|
57
|
+
debug = false,
|
|
58
|
+
debugTests = false,
|
|
59
|
+
detectOpenHandles,
|
|
60
|
+
env,
|
|
61
|
+
errorOnDeprecated,
|
|
62
|
+
expand,
|
|
63
|
+
forceExit,
|
|
64
|
+
generate = false,
|
|
65
|
+
json,
|
|
66
|
+
lastCommit,
|
|
67
|
+
listTests,
|
|
68
|
+
logHeapUsage,
|
|
69
|
+
maxWorkers,
|
|
70
|
+
noStackTrace,
|
|
71
|
+
notify,
|
|
72
|
+
onlyChanged,
|
|
73
|
+
outputFile,
|
|
74
|
+
passWithNoTests,
|
|
75
|
+
quiet,
|
|
76
|
+
removeCache,
|
|
77
|
+
runInBand,
|
|
78
|
+
setup,
|
|
79
|
+
showConfig,
|
|
80
|
+
silent,
|
|
81
|
+
testLocationInResults,
|
|
82
|
+
testNamePattern,
|
|
83
|
+
testPathPattern,
|
|
84
|
+
update,
|
|
85
|
+
useStderr,
|
|
86
|
+
verbose,
|
|
87
|
+
watch,
|
|
88
|
+
watchAll
|
|
89
|
+
} = options;
|
|
90
|
+
const useGenerate = generate || aiGenerate;
|
|
91
|
+
const useAnalyze = analyze || aiAnalyze;
|
|
92
|
+
const useDebug = debugTests || aiDebug;
|
|
93
|
+
log(`${cliName} testing...`, "info", quiet);
|
|
94
|
+
const spinner = createSpinner(quiet);
|
|
95
|
+
await LexConfig.parseConfig(options);
|
|
96
|
+
const { useTypescript } = LexConfig.config;
|
|
97
|
+
if (useTypescript) {
|
|
98
|
+
LexConfig.checkTypescriptConfig();
|
|
99
|
+
}
|
|
100
|
+
if (useGenerate) {
|
|
101
|
+
spinner.start("AI is analyzing code to generate test cases...");
|
|
102
|
+
try {
|
|
103
|
+
const uncoveredFiles = findUncoveredSourceFiles();
|
|
104
|
+
if (uncoveredFiles.length > 0) {
|
|
105
|
+
const targetFile = uncoveredFiles[0];
|
|
106
|
+
await aiFunction({
|
|
107
|
+
prompt: `Generate Jest unit tests for this file: ${targetFile}
|
|
108
|
+
|
|
109
|
+
${readFileSync(targetFile, "utf-8")}
|
|
110
|
+
|
|
111
|
+
Please create comprehensive tests that cover the main functionality. Include test fixtures and mocks where necessary.`,
|
|
112
|
+
task: "test",
|
|
113
|
+
file: targetFile,
|
|
114
|
+
context: true,
|
|
115
|
+
quiet
|
|
116
|
+
});
|
|
117
|
+
spinner.succeed(`AI test generation suggestions provided for ${targetFile}`);
|
|
118
|
+
} else {
|
|
119
|
+
spinner.succeed("All source files appear to have corresponding test files");
|
|
120
|
+
}
|
|
121
|
+
} catch (aiError) {
|
|
122
|
+
spinner.fail("Could not generate AI test suggestions");
|
|
123
|
+
if (!quiet) {
|
|
124
|
+
console.error("AI test generation error:", aiError);
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
const dirName = new URL(".", import.meta.url).pathname;
|
|
129
|
+
const dirPath = pathResolve(dirName, "../../..");
|
|
130
|
+
const jestPath = relativeNodePath("jest-cli/bin/jest.js", dirPath);
|
|
131
|
+
const jestConfigFile = config || pathResolve(dirName, "../../../jest.config.lex.js");
|
|
132
|
+
const jestSetupFile = setup || "";
|
|
133
|
+
const jestOptions = ["--no-cache"];
|
|
134
|
+
jestOptions.push("--config", jestConfigFile);
|
|
135
|
+
if (bail) {
|
|
136
|
+
jestOptions.push("--bail");
|
|
137
|
+
}
|
|
138
|
+
if (changedFilesWithAncestor) {
|
|
139
|
+
jestOptions.push("--changedFilesWithAncestor");
|
|
140
|
+
}
|
|
141
|
+
if (changedSince) {
|
|
142
|
+
jestOptions.push("--changedSince");
|
|
143
|
+
}
|
|
144
|
+
if (ci) {
|
|
145
|
+
jestOptions.push("--ci");
|
|
146
|
+
}
|
|
147
|
+
if (collectCoverageFrom) {
|
|
148
|
+
jestOptions.push("--collectCoverageFrom", collectCoverageFrom);
|
|
149
|
+
}
|
|
150
|
+
if (colors) {
|
|
151
|
+
jestOptions.push("--colors");
|
|
152
|
+
}
|
|
153
|
+
if (debug) {
|
|
154
|
+
jestOptions.push("--debug");
|
|
155
|
+
}
|
|
156
|
+
if (detectOpenHandles) {
|
|
157
|
+
jestOptions.push("--detectOpenHandles");
|
|
158
|
+
}
|
|
159
|
+
if (env) {
|
|
160
|
+
jestOptions.push("--env");
|
|
161
|
+
}
|
|
162
|
+
if (errorOnDeprecated) {
|
|
163
|
+
jestOptions.push("--errorOnDeprecated");
|
|
164
|
+
}
|
|
165
|
+
if (expand) {
|
|
166
|
+
jestOptions.push("--expand");
|
|
167
|
+
}
|
|
168
|
+
if (forceExit) {
|
|
169
|
+
jestOptions.push("--forceExit");
|
|
170
|
+
}
|
|
171
|
+
if (json) {
|
|
172
|
+
jestOptions.push("--json");
|
|
173
|
+
}
|
|
174
|
+
if (lastCommit) {
|
|
175
|
+
jestOptions.push("--lastCommit");
|
|
176
|
+
}
|
|
177
|
+
if (listTests) {
|
|
178
|
+
jestOptions.push("--listTests");
|
|
179
|
+
}
|
|
180
|
+
if (logHeapUsage) {
|
|
181
|
+
jestOptions.push("--logHeapUsage");
|
|
182
|
+
}
|
|
183
|
+
if (maxWorkers) {
|
|
184
|
+
jestOptions.push("--maxWorkers", maxWorkers);
|
|
185
|
+
}
|
|
186
|
+
if (noStackTrace) {
|
|
187
|
+
jestOptions.push("--noStackTrace");
|
|
188
|
+
}
|
|
189
|
+
if (notify) {
|
|
190
|
+
jestOptions.push("--notify");
|
|
191
|
+
}
|
|
192
|
+
if (onlyChanged) {
|
|
193
|
+
jestOptions.push("--onlyChanged");
|
|
194
|
+
}
|
|
195
|
+
let tempOutputFile = outputFile;
|
|
196
|
+
if ((useAnalyze || useDebug) && !outputFile) {
|
|
197
|
+
tempOutputFile = ".lex-test-results.json";
|
|
198
|
+
jestOptions.push("--json", "--outputFile", tempOutputFile);
|
|
199
|
+
} else if (outputFile) {
|
|
200
|
+
jestOptions.push("--outputFile", outputFile);
|
|
201
|
+
}
|
|
202
|
+
if (passWithNoTests) {
|
|
203
|
+
jestOptions.push("--passWithNoTests");
|
|
204
|
+
}
|
|
205
|
+
if (runInBand) {
|
|
206
|
+
jestOptions.push("--runInBand");
|
|
207
|
+
}
|
|
208
|
+
if (showConfig) {
|
|
209
|
+
jestOptions.push("--showConfig");
|
|
210
|
+
}
|
|
211
|
+
if (silent) {
|
|
212
|
+
jestOptions.push("--silent");
|
|
213
|
+
}
|
|
214
|
+
if (testLocationInResults) {
|
|
215
|
+
jestOptions.push("--testLocationInResults");
|
|
216
|
+
}
|
|
217
|
+
if (testNamePattern) {
|
|
218
|
+
jestOptions.push("--testNamePattern", testNamePattern);
|
|
219
|
+
}
|
|
220
|
+
if (testPathPattern) {
|
|
221
|
+
jestOptions.push("--testPathPattern", testPathPattern);
|
|
222
|
+
}
|
|
223
|
+
if (useStderr) {
|
|
224
|
+
jestOptions.push("--useStderr");
|
|
225
|
+
}
|
|
226
|
+
if (verbose) {
|
|
227
|
+
jestOptions.push("--verbose");
|
|
228
|
+
}
|
|
229
|
+
if (watchAll) {
|
|
230
|
+
jestOptions.push("--watchAll");
|
|
231
|
+
}
|
|
232
|
+
if (removeCache) {
|
|
233
|
+
jestOptions.push("--no-cache");
|
|
234
|
+
}
|
|
235
|
+
if (jestSetupFile !== "") {
|
|
236
|
+
const cwd = process.cwd();
|
|
237
|
+
jestOptions.push(`--setupFilesAfterEnv=${pathResolve(cwd, jestSetupFile)}`);
|
|
238
|
+
}
|
|
239
|
+
if (update) {
|
|
240
|
+
jestOptions.push("--updateSnapshot");
|
|
241
|
+
}
|
|
242
|
+
if (watch) {
|
|
243
|
+
jestOptions.push("--watch", watch);
|
|
244
|
+
}
|
|
245
|
+
if (args) {
|
|
246
|
+
jestOptions.push(...args);
|
|
247
|
+
}
|
|
248
|
+
try {
|
|
249
|
+
await execa(jestPath, jestOptions, {
|
|
250
|
+
encoding: "utf8",
|
|
251
|
+
stdio: "inherit"
|
|
252
|
+
});
|
|
253
|
+
spinner.succeed("Testing completed!");
|
|
254
|
+
if (useAnalyze) {
|
|
255
|
+
spinner.start("AI is analyzing test coverage and suggesting improvements...");
|
|
256
|
+
try {
|
|
257
|
+
const testResults = processTestResults(tempOutputFile);
|
|
258
|
+
const filePatterns = getTestFilePatterns(testPathPattern);
|
|
259
|
+
await aiFunction({
|
|
260
|
+
prompt: `Analyze these Jest test results and suggest test coverage improvements:
|
|
261
|
+
|
|
262
|
+
${JSON.stringify(testResults, null, 2)}
|
|
263
|
+
|
|
264
|
+
Test patterns: ${filePatterns.join(", ")}
|
|
265
|
+
|
|
266
|
+
Please provide:
|
|
267
|
+
1. Analysis of current coverage gaps
|
|
268
|
+
2. Suggestions for improving test cases
|
|
269
|
+
3. Recommendations for additional integration test scenarios
|
|
270
|
+
4. Best practices for increasing test effectiveness`,
|
|
271
|
+
task: "optimize",
|
|
272
|
+
context: true,
|
|
273
|
+
quiet
|
|
274
|
+
});
|
|
275
|
+
spinner.succeed("AI test analysis complete");
|
|
276
|
+
} catch (aiError) {
|
|
277
|
+
spinner.fail("Could not generate AI test analysis");
|
|
278
|
+
if (!quiet) {
|
|
279
|
+
console.error("AI analysis error:", aiError);
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
callback(0);
|
|
284
|
+
return 0;
|
|
285
|
+
} catch (error) {
|
|
286
|
+
log(`
|
|
287
|
+
${cliName} Error: Check for unit test errors and/or coverage.`, "error", quiet);
|
|
288
|
+
spinner.fail("Testing failed!");
|
|
289
|
+
if (useDebug) {
|
|
290
|
+
spinner.start("AI is analyzing test failures...");
|
|
291
|
+
try {
|
|
292
|
+
const testResults = processTestResults(tempOutputFile);
|
|
293
|
+
await aiFunction({
|
|
294
|
+
prompt: `Debug these failed Jest tests and suggest fixes:
|
|
295
|
+
|
|
296
|
+
${JSON.stringify(error.message, null, 2)}
|
|
297
|
+
|
|
298
|
+
Test results: ${JSON.stringify(testResults, null, 2)}
|
|
299
|
+
|
|
300
|
+
Please provide:
|
|
301
|
+
1. Analysis of why the tests are failing
|
|
302
|
+
2. Specific suggestions to fix each failing test
|
|
303
|
+
3. Any potential issues with test fixtures or mocks
|
|
304
|
+
4. Code examples for solutions`,
|
|
305
|
+
task: "help",
|
|
306
|
+
context: true,
|
|
307
|
+
quiet
|
|
308
|
+
});
|
|
309
|
+
spinner.succeed("AI debugging assistance complete");
|
|
310
|
+
} catch (aiError) {
|
|
311
|
+
spinner.fail("Could not generate AI debugging assistance");
|
|
312
|
+
if (!quiet) {
|
|
313
|
+
console.error("AI debugging error:", aiError);
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
}
|
|
317
|
+
callback(1);
|
|
318
|
+
return 1;
|
|
319
|
+
}
|
|
320
|
+
};
|
|
321
|
+
var test_default = test;
|
|
322
|
+
export {
|
|
323
|
+
test_default as default,
|
|
324
|
+
getTestFilePatterns,
|
|
325
|
+
test
|
|
326
|
+
};
|
|
327
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/commands/test/test.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2018-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {execa} from 'execa';\nimport {readFileSync} from 'fs';\nimport {sync as globSync} from 'glob';\nimport {resolve as pathResolve} from 'path';\nimport {URL} from 'url';\n\nimport {LexConfig} from '../../LexConfig.js';\nimport {createSpinner} from '../../utils/app.js';\nimport {relativeNodePath} from '../../utils/file.js';\nimport {log} from '../../utils/log.js';\nimport {aiFunction} from '../ai/ai.js';\n\nexport interface TestOptions {\n  readonly analyze?: boolean;\n  readonly aiDebug?: boolean;\n  readonly aiGenerate?: boolean;\n  readonly aiAnalyze?: boolean;\n  readonly bail?: boolean;\n  readonly changedFilesWithAncestor?: boolean;\n  readonly changedSince?: string;\n  readonly ci?: boolean;\n  readonly cliName?: string;\n  readonly collectCoverageFrom?: string;\n  readonly colors?: boolean;\n  readonly config?: string;\n  readonly debug?: boolean;\n  readonly debugTests?: boolean;\n  readonly detectOpenHandles?: boolean;\n  readonly env?: string;\n  readonly errorOnDeprecated?: boolean;\n  readonly expand?: boolean;\n  readonly forceExit?: boolean;\n  readonly generate?: boolean;\n  readonly json?: boolean;\n  readonly lastCommit?: boolean;\n  readonly listTests?: boolean;\n  readonly logHeapUsage?: boolean;\n  readonly maxWorkers?: string;\n  readonly noStackTrace?: boolean;\n  readonly notify?: boolean;\n  readonly onlyChanged?: boolean;\n  readonly outputFile?: string;\n  readonly passWithNoTests?: boolean;\n  readonly quiet?: boolean;\n  readonly removeCache?: boolean;\n  readonly runInBand?: boolean;\n  readonly setup?: string;\n  readonly showConfig?: boolean;\n  readonly silent?: boolean;\n  readonly testLocationInResults?: boolean;\n  readonly testNamePattern?: string;\n  readonly testPathPattern?: string;\n  readonly update?: boolean;\n  readonly useStderr?: boolean;\n  readonly verbose?: boolean;\n  readonly watch?: string;\n  readonly watchAll?: boolean;\n}\n\nexport type TestCallback = typeof process.exit;\n\nexport const getTestFilePatterns = (testPathPattern?: string): string[] => {\n  const defaultPatterns = ['**/*.test.*', '**/*.spec.*'];\n\n  if(!testPathPattern) {\n    return defaultPatterns;\n  }\n\n  return [testPathPattern];\n};\n\nconst findUncoveredSourceFiles = (): string[] => {\n  const sourceFiles = globSync('src/**/*.{ts,tsx,js,jsx}', {\n    cwd: process.cwd(),\n    ignore: ['**/node_modules/**', '**/dist/**', '**/*.test.*', '**/*.spec.*']\n  });\n\n  const testFiles = globSync('**/*.{test,spec}.{ts,tsx,js,jsx}', {\n    cwd: process.cwd(),\n    ignore: ['**/node_modules/**', '**/dist/**']\n  });\n\n  // Simple heuristic to find source files without corresponding test files\n  return sourceFiles.filter((sourceFile) => {\n    const baseName = sourceFile.replace(/\\.[^/.]+$/, '');\n    return !testFiles.some((testFile) => testFile.includes(baseName));\n  });\n};\n\nconst processTestResults = (outputFile?: string): any => {\n  if(!outputFile) {\n    return null;\n  }\n\n  try {\n    const content = readFileSync(outputFile, 'utf-8');\n    return JSON.parse(content);\n  } catch(_error) {\n    return null;\n  }\n};\n\nexport const test = async (options: TestOptions, args: string[], callback: TestCallback = process.exit): Promise<number> => {\n  const {\n    analyze = false,\n    aiAnalyze = false,\n    aiDebug = false,\n    aiGenerate = false,\n    bail,\n    changedFilesWithAncestor,\n    changedSince,\n    ci,\n    cliName = 'Lex',\n    collectCoverageFrom,\n    colors,\n    config,\n    debug = false,\n    debugTests = false,\n    detectOpenHandles,\n    env,\n    errorOnDeprecated,\n    expand,\n    forceExit,\n    generate = false,\n    json,\n    lastCommit,\n    listTests,\n    logHeapUsage,\n    maxWorkers,\n    noStackTrace,\n    notify,\n    onlyChanged,\n    outputFile,\n    passWithNoTests,\n    quiet,\n    removeCache,\n    runInBand,\n    setup,\n    showConfig,\n    silent,\n    testLocationInResults,\n    testNamePattern,\n    testPathPattern,\n    update,\n    useStderr,\n    verbose,\n    watch,\n    watchAll\n  } = options;\n\n  const useGenerate = generate || aiGenerate;\n  const useAnalyze = analyze || aiAnalyze;\n  const useDebug = debugTests || aiDebug;\n\n  log(`${cliName} testing...`, 'info', quiet);\n\n  const spinner = createSpinner(quiet);\n\n  await LexConfig.parseConfig(options);\n\n  const {useTypescript} = LexConfig.config;\n\n  if(useTypescript) {\n    LexConfig.checkTypescriptConfig();\n  }\n\n  if(useGenerate) {\n    spinner.start('AI is analyzing code to generate test cases...');\n\n    try {\n      const uncoveredFiles = findUncoveredSourceFiles();\n\n      if(uncoveredFiles.length > 0) {\n        const targetFile = uncoveredFiles[0];\n\n        await aiFunction({\n          prompt: `Generate Jest unit tests for this file: ${targetFile}\\n\\n${readFileSync(targetFile, 'utf-8')}\\n\\nPlease create comprehensive tests that cover the main functionality. Include test fixtures and mocks where necessary.`,\n          task: 'test',\n          file: targetFile,\n          context: true,\n          quiet\n        });\n\n        spinner.succeed(`AI test generation suggestions provided for ${targetFile}`);\n      } else {\n        spinner.succeed('All source files appear to have corresponding test files');\n      }\n    } catch(aiError) {\n      spinner.fail('Could not generate AI test suggestions');\n      if(!quiet) {\n        console.error('AI test generation error:', aiError);\n      }\n    }\n  }\n\n  const dirName = new URL('.', import.meta.url).pathname;\n  const dirPath: string = pathResolve(dirName, '../../..');\n  const jestPath: string = relativeNodePath('jest-cli/bin/jest.js', dirPath);\n  const jestConfigFile: string = config || pathResolve(dirName, '../../../jest.config.lex.js');\n  const jestSetupFile: string = setup || '';\n  const jestOptions: string[] = ['--no-cache'];\n\n  jestOptions.push('--config', jestConfigFile);\n\n  if(bail) {\n    jestOptions.push('--bail');\n  }\n\n  if(changedFilesWithAncestor) {\n    jestOptions.push('--changedFilesWithAncestor');\n  }\n\n  if(changedSince) {\n    jestOptions.push('--changedSince');\n  }\n\n  if(ci) {\n    jestOptions.push('--ci');\n  }\n\n  if(collectCoverageFrom) {\n    jestOptions.push('--collectCoverageFrom', collectCoverageFrom);\n  }\n\n  if(colors) {\n    jestOptions.push('--colors');\n  }\n\n  if(debug) {\n    jestOptions.push('--debug');\n  }\n\n  if(detectOpenHandles) {\n    jestOptions.push('--detectOpenHandles');\n  }\n\n  if(env) {\n    jestOptions.push('--env');\n  }\n\n  if(errorOnDeprecated) {\n    jestOptions.push('--errorOnDeprecated');\n  }\n\n  if(expand) {\n    jestOptions.push('--expand');\n  }\n\n  if(forceExit) {\n    jestOptions.push('--forceExit');\n  }\n\n  if(json) {\n    jestOptions.push('--json');\n  }\n\n  if(lastCommit) {\n    jestOptions.push('--lastCommit');\n  }\n\n  if(listTests) {\n    jestOptions.push('--listTests');\n  }\n\n  if(logHeapUsage) {\n    jestOptions.push('--logHeapUsage');\n  }\n\n  if(maxWorkers) {\n    jestOptions.push('--maxWorkers', maxWorkers);\n  }\n\n  if(noStackTrace) {\n    jestOptions.push('--noStackTrace');\n  }\n\n  if(notify) {\n    jestOptions.push('--notify');\n  }\n\n  if(onlyChanged) {\n    jestOptions.push('--onlyChanged');\n  }\n\n  let tempOutputFile = outputFile;\n\n  if((useAnalyze || useDebug) && !outputFile) {\n    tempOutputFile = '.lex-test-results.json';\n    jestOptions.push('--json', '--outputFile', tempOutputFile);\n  } else if(outputFile) {\n    jestOptions.push('--outputFile', outputFile);\n  }\n\n  if(passWithNoTests) {\n    jestOptions.push('--passWithNoTests');\n  }\n\n  if(runInBand) {\n    jestOptions.push('--runInBand');\n  }\n\n  if(showConfig) {\n    jestOptions.push('--showConfig');\n  }\n\n  if(silent) {\n    jestOptions.push('--silent');\n  }\n\n  if(testLocationInResults) {\n    jestOptions.push('--testLocationInResults');\n  }\n\n  if(testNamePattern) {\n    jestOptions.push('--testNamePattern', testNamePattern);\n  }\n\n  if(testPathPattern) {\n    jestOptions.push('--testPathPattern', testPathPattern);\n  }\n\n  if(useStderr) {\n    jestOptions.push('--useStderr');\n  }\n\n  if(verbose) {\n    jestOptions.push('--verbose');\n  }\n\n  if(watchAll) {\n    jestOptions.push('--watchAll');\n  }\n\n  if(removeCache) {\n    jestOptions.push('--no-cache');\n  }\n\n  if(jestSetupFile !== '') {\n    const cwd: string = process.cwd();\n    jestOptions.push(`--setupFilesAfterEnv=${pathResolve(cwd, jestSetupFile)}`);\n  }\n\n  if(update) {\n    jestOptions.push('--updateSnapshot');\n  }\n\n  if(watch) {\n    jestOptions.push('--watch', watch);\n  }\n\n  if(args) {\n    jestOptions.push(...args);\n  }\n\n  try {\n    await execa(jestPath, jestOptions, {\n      encoding: 'utf8',\n      stdio: 'inherit'\n    });\n\n    spinner.succeed('Testing completed!');\n\n    if(useAnalyze) {\n      spinner.start('AI is analyzing test coverage and suggesting improvements...');\n\n      try {\n        const testResults = processTestResults(tempOutputFile);\n        const filePatterns = getTestFilePatterns(testPathPattern);\n\n        await aiFunction({\n          prompt: `Analyze these Jest test results and suggest test coverage improvements:\n          \n${JSON.stringify(testResults, null, 2)}\n\nTest patterns: ${filePatterns.join(', ')}\n\nPlease provide:\n1. Analysis of current coverage gaps\n2. Suggestions for improving test cases\n3. Recommendations for additional integration test scenarios\n4. Best practices for increasing test effectiveness`,\n          task: 'optimize',\n          context: true,\n          quiet\n        });\n\n        spinner.succeed('AI test analysis complete');\n      } catch(aiError) {\n        spinner.fail('Could not generate AI test analysis');\n        if(!quiet) {\n          console.error('AI analysis error:', aiError);\n        }\n      }\n    }\n\n    callback(0);\n    return 0;\n  } catch(error) {\n    log(`\\n${cliName} Error: Check for unit test errors and/or coverage.`, 'error', quiet);\n\n    spinner.fail('Testing failed!');\n\n    if(useDebug) {\n      spinner.start('AI is analyzing test failures...');\n\n      try {\n        const testResults = processTestResults(tempOutputFile);\n\n        await aiFunction({\n          prompt: `Debug these failed Jest tests and suggest fixes:\n          \n${JSON.stringify(error.message, null, 2)}\n\nTest results: ${JSON.stringify(testResults, null, 2)}\n\nPlease provide:\n1. Analysis of why the tests are failing\n2. Specific suggestions to fix each failing test\n3. Any potential issues with test fixtures or mocks\n4. Code examples for solutions`,\n          task: 'help',\n          context: true,\n          quiet\n        });\n\n        spinner.succeed('AI debugging assistance complete');\n      } catch(aiError) {\n        spinner.fail('Could not generate AI debugging assistance');\n        if(!quiet) {\n          console.error('AI debugging error:', aiError);\n        }\n      }\n    }\n\n    callback(1);\n    return 1;\n  }\n};\n\nexport default test;"],
  "mappings": "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",
  "names": []
}

|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import { execa } from "execa";
|
|
2
|
+
import { resolve as pathResolve } from "path";
|
|
3
|
+
import { LexConfig } from "../../LexConfig.js";
|
|
4
|
+
import { createSpinner } from "../../utils/app.js";
|
|
5
|
+
import { log } from "../../utils/log.js";
|
|
6
|
+
const update = async (cmd, callback = process.exit) => {
|
|
7
|
+
const { cliName = "Lex", packageManager: cmdPackageManager, quiet, registry } = cmd;
|
|
8
|
+
log(`${cliName} updating packages...`, "info", quiet);
|
|
9
|
+
const spinner = createSpinner(quiet);
|
|
10
|
+
await LexConfig.parseConfig(cmd);
|
|
11
|
+
const { packageManager: configPackageManager } = LexConfig.config;
|
|
12
|
+
const packageManager = cmdPackageManager || configPackageManager || "npm";
|
|
13
|
+
const isNpm = packageManager === "npm";
|
|
14
|
+
const updateApp = isNpm ? "npx" : "yarn";
|
|
15
|
+
const dirName = new URL(".", import.meta.url).pathname;
|
|
16
|
+
const dirPath = pathResolve(dirName, "../..");
|
|
17
|
+
const updateOptions = isNpm ? [
|
|
18
|
+
"npm-check-updates",
|
|
19
|
+
"--concurrency",
|
|
20
|
+
"10",
|
|
21
|
+
"--packageManager",
|
|
22
|
+
packageManager,
|
|
23
|
+
"--pre",
|
|
24
|
+
"0",
|
|
25
|
+
"--target",
|
|
26
|
+
"latest",
|
|
27
|
+
...cmd.interactive ? ["--interactive"] : [],
|
|
28
|
+
"--upgrade"
|
|
29
|
+
] : [cmd.interactive ? "upgrade-interactive" : "upgrade", "--latest"];
|
|
30
|
+
if (registry) {
|
|
31
|
+
updateOptions.push("--registry", registry);
|
|
32
|
+
}
|
|
33
|
+
try {
|
|
34
|
+
await execa(updateApp, updateOptions, {
|
|
35
|
+
encoding: "utf8",
|
|
36
|
+
stdio: "inherit"
|
|
37
|
+
});
|
|
38
|
+
if (isNpm) {
|
|
39
|
+
await execa("npm", ["i", "--force"], {
|
|
40
|
+
encoding: "utf8",
|
|
41
|
+
stdio: "inherit"
|
|
42
|
+
});
|
|
43
|
+
await execa("npm", ["audit", "fix"], {
|
|
44
|
+
encoding: "utf8",
|
|
45
|
+
stdio: "inherit"
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
spinner.succeed("Successfully updated packages!");
|
|
49
|
+
callback(0);
|
|
50
|
+
return 0;
|
|
51
|
+
} catch (error) {
|
|
52
|
+
log(`
|
|
53
|
+
${cliName} Error: ${error.message}`, "error", quiet);
|
|
54
|
+
spinner.fail("Failed to updated packages.");
|
|
55
|
+
callback(1);
|
|
56
|
+
return 1;
|
|
57
|
+
}
|
|
58
|
+
};
|
|
59
|
+
export {
|
|
60
|
+
update
|
|
61
|
+
};
|
|
62
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2NvbW1hbmRzL3VwZGF0ZS91cGRhdGUudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE4LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5pbXBvcnQge2V4ZWNhfSBmcm9tICdleGVjYSc7XG5pbXBvcnQge3Jlc29sdmUgYXMgcGF0aFJlc29sdmV9IGZyb20gJ3BhdGgnO1xuXG5pbXBvcnQge0xleENvbmZpZ30gZnJvbSAnLi4vLi4vTGV4Q29uZmlnLmpzJztcbmltcG9ydCB7Y3JlYXRlU3Bpbm5lcn0gZnJvbSAnLi4vLi4vdXRpbHMvYXBwLmpzJztcbmltcG9ydCB7bG9nfSBmcm9tICcuLi8uLi91dGlscy9sb2cuanMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFVwZGF0ZU9wdGlvbnMge1xuICByZWFkb25seSBjbGlOYW1lPzogc3RyaW5nO1xuICByZWFkb25seSBpbnRlcmFjdGl2ZT86IGJvb2xlYW47XG4gIHJlYWRvbmx5IHBhY2thZ2VNYW5hZ2VyPzogc3RyaW5nO1xuICByZWFkb25seSBxdWlldD86IGJvb2xlYW47XG4gIHJlYWRvbmx5IHJlZ2lzdHJ5Pzogc3RyaW5nO1xufVxuXG5leHBvcnQgdHlwZSBVcGRhdGVDYWxsYmFjayA9IHR5cGVvZiBwcm9jZXNzLmV4aXQ7XG5cbmV4cG9ydCBjb25zdCB1cGRhdGUgPSBhc3luYyAoY21kOiBVcGRhdGVPcHRpb25zLCBjYWxsYmFjazogVXBkYXRlQ2FsbGJhY2sgPSBwcm9jZXNzLmV4aXQpOiBQcm9taXNlPG51bWJlcj4gPT4ge1xuICBjb25zdCB7Y2xpTmFtZSA9ICdMZXgnLCBwYWNrYWdlTWFuYWdlcjogY21kUGFja2FnZU1hbmFnZXIsIHF1aWV0LCByZWdpc3RyeX0gPSBjbWQ7XG5cbiAgbG9nKGAke2NsaU5hbWV9IHVwZGF0aW5nIHBhY2thZ2VzLi4uYCwgJ2luZm8nLCBxdWlldCk7XG5cbiAgY29uc3Qgc3Bpbm5lciA9IGNyZWF0ZVNwaW5uZXIocXVpZXQpO1xuXG4gIGF3YWl0IExleENvbmZpZy5wYXJzZUNvbmZpZyhjbWQpO1xuXG4gIGNvbnN0IHtwYWNrYWdlTWFuYWdlcjogY29uZmlnUGFja2FnZU1hbmFnZXJ9ID0gTGV4Q29uZmlnLmNvbmZpZztcbiAgY29uc3QgcGFja2FnZU1hbmFnZXI6IHN0cmluZyA9IGNtZFBhY2thZ2VNYW5hZ2VyIHx8IGNvbmZpZ1BhY2thZ2VNYW5hZ2VyIHx8ICducG0nO1xuICBjb25zdCBpc05wbTogYm9vbGVhbiA9IHBhY2thZ2VNYW5hZ2VyID09PSAnbnBtJztcbiAgY29uc3QgdXBkYXRlQXBwOiBzdHJpbmcgPSBpc05wbSA/ICducHgnIDogJ3lhcm4nO1xuICBjb25zdCBkaXJOYW1lID0gbmV3IFVSTCgnLicsIGltcG9ydC5tZXRhLnVybCkucGF0aG5hbWU7XG4gIGNvbnN0IGRpclBhdGg6IHN0cmluZyA9IHBhdGhSZXNvbHZlKGRpck5hbWUsICcuLi8uLicpO1xuICBcbiAgY29uc3QgdXBkYXRlT3B0aW9uczogc3RyaW5nW10gPSBpc05wbVxuICAgID8gW1xuICAgICAgJ25wbS1jaGVjay11cGRhdGVzJyxcbiAgICAgICctLWNvbmN1cnJlbmN5JywgJzEwJyxcbiAgICAgICctLXBhY2thZ2VNYW5hZ2VyJywgcGFja2FnZU1hbmFnZXIsXG4gICAgICAnLS1wcmUnLCAnMCcsXG4gICAgICAnLS10YXJnZXQnLCAnbGF0ZXN0JyxcbiAgICAgIC4uLihjbWQuaW50ZXJhY3RpdmUgPyBbJy0taW50ZXJhY3RpdmUnXSA6IFtdKSxcbiAgICAgICctLXVwZ3JhZGUnXG4gICAgXVxuICAgIDogW2NtZC5pbnRlcmFjdGl2ZSA/ICd1cGdyYWRlLWludGVyYWN0aXZlJyA6ICd1cGdyYWRlJywgJy0tbGF0ZXN0J107XG5cbiAgaWYocmVnaXN0cnkpIHtcbiAgICB1cGRhdGVPcHRpb25zLnB1c2goJy0tcmVnaXN0cnknLCByZWdpc3RyeSk7XG4gIH1cblxuICB0cnkge1xuICAgIGF3YWl0IGV4ZWNhKHVwZGF0ZUFwcCwgdXBkYXRlT3B0aW9ucywge1xuICAgICAgZW5jb2Rpbmc6ICd1dGY4JyxcbiAgICAgIHN0ZGlvOiAnaW5oZXJpdCdcbiAgICB9KTtcblxuICAgIGlmKGlzTnBtKSB7XG4gICAgICBhd2FpdCBleGVjYSgnbnBtJywgWydpJywgJy0tZm9yY2UnXSwge1xuICAgICAgICBlbmNvZGluZzogJ3V0ZjgnLFxuICAgICAgICBzdGRpbzogJ2luaGVyaXQnXG4gICAgICB9KTtcblxuICAgICAgYXdhaXQgZXhlY2EoJ25wbScsIFsnYXVkaXQnLCAnZml4J10sIHtcbiAgICAgICAgZW5jb2Rpbmc6ICd1dGY4JyxcbiAgICAgICAgc3RkaW86ICdpbmhlcml0J1xuICAgICAgfSk7XG4gICAgfVxuXG4gICAgc3Bpbm5lci5zdWNjZWVkKCdTdWNjZXNzZnVsbHkgdXBkYXRlZCBwYWNrYWdlcyEnKTtcblxuICAgIGNhbGxiYWNrKDApO1xuICAgIHJldHVybiAwO1xuICB9IGNhdGNoKGVycm9yKSB7XG4gICAgbG9nKGBcXG4ke2NsaU5hbWV9IEVycm9yOiAke2Vycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuXG4gICAgc3Bpbm5lci5mYWlsKCdGYWlsZWQgdG8gdXBkYXRlZCBwYWNrYWdlcy4nKTtcblxuICAgIGNhbGxiYWNrKDEpO1xuICAgIHJldHVybiAxO1xuICB9XG59OyJdLAogICJtYXBwaW5ncyI6ICJBQUlBLFNBQVEsYUFBWTtBQUNwQixTQUFRLFdBQVcsbUJBQWtCO0FBRXJDLFNBQVEsaUJBQWdCO0FBQ3hCLFNBQVEscUJBQW9CO0FBQzVCLFNBQVEsV0FBVTtBQVlYLE1BQU0sU0FBUyxPQUFPLEtBQW9CLFdBQTJCLFFBQVEsU0FBMEI7QUFDNUcsUUFBTSxFQUFDLFVBQVUsT0FBTyxnQkFBZ0IsbUJBQW1CLE9BQU8sU0FBUSxJQUFJO0FBRTlFLE1BQUksR0FBRyxPQUFPLHlCQUF5QixRQUFRLEtBQUs7QUFFcEQsUUFBTSxVQUFVLGNBQWMsS0FBSztBQUVuQyxRQUFNLFVBQVUsWUFBWSxHQUFHO0FBRS9CLFFBQU0sRUFBQyxnQkFBZ0IscUJBQW9CLElBQUksVUFBVTtBQUN6RCxRQUFNLGlCQUF5QixxQkFBcUIsd0JBQXdCO0FBQzVFLFFBQU0sUUFBaUIsbUJBQW1CO0FBQzFDLFFBQU0sWUFBb0IsUUFBUSxRQUFRO0FBQzFDLFFBQU0sVUFBVSxJQUFJLElBQUksS0FBSyxZQUFZLEdBQUcsRUFBRTtBQUM5QyxRQUFNLFVBQWtCLFlBQVksU0FBUyxPQUFPO0FBRXBELFFBQU0sZ0JBQTBCLFFBQzVCO0FBQUEsSUFDQTtBQUFBLElBQ0E7QUFBQSxJQUFpQjtBQUFBLElBQ2pCO0FBQUEsSUFBb0I7QUFBQSxJQUNwQjtBQUFBLElBQVM7QUFBQSxJQUNUO0FBQUEsSUFBWTtBQUFBLElBQ1osR0FBSSxJQUFJLGNBQWMsQ0FBQyxlQUFlLElBQUksQ0FBQztBQUFBLElBQzNDO0FBQUEsRUFDRixJQUNFLENBQUMsSUFBSSxjQUFjLHdCQUF3QixXQUFXLFVBQVU7QUFFcEUsTUFBRyxVQUFVO0FBQ1gsa0JBQWMsS0FBSyxjQUFjLFFBQVE7QUFBQSxFQUMzQztBQUVBLE1BQUk7QUFDRixVQUFNLE1BQU0sV0FBVyxlQUFlO0FBQUEsTUFDcEMsVUFBVTtBQUFBLE1BQ1YsT0FBTztBQUFBLElBQ1QsQ0FBQztBQUVELFFBQUcsT0FBTztBQUNSLFlBQU0sTUFBTSxPQUFPLENBQUMsS0FBSyxTQUFTLEdBQUc7QUFBQSxRQUNuQyxVQUFVO0FBQUEsUUFDVixPQUFPO0FBQUEsTUFDVCxDQUFDO0FBRUQsWUFBTSxNQUFNLE9BQU8sQ0FBQyxTQUFTLEtBQUssR0FBRztBQUFBLFFBQ25DLFVBQVU7QUFBQSxRQUNWLE9BQU87QUFBQSxNQUNULENBQUM7QUFBQSxJQUNIO0FBRUEsWUFBUSxRQUFRLGdDQUFnQztBQUVoRCxhQUFTLENBQUM7QUFDVixXQUFPO0FBQUEsRUFDVCxTQUFRLE9BQU87QUFDYixRQUFJO0FBQUEsRUFBSyxPQUFPLFdBQVcsTUFBTSxPQUFPLElBQUksU0FBUyxLQUFLO0FBRTFELFlBQVEsS0FBSyw2QkFBNkI7QUFFMUMsYUFBUyxDQUFDO0FBQ1YsV0FBTztBQUFBLEVBQ1Q7QUFDRjsiLAogICJuYW1lcyI6IFtdCn0K
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import { compareVersions } from "compare-versions";
|
|
2
|
+
import { execa } from "execa";
|
|
3
|
+
import { readFileSync } from "fs";
|
|
4
|
+
import latestVersion from "latest-version";
|
|
5
|
+
import { fileURLToPath } from "url";
|
|
6
|
+
import { LexConfig } from "../../LexConfig.js";
|
|
7
|
+
import { createSpinner } from "../../utils/app.js";
|
|
8
|
+
import { log } from "../../utils/log.js";
|
|
9
|
+
import { parseVersion } from "../versions/versions.js";
|
|
10
|
+
const packagePath = fileURLToPath(new URL("../../../package.json", import.meta.url));
|
|
11
|
+
const packageJson = JSON.parse(readFileSync(packagePath, "utf8"));
|
|
12
|
+
const upgrade = async (cmd, callback = process.exit) => {
|
|
13
|
+
const { cliName = "Lex", cliPackage = "@nlabs/lex", quiet } = cmd;
|
|
14
|
+
log(`Upgrading ${cliName}...`, "info", quiet);
|
|
15
|
+
const spinner = createSpinner(quiet);
|
|
16
|
+
await LexConfig.parseConfig(cmd);
|
|
17
|
+
return latestVersion("@nlabs/lex").then(async (latest) => {
|
|
18
|
+
const current = parseVersion(packageJson.version);
|
|
19
|
+
const versionDiff = compareVersions(latest, current);
|
|
20
|
+
if (versionDiff === 0) {
|
|
21
|
+
log(`
|
|
22
|
+
Currently up-to-date. Version ${latest} is the latest.`, "note", quiet);
|
|
23
|
+
callback(0);
|
|
24
|
+
return 0;
|
|
25
|
+
}
|
|
26
|
+
log(`
|
|
27
|
+
Currently out of date. Upgrading from version ${current} to ${latest}...`, "note", quiet);
|
|
28
|
+
const upgradeOptions = ["install", "-g", `${cliPackage}@latest`];
|
|
29
|
+
await execa("npm", upgradeOptions, {
|
|
30
|
+
encoding: "utf8",
|
|
31
|
+
stdio: "inherit"
|
|
32
|
+
});
|
|
33
|
+
spinner.succeed(`Successfully updated ${cliName}!`);
|
|
34
|
+
callback(0);
|
|
35
|
+
return 0;
|
|
36
|
+
}).catch((error) => {
|
|
37
|
+
log(`
|
|
38
|
+
${cliName} Error: ${error.message}`, "error", quiet);
|
|
39
|
+
spinner.fail("Failed to updated packages.");
|
|
40
|
+
callback(1);
|
|
41
|
+
return 1;
|
|
42
|
+
});
|
|
43
|
+
};
|
|
44
|
+
export {
|
|
45
|
+
upgrade
|
|
46
|
+
};
|
|
47
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2NvbW1hbmRzL3VwZ3JhZGUvdXBncmFkZS50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTgtUHJlc2VudCwgTml0cm9nZW4gTGFicywgSW5jLlxuICogQ29weXJpZ2h0cyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSB0aGUgYWNjb21wYW55aW5nIExJQ0VOU0UgZmlsZSBmb3IgdGVybXMuXG4gKi9cbmltcG9ydCB7Y29tcGFyZVZlcnNpb25zfSBmcm9tICdjb21wYXJlLXZlcnNpb25zJztcbmltcG9ydCB7ZXhlY2F9IGZyb20gJ2V4ZWNhJztcbmltcG9ydCB7cmVhZEZpbGVTeW5jfSBmcm9tICdmcyc7XG5pbXBvcnQgbGF0ZXN0VmVyc2lvbiBmcm9tICdsYXRlc3QtdmVyc2lvbic7XG5pbXBvcnQge2ZpbGVVUkxUb1BhdGh9IGZyb20gJ3VybCc7XG5cbmltcG9ydCB7TGV4Q29uZmlnfSBmcm9tICcuLi8uLi9MZXhDb25maWcuanMnO1xuaW1wb3J0IHtjcmVhdGVTcGlubmVyfSBmcm9tICcuLi8uLi91dGlscy9hcHAuanMnO1xuaW1wb3J0IHtsb2d9IGZyb20gJy4uLy4uL3V0aWxzL2xvZy5qcyc7XG5pbXBvcnQge3BhcnNlVmVyc2lvbn0gZnJvbSAnLi4vdmVyc2lvbnMvdmVyc2lvbnMuanMnO1xuXG5jb25zdCBwYWNrYWdlUGF0aCA9IGZpbGVVUkxUb1BhdGgobmV3IFVSTCgnLi4vLi4vLi4vcGFja2FnZS5qc29uJywgaW1wb3J0Lm1ldGEudXJsKSk7XG5jb25zdCBwYWNrYWdlSnNvbiA9IEpTT04ucGFyc2UocmVhZEZpbGVTeW5jKHBhY2thZ2VQYXRoLCAndXRmOCcpKTtcblxuZXhwb3J0IGludGVyZmFjZSBVcGdyYWRlT3B0aW9ucyB7XG4gIHJlYWRvbmx5IGNsaU5hbWU/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IGNsaVBhY2thZ2U/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IHF1aWV0PzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IHR5cGUgVXBncmFkZUNhbGxiYWNrID0gdHlwZW9mIHByb2Nlc3MuZXhpdDtcblxuZXhwb3J0IGNvbnN0IHVwZ3JhZGUgPSBhc3luYyAoY21kOiBVcGdyYWRlT3B0aW9ucywgY2FsbGJhY2s6IFVwZ3JhZGVDYWxsYmFjayA9IHByb2Nlc3MuZXhpdCk6IFByb21pc2U8bnVtYmVyPiA9PiB7XG4gIGNvbnN0IHtjbGlOYW1lID0gJ0xleCcsIGNsaVBhY2thZ2UgPSAnQG5sYWJzL2xleCcsIHF1aWV0fSA9IGNtZDtcblxuICAvLyBEaXNwbGF5IHN0YXR1c1xuICBsb2coYFVwZ3JhZGluZyAke2NsaU5hbWV9Li4uYCwgJ2luZm8nLCBxdWlldCk7XG5cbiAgLy8gU3Bpbm5lclxuICBjb25zdCBzcGlubmVyID0gY3JlYXRlU3Bpbm5lcihxdWlldCk7XG5cbiAgLy8gR2V0IGN1c3RvbSBjb25maWd1cmF0aW9uXG4gIGF3YWl0IExleENvbmZpZy5wYXJzZUNvbmZpZyhjbWQpO1xuXG4gIHJldHVybiBsYXRlc3RWZXJzaW9uKCdAbmxhYnMvbGV4JylcbiAgICAudGhlbihhc3luYyAobGF0ZXN0OiBzdHJpbmcpID0+IHtcbiAgICAgIGNvbnN0IGN1cnJlbnQ6IHN0cmluZyA9IHBhcnNlVmVyc2lvbihwYWNrYWdlSnNvbi52ZXJzaW9uKTtcbiAgICAgIGNvbnN0IHZlcnNpb25EaWZmOiBudW1iZXIgPSBjb21wYXJlVmVyc2lvbnMobGF0ZXN0LCBjdXJyZW50KTtcblxuICAgICAgaWYodmVyc2lvbkRpZmYgPT09IDApIHtcbiAgICAgICAgbG9nKGBcXG5DdXJyZW50bHkgdXAtdG8tZGF0ZS4gVmVyc2lvbiAke2xhdGVzdH0gaXMgdGhlIGxhdGVzdC5gLCAnbm90ZScsIHF1aWV0KTtcbiAgICAgICAgY2FsbGJhY2soMCk7XG4gICAgICAgIHJldHVybiAwO1xuICAgICAgfVxuXG4gICAgICBsb2coYFxcbkN1cnJlbnRseSBvdXQgb2YgZGF0ZS4gVXBncmFkaW5nIGZyb20gdmVyc2lvbiAke2N1cnJlbnR9IHRvICR7bGF0ZXN0fS4uLmAsICdub3RlJywgcXVpZXQpO1xuXG4gICAgICBjb25zdCB1cGdyYWRlT3B0aW9uczogc3RyaW5nW10gPSBbJ2luc3RhbGwnLCAnLWcnLCBgJHtjbGlQYWNrYWdlfUBsYXRlc3RgXTtcblxuICAgICAgYXdhaXQgZXhlY2EoJ25wbScsIHVwZ3JhZGVPcHRpb25zLCB7XG4gICAgICAgIGVuY29kaW5nOiAndXRmOCcsXG4gICAgICAgIHN0ZGlvOiAnaW5oZXJpdCdcbiAgICAgIH0pO1xuXG4gICAgICAvLyBTdG9wIGxvYWRlclxuICAgICAgc3Bpbm5lci5zdWNjZWVkKGBTdWNjZXNzZnVsbHkgdXBkYXRlZCAke2NsaU5hbWV9IWApO1xuXG4gICAgICAvLyBTdG9wIHByb2Nlc3NcbiAgICAgIGNhbGxiYWNrKDApO1xuICAgICAgcmV0dXJuIDA7XG4gICAgfSlcbiAgICAuY2F0Y2goKGVycm9yKSA9PiB7XG4gICAgICAvLyBEaXNwbGF5IGVycm9yIG1lc3NhZ2VcbiAgICAgIGxvZyhgXFxuJHtjbGlOYW1lfSBFcnJvcjogJHtlcnJvci5tZXNzYWdlfWAsICdlcnJvcicsIHF1aWV0KTtcblxuICAgICAgLy8gU3RvcCBzcGlubmVyXG4gICAgICBzcGlubmVyLmZhaWwoJ0ZhaWxlZCB0byB1cGRhdGVkIHBhY2thZ2VzLicpO1xuXG4gICAgICAvLyBLaWxsIHByb2Nlc3NcbiAgICAgIGNhbGxiYWNrKDEpO1xuICAgICAgcmV0dXJuIDE7XG4gICAgfSk7XG59OyJdLAogICJtYXBwaW5ncyI6ICJBQUlBLFNBQVEsdUJBQXNCO0FBQzlCLFNBQVEsYUFBWTtBQUNwQixTQUFRLG9CQUFtQjtBQUMzQixPQUFPLG1CQUFtQjtBQUMxQixTQUFRLHFCQUFvQjtBQUU1QixTQUFRLGlCQUFnQjtBQUN4QixTQUFRLHFCQUFvQjtBQUM1QixTQUFRLFdBQVU7QUFDbEIsU0FBUSxvQkFBbUI7QUFFM0IsTUFBTSxjQUFjLGNBQWMsSUFBSSxJQUFJLHlCQUF5QixZQUFZLEdBQUcsQ0FBQztBQUNuRixNQUFNLGNBQWMsS0FBSyxNQUFNLGFBQWEsYUFBYSxNQUFNLENBQUM7QUFVekQsTUFBTSxVQUFVLE9BQU8sS0FBcUIsV0FBNEIsUUFBUSxTQUEwQjtBQUMvRyxRQUFNLEVBQUMsVUFBVSxPQUFPLGFBQWEsY0FBYyxNQUFLLElBQUk7QUFHNUQsTUFBSSxhQUFhLE9BQU8sT0FBTyxRQUFRLEtBQUs7QUFHNUMsUUFBTSxVQUFVLGNBQWMsS0FBSztBQUduQyxRQUFNLFVBQVUsWUFBWSxHQUFHO0FBRS9CLFNBQU8sY0FBYyxZQUFZLEVBQzlCLEtBQUssT0FBTyxXQUFtQjtBQUM5QixVQUFNLFVBQWtCLGFBQWEsWUFBWSxPQUFPO0FBQ3hELFVBQU0sY0FBc0IsZ0JBQWdCLFFBQVEsT0FBTztBQUUzRCxRQUFHLGdCQUFnQixHQUFHO0FBQ3BCLFVBQUk7QUFBQSxnQ0FBbUMsTUFBTSxtQkFBbUIsUUFBUSxLQUFLO0FBQzdFLGVBQVMsQ0FBQztBQUNWLGFBQU87QUFBQSxJQUNUO0FBRUEsUUFBSTtBQUFBLGdEQUFtRCxPQUFPLE9BQU8sTUFBTSxPQUFPLFFBQVEsS0FBSztBQUUvRixVQUFNLGlCQUEyQixDQUFDLFdBQVcsTUFBTSxHQUFHLFVBQVUsU0FBUztBQUV6RSxVQUFNLE1BQU0sT0FBTyxnQkFBZ0I7QUFBQSxNQUNqQyxVQUFVO0FBQUEsTUFDVixPQUFPO0FBQUEsSUFDVCxDQUFDO0FBR0QsWUFBUSxRQUFRLHdCQUF3QixPQUFPLEdBQUc7QUFHbEQsYUFBUyxDQUFDO0FBQ1YsV0FBTztBQUFBLEVBQ1QsQ0FBQyxFQUNBLE1BQU0sQ0FBQyxVQUFVO0FBRWhCLFFBQUk7QUFBQSxFQUFLLE9BQU8sV0FBVyxNQUFNLE9BQU8sSUFBSSxTQUFTLEtBQUs7QUFHMUQsWUFBUSxLQUFLLDZCQUE2QjtBQUcxQyxhQUFTLENBQUM7QUFDVixXQUFPO0FBQUEsRUFDVCxDQUFDO0FBQ0w7IiwKICAibmFtZXMiOiBbXQp9Cg==
|