@nlabs/lex 1.48.6 → 1.49.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/.storybook/main.ts +9 -2
- package/.vscode/settings.json +1 -6
- package/README.md +276 -4
- package/eslint.config.mjs +24 -0
- package/examples/lex.config.js +18 -8
- package/examples/serverless-example/README.md +109 -0
- package/examples/serverless-example/dist/handlers/echo.js +15 -0
- package/examples/serverless-example/dist/handlers/graphql.js +137 -0
- package/examples/serverless-example/dist/handlers/hello.js +15 -0
- package/examples/serverless-example/dist/handlers/test.js +17 -0
- package/examples/serverless-example/dist/handlers/websocket.js +14 -0
- package/examples/serverless-example/lex.config.mjs +74 -0
- package/jest.config.mjs +13 -12
- package/{dist → lib}/LexConfig.d.ts +14 -6
- package/lib/LexConfig.js +268 -0
- package/lib/commands/ai/ai.js +303 -0
- package/{dist → lib}/commands/build/build.d.ts +3 -0
- package/lib/commands/build/build.js +494 -0
- package/{dist → lib}/commands/clean/clean.js +1 -1
- package/lib/commands/compile/compile.js +241 -0
- package/lib/commands/copy/copy.js +38 -0
- package/{dist → lib}/commands/create/create.js +1 -1
- package/{dist → lib}/commands/dev/dev.d.ts +2 -0
- package/lib/commands/dev/dev.js +286 -0
- package/{dist → lib}/commands/init/init.js +1 -1
- package/{dist → lib}/commands/lint/lint.d.ts +4 -1
- package/lib/commands/lint/lint.js +993 -0
- package/{dist → lib}/commands/migrate/migrate.js +1 -1
- package/lib/commands/publish/publish.js +104 -0
- package/lib/commands/serverless/serverless.d.ts +17 -0
- package/lib/commands/serverless/serverless.js +662 -0
- package/lib/commands/storybook/storybook.js +249 -0
- package/lib/commands/test/test.js +428 -0
- package/lib/commands/update/update.js +128 -0
- package/lib/commands/versions/versions.js +41 -0
- package/{dist → lib}/create/changelog.js +1 -1
- package/{dist → lib}/index.d.ts +2 -0
- package/{dist → lib}/index.js +3 -1
- package/lib/lex.js +73 -0
- package/lib/storybook/index.d.ts +5 -0
- package/lib/types.js +1 -0
- package/lib/utils/aiService.d.ts +9 -0
- package/lib/utils/aiService.js +299 -0
- package/{dist → lib}/utils/app.d.ts +3 -0
- package/lib/utils/app.js +296 -0
- package/lib/utils/deepMerge.js +26 -0
- package/{dist → lib}/utils/file.d.ts +7 -3
- package/lib/utils/file.js +229 -0
- package/lib/utils/translations.d.ts +1 -0
- package/lib/utils/translations.js +74 -0
- package/package.json +62 -50
- package/postcss.config.js +5 -3
- package/tsconfig.build.json +2 -2
- package/webpack.config.js +229 -39
- package/dist/LexConfig.js +0 -286
- package/dist/commands/ai/ai.js +0 -303
- package/dist/commands/build/build.js +0 -404
- package/dist/commands/compile/compile.js +0 -234
- package/dist/commands/copy/copy.js +0 -38
- package/dist/commands/dev/dev.js +0 -74
- package/dist/commands/lint/lint.js +0 -811
- package/dist/commands/publish/publish.js +0 -104
- package/dist/commands/storybook/storybook.js +0 -249
- package/dist/commands/test/test.js +0 -429
- package/dist/commands/update/update.js +0 -132
- package/dist/commands/versions/versions.js +0 -41
- package/dist/lex.js +0 -70
- package/dist/utils/aiService.d.ts +0 -9
- package/dist/utils/aiService.js +0 -299
- package/dist/utils/app.js +0 -267
- package/dist/utils/deepMerge.js +0 -24
- package/dist/utils/file.js +0 -185
- package/emptyModule.js +0 -0
- package/eslint.config.js +0 -7
- /package/{dist → lib}/Button.stories.d.ts +0 -0
- /package/{dist → lib}/commands/ai/ai.d.ts +0 -0
- /package/{dist → lib}/commands/ai/index.d.ts +0 -0
- /package/{dist → lib}/commands/ai/index.js +0 -0
- /package/{dist → lib}/commands/clean/clean.d.ts +0 -0
- /package/{dist → lib}/commands/compile/compile.d.ts +0 -0
- /package/{dist → lib}/commands/config/config.d.ts +0 -0
- /package/{dist → lib}/commands/config/config.js +0 -0
- /package/{dist → lib}/commands/copy/copy.d.ts +0 -0
- /package/{dist → lib}/commands/create/create.d.ts +0 -0
- /package/{dist → lib}/commands/init/init.d.ts +0 -0
- /package/{dist → lib}/commands/link/link.d.ts +0 -0
- /package/{dist → lib}/commands/link/link.js +0 -0
- /package/{dist → lib}/commands/lint/autofix.d.ts +0 -0
- /package/{dist → lib}/commands/migrate/migrate.d.ts +0 -0
- /package/{dist → lib}/commands/publish/publish.d.ts +0 -0
- /package/{dist → lib}/commands/storybook/storybook.d.ts +0 -0
- /package/{dist → lib}/commands/test/test.d.ts +0 -0
- /package/{dist → lib}/commands/update/update.d.ts +0 -0
- /package/{dist → lib}/commands/upgrade/upgrade.d.ts +0 -0
- /package/{dist → lib}/commands/upgrade/upgrade.js +0 -0
- /package/{dist → lib}/commands/versions/versions.d.ts +0 -0
- /package/{dist → lib}/create/changelog.d.ts +0 -0
- /package/{dist → lib}/lex.d.ts +0 -0
- /package/{dist/types.js → lib/storybook/index.js} +0 -0
- /package/{dist → lib}/test-react/index.d.ts +0 -0
- /package/{dist → lib}/test-react/index.js +0 -0
- /package/{dist → lib}/types.d.ts +0 -0
- /package/{dist → lib}/utils/deepMerge.d.ts +0 -0
- /package/{dist → lib}/utils/log.d.ts +0 -0
- /package/{dist → lib}/utils/log.js +0 -0
- /package/{dist → lib}/utils/reactShim.d.ts +0 -0
- /package/{dist → lib}/utils/reactShim.js +0 -0
package/lib/LexConfig.js
ADDED
|
@@ -0,0 +1,268 @@
|
|
|
1
|
+
import { existsSync, readFileSync, writeFileSync } from "fs";
|
|
2
|
+
import { dirname, extname as pathExtname, resolve as pathResolve } from "path";
|
|
3
|
+
import { URL } from "url";
|
|
4
|
+
import { getDirName, getLexPackageJsonPath, relativeFilePath } from "./utils/file.js";
|
|
5
|
+
import { log } from "./utils/log.js";
|
|
6
|
+
const cwd = process.cwd();
|
|
7
|
+
const defaultConfigValues = {
|
|
8
|
+
ai: {
|
|
9
|
+
maxTokens: 4e3,
|
|
10
|
+
model: "gpt-4o",
|
|
11
|
+
provider: "none",
|
|
12
|
+
temperature: 0.1
|
|
13
|
+
},
|
|
14
|
+
configFiles: [],
|
|
15
|
+
copyFiles: [],
|
|
16
|
+
entryHTML: "index.html",
|
|
17
|
+
entryJs: "index.js",
|
|
18
|
+
env: null,
|
|
19
|
+
esbuild: {
|
|
20
|
+
drop: ["console", "debugger"],
|
|
21
|
+
legalComments: "none",
|
|
22
|
+
metafile: false,
|
|
23
|
+
minify: true,
|
|
24
|
+
pure: ["console.log", "console.warn", "console.error"],
|
|
25
|
+
sourcemap: false,
|
|
26
|
+
splitting: true,
|
|
27
|
+
treeShaking: true
|
|
28
|
+
},
|
|
29
|
+
eslint: {},
|
|
30
|
+
jest: {},
|
|
31
|
+
outputFullPath: pathResolve(cwd, "./dist"),
|
|
32
|
+
outputHash: false,
|
|
33
|
+
outputPath: "./dist",
|
|
34
|
+
packageManager: "npm",
|
|
35
|
+
preset: "web",
|
|
36
|
+
sourceFullPath: pathResolve(cwd, "./src"),
|
|
37
|
+
sourcePath: "./src",
|
|
38
|
+
targetEnvironment: "web",
|
|
39
|
+
useGraphQl: false,
|
|
40
|
+
useTypescript: false,
|
|
41
|
+
webpack: {
|
|
42
|
+
publicPath: "./src/static"
|
|
43
|
+
}
|
|
44
|
+
};
|
|
45
|
+
const getTypeScriptConfigPath = (configName) => {
|
|
46
|
+
const cwd2 = process.cwd();
|
|
47
|
+
if (configName === "tsconfig.build.json") {
|
|
48
|
+
const projectBuildConfig = pathResolve(cwd2, "tsconfig.build.json");
|
|
49
|
+
if (existsSync(projectBuildConfig)) {
|
|
50
|
+
return projectBuildConfig;
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
if (configName === "tsconfig.lint.json") {
|
|
54
|
+
const projectLintConfig = pathResolve(cwd2, "tsconfig.eslint.json");
|
|
55
|
+
if (existsSync(projectLintConfig)) {
|
|
56
|
+
return projectLintConfig;
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
if (configName === "tsconfig.test.json") {
|
|
60
|
+
const projectTestConfig = pathResolve(cwd2, "tsconfig.test.json");
|
|
61
|
+
if (existsSync(projectTestConfig)) {
|
|
62
|
+
return projectTestConfig;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
const projectConfigPath = pathResolve(cwd2, configName);
|
|
66
|
+
if (existsSync(projectConfigPath)) {
|
|
67
|
+
return projectConfigPath;
|
|
68
|
+
}
|
|
69
|
+
const lexDir = LexConfig.getLexDir();
|
|
70
|
+
return pathResolve(lexDir, configName);
|
|
71
|
+
};
|
|
72
|
+
class LexConfig {
|
|
73
|
+
static config = {
|
|
74
|
+
...defaultConfigValues
|
|
75
|
+
};
|
|
76
|
+
static set useTypescript(value) {
|
|
77
|
+
LexConfig.config.useTypescript = value;
|
|
78
|
+
const { sourceFullPath } = LexConfig.config;
|
|
79
|
+
const { entryJs } = LexConfig.config;
|
|
80
|
+
if (entryJs === "index.js" && value) {
|
|
81
|
+
const indexPath = pathResolve(cwd, sourceFullPath, "index.tsx");
|
|
82
|
+
const hasIndexTsx = existsSync(indexPath);
|
|
83
|
+
if (hasIndexTsx) {
|
|
84
|
+
LexConfig.config.entryJs = "index.tsx";
|
|
85
|
+
} else {
|
|
86
|
+
LexConfig.config.entryJs = "index.ts";
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
static getLexDir() {
|
|
91
|
+
return dirname(getLexPackageJsonPath());
|
|
92
|
+
}
|
|
93
|
+
static updateConfig(updatedConfig) {
|
|
94
|
+
const { outputFullPath, outputPath, sourcePath, sourceFullPath, useTypescript, ai } = updatedConfig;
|
|
95
|
+
const cwd2 = process.cwd();
|
|
96
|
+
if (useTypescript !== void 0) {
|
|
97
|
+
LexConfig.useTypescript = useTypescript;
|
|
98
|
+
}
|
|
99
|
+
if (outputPath !== void 0 && outputFullPath === void 0) {
|
|
100
|
+
updatedConfig.outputFullPath = pathResolve(cwd2, outputPath);
|
|
101
|
+
}
|
|
102
|
+
if (sourcePath !== void 0 && sourceFullPath === void 0) {
|
|
103
|
+
updatedConfig.sourceFullPath = pathResolve(cwd2, sourcePath);
|
|
104
|
+
}
|
|
105
|
+
if (ai) {
|
|
106
|
+
LexConfig.config.ai = { ...LexConfig.config.ai, ...ai };
|
|
107
|
+
if (process.env.CURSOR_IDE === "true" && LexConfig.config.ai.provider === "none") {
|
|
108
|
+
LexConfig.config.ai.provider = "cursor";
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
LexConfig.config = { ...LexConfig.config, ...updatedConfig };
|
|
112
|
+
return LexConfig.config;
|
|
113
|
+
}
|
|
114
|
+
static addConfigParams(cmd, params) {
|
|
115
|
+
const nameProperty = "_name";
|
|
116
|
+
const { environment, outputPath, sourcePath, typescript } = cmd;
|
|
117
|
+
if (outputPath !== void 0) {
|
|
118
|
+
params.outputPath = outputPath;
|
|
119
|
+
params.outputFullPath = pathResolve(cwd, outputPath);
|
|
120
|
+
}
|
|
121
|
+
if (sourcePath !== void 0) {
|
|
122
|
+
params.sourcePath = sourcePath;
|
|
123
|
+
params.sourceFullPath = pathResolve(cwd, sourcePath);
|
|
124
|
+
}
|
|
125
|
+
if (typescript !== void 0) {
|
|
126
|
+
params.useTypescript = true;
|
|
127
|
+
}
|
|
128
|
+
if (environment !== void 0) {
|
|
129
|
+
params.targetEnvironment = environment === "web" ? "web" : "node";
|
|
130
|
+
}
|
|
131
|
+
process.env.LEX_CONFIG = JSON.stringify(
|
|
132
|
+
{
|
|
133
|
+
...LexConfig.updateConfig(params),
|
|
134
|
+
commandName: cmd[nameProperty],
|
|
135
|
+
isStatic: cmd.static
|
|
136
|
+
},
|
|
137
|
+
null,
|
|
138
|
+
0
|
|
139
|
+
);
|
|
140
|
+
}
|
|
141
|
+
static async parseConfig(cmd, isRoot = true) {
|
|
142
|
+
const { cliName = "Lex", lexConfig, lexConfigName, quiet, typescript, debug = false } = cmd;
|
|
143
|
+
const configFormats = ["js", "mjs", "cjs", "ts", "json"];
|
|
144
|
+
const configBaseName = lexConfigName || "lex.config";
|
|
145
|
+
let configPath = lexConfig || "";
|
|
146
|
+
let configExists = lexConfig ? existsSync(configPath) : false;
|
|
147
|
+
if (!configPath || !configExists) {
|
|
148
|
+
if (debug) {
|
|
149
|
+
log(`Searching for config files with base name: ${configBaseName}`, "info", quiet);
|
|
150
|
+
}
|
|
151
|
+
for (const format of configFormats) {
|
|
152
|
+
const potentialPath = isRoot ? pathResolve(cwd, `./${configBaseName}.${format}`) : relativeFilePath(`${configBaseName}.${format}`, cwd);
|
|
153
|
+
if (debug) {
|
|
154
|
+
log(`Checking for config file: ${potentialPath}`, "info", quiet);
|
|
155
|
+
}
|
|
156
|
+
if (existsSync(potentialPath)) {
|
|
157
|
+
configPath = potentialPath;
|
|
158
|
+
configExists = true;
|
|
159
|
+
break;
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
if (configExists) {
|
|
164
|
+
log(`Using ${cliName} configuration file: ${configPath}`, "note", quiet);
|
|
165
|
+
const ext = pathExtname(configPath);
|
|
166
|
+
if (ext === ".json") {
|
|
167
|
+
const configContent = readFileSync(configPath, "utf8");
|
|
168
|
+
if (configContent) {
|
|
169
|
+
let configJson;
|
|
170
|
+
try {
|
|
171
|
+
configJson = JSON.parse(configContent)?.default || {};
|
|
172
|
+
} catch (error) {
|
|
173
|
+
log(`
|
|
174
|
+
${cliName} Error: Failed to parse JSON config: ${error.message}`, "error", quiet);
|
|
175
|
+
configJson = {};
|
|
176
|
+
}
|
|
177
|
+
LexConfig.addConfigParams(cmd, configJson);
|
|
178
|
+
} else {
|
|
179
|
+
log(`
|
|
180
|
+
${cliName} Error: Config file malformed, ${configPath}`, "error", quiet);
|
|
181
|
+
}
|
|
182
|
+
} else if ([".js", ".mjs", ".cjs", ".ts"].includes(ext)) {
|
|
183
|
+
try {
|
|
184
|
+
let lexCustomConfig;
|
|
185
|
+
if (ext === ".cjs") {
|
|
186
|
+
const fileUrl = new URL(`file:///${pathResolve(configPath)}`).href;
|
|
187
|
+
if (debug) {
|
|
188
|
+
log(`Loading CommonJS config from: ${fileUrl}`, "info", quiet);
|
|
189
|
+
}
|
|
190
|
+
lexCustomConfig = await import(fileUrl);
|
|
191
|
+
} else {
|
|
192
|
+
if (debug) {
|
|
193
|
+
log(`Loading ESM/TS config from: ${configPath}`, "info", quiet);
|
|
194
|
+
}
|
|
195
|
+
lexCustomConfig = await import(configPath);
|
|
196
|
+
}
|
|
197
|
+
const config = lexCustomConfig.default || lexCustomConfig;
|
|
198
|
+
if (debug) {
|
|
199
|
+
log(`Loaded config: ${JSON.stringify(config, null, 2)}`, "info", quiet);
|
|
200
|
+
}
|
|
201
|
+
if (!config) {
|
|
202
|
+
log(`
|
|
203
|
+
${cliName} Warning: Config file loaded but no configuration found`, "warn", quiet);
|
|
204
|
+
}
|
|
205
|
+
LexConfig.addConfigParams(cmd, config || {});
|
|
206
|
+
} catch (error) {
|
|
207
|
+
log(`
|
|
208
|
+
${cliName} Error: Failed to load config file: ${error.message}`, "error", quiet);
|
|
209
|
+
if (debug) {
|
|
210
|
+
console.error(error);
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
} else {
|
|
214
|
+
log(`
|
|
215
|
+
${cliName} Error: Config file must be a JS, CJS, MJS, TS, or JSON file.`, "error", quiet);
|
|
216
|
+
}
|
|
217
|
+
} else {
|
|
218
|
+
if (debug) {
|
|
219
|
+
log("No config file found. Using default configuration.", "info", quiet);
|
|
220
|
+
}
|
|
221
|
+
LexConfig.useTypescript = !!typescript;
|
|
222
|
+
LexConfig.addConfigParams(cmd, LexConfig.config);
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
static checkTypescriptConfig() {
|
|
226
|
+
const tsconfigPath = pathResolve(cwd, "./tsconfig.json");
|
|
227
|
+
if (!existsSync(tsconfigPath)) {
|
|
228
|
+
const dirName = getDirName();
|
|
229
|
+
writeFileSync(tsconfigPath, readFileSync(pathResolve(dirName, "../../../tsconfig.base.json")));
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
static checkCompileTypescriptConfig() {
|
|
233
|
+
const lexDir = LexConfig.getLexDir();
|
|
234
|
+
const tsconfigCompilePath = pathResolve(lexDir, "./tsconfig.build.json");
|
|
235
|
+
if (!existsSync(tsconfigCompilePath)) {
|
|
236
|
+
const templatePath = pathResolve(lexDir, "tsconfig.build.json");
|
|
237
|
+
if (existsSync(templatePath)) {
|
|
238
|
+
writeFileSync(tsconfigCompilePath, readFileSync(templatePath));
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
static checkLintTypescriptConfig() {
|
|
243
|
+
const lexDir = LexConfig.getLexDir();
|
|
244
|
+
const tsconfigLintPath = pathResolve(lexDir, "./tsconfig.lint.json");
|
|
245
|
+
if (!existsSync(tsconfigLintPath)) {
|
|
246
|
+
const templatePath = pathResolve(lexDir, "tsconfig.lint.json");
|
|
247
|
+
if (existsSync(templatePath)) {
|
|
248
|
+
writeFileSync(tsconfigLintPath, readFileSync(templatePath));
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
static checkTestTypescriptConfig() {
|
|
253
|
+
const lexDir = LexConfig.getLexDir();
|
|
254
|
+
const tsconfigTestPath = pathResolve(lexDir, "./tsconfig.test.json");
|
|
255
|
+
if (!existsSync(tsconfigTestPath)) {
|
|
256
|
+
const templatePath = pathResolve(lexDir, "tsconfig.test.json");
|
|
257
|
+
if (existsSync(templatePath)) {
|
|
258
|
+
writeFileSync(tsconfigTestPath, readFileSync(templatePath));
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
export {
|
|
264
|
+
LexConfig,
|
|
265
|
+
defaultConfigValues,
|
|
266
|
+
getTypeScriptConfigPath
|
|
267
|
+
};
|
|
268
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/LexConfig.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 {existsSync, readFileSync, writeFileSync} from 'fs';\nimport {dirname, extname as pathExtname, resolve as pathResolve} from 'path';\nimport {URL} from 'url';\n\nimport {getDirName, getLexPackageJsonPath, relativeFilePath} from './utils/file.js';\nimport {log} from './utils/log.js';\n\nimport type {Linter} from 'eslint';\n\nconst cwd: string = process.cwd();\n\nexport interface EsbuildConfig {\n  [key: string]: unknown;\n  entryPoints?: string[];\n  outdir?: string;\n  platform?: 'node' | 'browser';\n  target?: string;\n  format?: 'cjs' | 'esm';\n  minify?: boolean;\n  treeShaking?: boolean;\n  drop?: string[];\n  pure?: string[];\n  external?: string[];\n  splitting?: boolean;\n  metafile?: boolean;\n  sourcemap?: boolean | 'inline' | 'external';\n  legalComments?: 'none' | 'inline' | 'eof' | 'linked' | 'separate';\n  banner?: Record<string, string>;\n  footer?: Record<string, string>;\n  define?: Record<string, string>;\n}\n\nexport interface JestConfig {\n  [key: string]: unknown;\n  roots?: string[];\n  testEnvironment?: string;\n  transform?: Record<string, [string, Record<string, unknown>]>;\n  transformIgnorePatterns?: string[];\n  moduleNameMapper?: Record<string, string>;\n  extensionsToTreatAsEsm?: string[];\n  preset?: string;\n}\n\nexport interface WebpackConfig {\n  [key: string]: unknown;\n  entry?: string | string[];\n  output?: Record<string, unknown>;\n  module?: Record<string, unknown>;\n  plugins?: unknown[];\n  publicPath?: string;\n}\n\nexport interface AIConfig {\n  provider?: 'cursor' | 'copilot' | 'openai' | 'anthropic' | 'none';\n  apiKey?: string;\n  model?: string;\n  maxTokens?: number;\n  temperature?: number;\n}\n\nexport interface ESLintConfig {\n  [key: string]: unknown;\n  extends?: string[];\n  rules?: Linter.RulesRecord;\n}\n\nexport interface LexConfigType {\n  ai?: AIConfig;\n  configFiles?: string[];\n  copyFiles?: string[];\n  entryHTML?: string;\n  entryJs?: string;\n  env?: object;\n  esbuild?: EsbuildConfig;\n  eslint?: ESLintConfig;\n  gitUrl?: string;\n  jest?: JestConfig;\n  libraryName?: string;\n  libraryTarget?: string;\n  outputFile?: string;\n  outputFullPath?: string;\n  outputHash?: boolean;\n  outputPath?: string;\n  packageManager?: 'npm' | 'yarn';\n  preset?: 'web' | 'node' | 'lambda' | 'mobile';\n  sourceFullPath?: string;\n  sourcePath?: string;\n  targetEnvironment?: 'node' | 'web';\n  useGraphQl?: boolean;\n  useTypescript?: boolean;\n  webpack?: WebpackConfig;\n}\n\nexport type Config = LexConfigType;\n\nexport const defaultConfigValues: LexConfigType = {\n  ai: {\n    maxTokens: 4000,\n    model: 'gpt-4o',\n    provider: 'none',\n    temperature: 0.1\n  },\n  configFiles: [],\n  copyFiles: [],\n  entryHTML: 'index.html',\n  entryJs: 'index.js',\n  env: null,\n  esbuild: {\n    drop: ['console', 'debugger'],\n    legalComments: 'none',\n    metafile: false,\n    minify: true,\n    pure: ['console.log', 'console.warn', 'console.error'],\n    sourcemap: false,\n    splitting: true,\n    treeShaking: true\n  },\n  eslint: {},\n  jest: {},\n  outputFullPath: pathResolve(cwd, './dist'),\n  outputHash: false,\n  outputPath: './dist',\n  packageManager: 'npm',\n  preset: 'web',\n  sourceFullPath: pathResolve(cwd, './src'),\n  sourcePath: './src',\n  targetEnvironment: 'web',\n  useGraphQl: false,\n  useTypescript: false,\n  webpack: {\n    publicPath: './src/static'\n  }\n};\n\nexport const getTypeScriptConfigPath = (configName: string): string => {\n  const cwd = process.cwd();\n\n  if(configName === 'tsconfig.build.json') {\n    const projectBuildConfig = pathResolve(cwd, 'tsconfig.build.json');\n    if(existsSync(projectBuildConfig)) {\n      return projectBuildConfig;\n    }\n  }\n\n  if(configName === 'tsconfig.lint.json') {\n    const projectLintConfig = pathResolve(cwd, 'tsconfig.eslint.json');\n    if(existsSync(projectLintConfig)) {\n      return projectLintConfig;\n    }\n  }\n\n  if(configName === 'tsconfig.test.json') {\n    const projectTestConfig = pathResolve(cwd, 'tsconfig.test.json');\n    if(existsSync(projectTestConfig)) {\n      return projectTestConfig;\n    }\n  }\n\n  const projectConfigPath = pathResolve(cwd, configName);\n  if(existsSync(projectConfigPath)) {\n    return projectConfigPath;\n  }\n\n  const lexDir = LexConfig.getLexDir();\n  return pathResolve(lexDir, configName);\n};\n\nexport class LexConfig {\n  static config: LexConfigType = {\n    ...defaultConfigValues\n  };\n\n  static set useTypescript(value: boolean) {\n    LexConfig.config.useTypescript = value;\n    const {sourceFullPath} = LexConfig.config;\n\n    const {entryJs} = LexConfig.config;\n\n    if(entryJs === 'index.js' && value) {\n      const indexPath: string = pathResolve(cwd, sourceFullPath, 'index.tsx');\n      const hasIndexTsx: boolean = existsSync(indexPath);\n\n      if(hasIndexTsx) {\n        LexConfig.config.entryJs = 'index.tsx';\n      } else {\n        LexConfig.config.entryJs = 'index.ts';\n      }\n    }\n  }\n\n  static getLexDir(): string {\n    return dirname(getLexPackageJsonPath());\n  }\n\n  static updateConfig(updatedConfig: LexConfigType): LexConfigType {\n    const {outputFullPath, outputPath, sourcePath, sourceFullPath, useTypescript, ai} = updatedConfig;\n    const cwd: string = process.cwd();\n\n    if(useTypescript !== undefined) {\n      LexConfig.useTypescript = useTypescript;\n    }\n\n    if(outputPath !== undefined && outputFullPath === undefined) {\n      updatedConfig.outputFullPath = pathResolve(cwd, outputPath);\n    }\n\n    if(sourcePath !== undefined && sourceFullPath === undefined) {\n      updatedConfig.sourceFullPath = pathResolve(cwd, sourcePath);\n    }\n\n    if(ai) {\n      LexConfig.config.ai = {...LexConfig.config.ai, ...ai};\n\n      if(process.env.CURSOR_IDE === 'true' && LexConfig.config.ai.provider === 'none') {\n        LexConfig.config.ai.provider = 'cursor';\n      }\n    }\n\n    LexConfig.config = {...LexConfig.config, ...updatedConfig};\n\n    return LexConfig.config;\n  }\n\n  static addConfigParams(cmd, params: LexConfigType) {\n    const nameProperty: string = '_name';\n    const {environment, outputPath, sourcePath, typescript} = cmd;\n\n    if(outputPath !== undefined) {\n      params.outputPath = outputPath;\n      params.outputFullPath = pathResolve(cwd, outputPath);\n    }\n\n    if(sourcePath !== undefined) {\n      params.sourcePath = sourcePath;\n      params.sourceFullPath = pathResolve(cwd, sourcePath);\n    }\n\n    if(typescript !== undefined) {\n      params.useTypescript = true;\n    }\n\n    if(environment !== undefined) {\n      params.targetEnvironment = environment === 'web' ? 'web' : 'node';\n    }\n\n    process.env.LEX_CONFIG = JSON.stringify(\n      {\n        ...LexConfig.updateConfig(params),\n        commandName: cmd[nameProperty],\n        isStatic: cmd.static\n      }, null, 0\n    );\n  }\n\n  static async parseConfig(cmd, isRoot: boolean = true): Promise<void> {\n    const {cliName = 'Lex', lexConfig, lexConfigName, quiet, typescript, debug = false} = cmd;\n    const configFormats = ['js', 'mjs', 'cjs', 'ts', 'json'];\n    const configBaseName: string = lexConfigName || 'lex.config';\n    let configPath: string = lexConfig || '';\n    let configExists: boolean = lexConfig ? existsSync(configPath) : false;\n\n    if(!configPath || !configExists) {\n      if(debug) {\n        log(`Searching for config files with base name: ${configBaseName}`, 'info', quiet);\n      }\n\n      for(const format of configFormats) {\n        const potentialPath = isRoot\n          ? pathResolve(cwd, `./${configBaseName}.${format}`)\n          : relativeFilePath(`${configBaseName}.${format}`, cwd);\n\n        if(debug) {\n          log(`Checking for config file: ${potentialPath}`, 'info', quiet);\n        }\n\n        if(existsSync(potentialPath)) {\n          configPath = potentialPath;\n          configExists = true;\n          break;\n        }\n      }\n    }\n\n    if(configExists) {\n      log(`Using ${cliName} configuration file: ${configPath}`, 'note', quiet);\n      const ext: string = pathExtname(configPath);\n\n      if(ext === '.json') {\n        const configContent: string = readFileSync(configPath, 'utf8');\n\n        if(configContent) {\n          let configJson: LexConfigType;\n\n          try {\n            configJson = JSON.parse(configContent)?.default || {};\n          } catch (error) {\n            log(`\\n${cliName} Error: Failed to parse JSON config: ${error.message}`, 'error', quiet);\n            configJson = {};\n          }\n\n          LexConfig.addConfigParams(cmd, configJson);\n        } else {\n          log(`\\n${cliName} Error: Config file malformed, ${configPath}`, 'error', quiet);\n        }\n      } else if(['.js', '.mjs', '.cjs', '.ts'].includes(ext)) {\n        try {\n          let lexCustomConfig;\n\n          if(ext === '.cjs') {\n            const fileUrl = new URL(`file:///${pathResolve(configPath)}`).href;\n\n            if(debug) {\n              log(`Loading CommonJS config from: ${fileUrl}`, 'info', quiet);\n            }\n            lexCustomConfig = await import(fileUrl);\n          } else {\n            if(debug) {\n              log(`Loading ESM/TS config from: ${configPath}`, 'info', quiet);\n            }\n\n            lexCustomConfig = await import(configPath);\n          }\n\n          const config = lexCustomConfig.default || lexCustomConfig;\n\n          if(debug) {\n            log(`Loaded config: ${JSON.stringify(config, null, 2)}`, 'info', quiet);\n          }\n\n          if(!config) {\n            log(`\\n${cliName} Warning: Config file loaded but no configuration found`, 'warn', quiet);\n          }\n\n          LexConfig.addConfigParams(cmd, config || {});\n        } catch (error) {\n          log(`\\n${cliName} Error: Failed to load config file: ${error.message}`, 'error', quiet);\n          if(debug) {\n            // eslint-disable-next-line no-console\n            console.error(error);\n          }\n        }\n      } else {\n        log(`\\n${cliName} Error: Config file must be a JS, CJS, MJS, TS, or JSON file.`, 'error', quiet);\n      }\n    } else {\n      if(debug) {\n        log('No config file found. Using default configuration.', 'info', quiet);\n      }\n\n      LexConfig.useTypescript = !!typescript;\n      LexConfig.addConfigParams(cmd, LexConfig.config);\n    }\n  }\n\n  static checkTypescriptConfig() {\n    const tsconfigPath: string = pathResolve(cwd, './tsconfig.json');\n\n    if(!existsSync(tsconfigPath)) {\n      const dirName = getDirName();\n      writeFileSync(tsconfigPath, readFileSync(pathResolve(dirName, '../../../tsconfig.base.json')));\n    }\n  }\n\n  static checkCompileTypescriptConfig() {\n    const lexDir = LexConfig.getLexDir();\n    const tsconfigCompilePath: string = pathResolve(lexDir, './tsconfig.build.json');\n\n    if(!existsSync(tsconfigCompilePath)) {\n      const templatePath = pathResolve(lexDir, 'tsconfig.build.json');\n      if(existsSync(templatePath)) {\n        writeFileSync(tsconfigCompilePath, readFileSync(templatePath));\n      }\n    }\n  }\n\n  static checkLintTypescriptConfig() {\n    const lexDir = LexConfig.getLexDir();\n    const tsconfigLintPath: string = pathResolve(lexDir, './tsconfig.lint.json');\n\n    if(!existsSync(tsconfigLintPath)) {\n      const templatePath = pathResolve(lexDir, 'tsconfig.lint.json');\n      if(existsSync(templatePath)) {\n        writeFileSync(tsconfigLintPath, readFileSync(templatePath));\n      }\n    }\n  }\n\n  static checkTestTypescriptConfig() {\n    const lexDir = LexConfig.getLexDir();\n    const tsconfigTestPath: string = pathResolve(lexDir, './tsconfig.test.json');\n\n    if(!existsSync(tsconfigTestPath)) {\n      const templatePath = pathResolve(lexDir, 'tsconfig.test.json');\n      if(existsSync(templatePath)) {\n        writeFileSync(tsconfigTestPath, readFileSync(templatePath));\n      }\n    }\n  }\n}"],
  "mappings": "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",
  "names": ["cwd"]
}

|
|
@@ -0,0 +1,303 @@
|
|
|
1
|
+
import chalk from "chalk";
|
|
2
|
+
import { Command } from "commander";
|
|
3
|
+
import { readFileSync } from "fs";
|
|
4
|
+
import { sync as globSync } from "glob";
|
|
5
|
+
import { LexConfig } from "../../LexConfig.js";
|
|
6
|
+
import { callAIService } from "../../utils/aiService.js";
|
|
7
|
+
import { log } from "../../utils/log.js";
|
|
8
|
+
if (process.env.CURSOR_EXTENSION === "true" || process.env.CURSOR_TERMINAL === "true" || process.env.CURSOR_APP === "true" || process.env.PATH?.includes("cursor") || process.env.CURSOR_SESSION_ID) {
|
|
9
|
+
process.env.CURSOR_IDE = "true";
|
|
10
|
+
}
|
|
11
|
+
const getFileContext = (filePath) => {
|
|
12
|
+
try {
|
|
13
|
+
const content = readFileSync(filePath, "utf-8");
|
|
14
|
+
return `File: ${filePath}
|
|
15
|
+
|
|
16
|
+
${content}`;
|
|
17
|
+
} catch (_error) {
|
|
18
|
+
return `Error reading file: ${filePath}`;
|
|
19
|
+
}
|
|
20
|
+
};
|
|
21
|
+
const getProjectContext = async (options) => {
|
|
22
|
+
const { file, task, context } = options;
|
|
23
|
+
if (context === false) {
|
|
24
|
+
return "";
|
|
25
|
+
}
|
|
26
|
+
let projectContext = "";
|
|
27
|
+
if (file) {
|
|
28
|
+
projectContext += getFileContext(file);
|
|
29
|
+
}
|
|
30
|
+
switch (task) {
|
|
31
|
+
case "generate":
|
|
32
|
+
const files = globSync("src/**/*.{ts,tsx,js,jsx}", {
|
|
33
|
+
cwd: process.cwd(),
|
|
34
|
+
ignore: ["**/node_modules/**", "**/dist/**", "**/*.test.*", "**/*.spec.*"],
|
|
35
|
+
maxDepth: 3
|
|
36
|
+
});
|
|
37
|
+
projectContext += `
|
|
38
|
+
|
|
39
|
+
Project structure:
|
|
40
|
+
${files.join("\n")}`;
|
|
41
|
+
break;
|
|
42
|
+
case "test":
|
|
43
|
+
if (file) {
|
|
44
|
+
const testConfig = getFileContext("jest.config.js");
|
|
45
|
+
projectContext += `
|
|
46
|
+
|
|
47
|
+
Test configuration:
|
|
48
|
+
${testConfig}`;
|
|
49
|
+
}
|
|
50
|
+
break;
|
|
51
|
+
case "optimize":
|
|
52
|
+
const webpackConfig = getFileContext("webpack.config.js");
|
|
53
|
+
projectContext += `
|
|
54
|
+
|
|
55
|
+
Webpack configuration:
|
|
56
|
+
${webpackConfig}`;
|
|
57
|
+
break;
|
|
58
|
+
default:
|
|
59
|
+
break;
|
|
60
|
+
}
|
|
61
|
+
return projectContext;
|
|
62
|
+
};
|
|
63
|
+
const constructPrompt = (options, projectContext) => {
|
|
64
|
+
const { task = "help", prompt = "" } = options;
|
|
65
|
+
const taskInstructions = {
|
|
66
|
+
generate: "Generate code according to the following request. Make sure it follows best practices and is well documented:",
|
|
67
|
+
explain: "Explain the following code in detail, including any patterns, potential issues, and improvement suggestions:",
|
|
68
|
+
test: "Generate comprehensive unit tests for the following code:",
|
|
69
|
+
optimize: "Analyze the following code/configuration and suggest optimization improvements:",
|
|
70
|
+
help: "Provide guidance on the following development question:",
|
|
71
|
+
ask: "Provide guidance on the following development question:",
|
|
72
|
+
analyze: "Analyze the following code:"
|
|
73
|
+
};
|
|
74
|
+
const taskInstruction = taskInstructions[task] || taskInstructions.help;
|
|
75
|
+
let fullPrompt = `${taskInstruction}
|
|
76
|
+
|
|
77
|
+
${prompt}`;
|
|
78
|
+
if (projectContext) {
|
|
79
|
+
fullPrompt += `
|
|
80
|
+
|
|
81
|
+
===CONTEXT===
|
|
82
|
+
${projectContext}`;
|
|
83
|
+
}
|
|
84
|
+
return fullPrompt;
|
|
85
|
+
};
|
|
86
|
+
const displayResponse = (response, options) => {
|
|
87
|
+
const { task = "help", quiet = false } = options;
|
|
88
|
+
let content = "";
|
|
89
|
+
if (typeof response === "string") {
|
|
90
|
+
content = response;
|
|
91
|
+
} else if (response.choices?.[0]?.message?.content) {
|
|
92
|
+
content = response.choices[0].message.content;
|
|
93
|
+
} else if (response.content) {
|
|
94
|
+
content = response.content;
|
|
95
|
+
} else {
|
|
96
|
+
content = "No response received from AI model";
|
|
97
|
+
}
|
|
98
|
+
const cleanedContent = cleanResponse(content, options);
|
|
99
|
+
switch (task) {
|
|
100
|
+
case "generate":
|
|
101
|
+
log("\nGenerated Code:\n", "success", quiet);
|
|
102
|
+
log(cleanedContent, "default", quiet);
|
|
103
|
+
break;
|
|
104
|
+
case "explain":
|
|
105
|
+
log("\nCode Explanation:\n", "success", quiet);
|
|
106
|
+
log(cleanedContent, "default", quiet);
|
|
107
|
+
break;
|
|
108
|
+
case "test":
|
|
109
|
+
log("\nGenerated Tests:\n", "success", quiet);
|
|
110
|
+
log(cleanedContent, "default", quiet);
|
|
111
|
+
break;
|
|
112
|
+
case "optimize":
|
|
113
|
+
log("\nOptimization Suggestions:\n", "success", quiet);
|
|
114
|
+
log(cleanedContent, "default", quiet);
|
|
115
|
+
break;
|
|
116
|
+
default:
|
|
117
|
+
log("\nAI Response:\n", "success", quiet);
|
|
118
|
+
log(cleanedContent, "default", quiet);
|
|
119
|
+
break;
|
|
120
|
+
}
|
|
121
|
+
};
|
|
122
|
+
const cleanResponse = (content, options) => {
|
|
123
|
+
const { prompt = "", task = "help" } = options;
|
|
124
|
+
if (!content) {
|
|
125
|
+
return content;
|
|
126
|
+
}
|
|
127
|
+
let cleanedContent = content;
|
|
128
|
+
const taskInstructions = {
|
|
129
|
+
generate: "Generate code according to the following request. Make sure it follows best practices and is well documented:",
|
|
130
|
+
explain: "Explain the following code in detail, including any patterns, potential issues, and improvement suggestions:",
|
|
131
|
+
test: "Generate comprehensive unit tests for the following code:",
|
|
132
|
+
optimize: "Analyze the following code/configuration and suggest optimization improvements:",
|
|
133
|
+
help: "Provide guidance on the following development question:",
|
|
134
|
+
ask: "Provide guidance on the following development question:",
|
|
135
|
+
analyze: "Analyze the following code:"
|
|
136
|
+
};
|
|
137
|
+
const instruction = taskInstructions[task] || "";
|
|
138
|
+
if (instruction && cleanedContent.includes(instruction)) {
|
|
139
|
+
cleanedContent = cleanedContent.replace(instruction, "").trim();
|
|
140
|
+
}
|
|
141
|
+
if (prompt && cleanedContent.includes(prompt)) {
|
|
142
|
+
cleanedContent = cleanedContent.replace(prompt, "").trim();
|
|
143
|
+
}
|
|
144
|
+
if (cleanedContent.includes("===CONTEXT===")) {
|
|
145
|
+
cleanedContent = cleanedContent.split("===CONTEXT===")[0].trim();
|
|
146
|
+
}
|
|
147
|
+
if (!cleanedContent) {
|
|
148
|
+
return content;
|
|
149
|
+
}
|
|
150
|
+
return cleanedContent;
|
|
151
|
+
};
|
|
152
|
+
const getProviderAuth = (provider) => {
|
|
153
|
+
if (process.cwd().includes("reaktor")) {
|
|
154
|
+
return "cursor-auth";
|
|
155
|
+
}
|
|
156
|
+
if (process.env.AI_API_KEY) {
|
|
157
|
+
return process.env.AI_API_KEY;
|
|
158
|
+
}
|
|
159
|
+
if (provider === "none" && process.env.CURSOR_IDE === "true") {
|
|
160
|
+
return "cursor-auth";
|
|
161
|
+
}
|
|
162
|
+
switch (provider) {
|
|
163
|
+
case "openai":
|
|
164
|
+
return process.env.OPENAI_API_KEY;
|
|
165
|
+
case "anthropic":
|
|
166
|
+
return process.env.ANTHROPIC_API_KEY;
|
|
167
|
+
case "cursor":
|
|
168
|
+
return "cursor-auth";
|
|
169
|
+
case "copilot":
|
|
170
|
+
return process.env.GITHUB_TOKEN;
|
|
171
|
+
case "none":
|
|
172
|
+
return void 0;
|
|
173
|
+
default:
|
|
174
|
+
return void 0;
|
|
175
|
+
}
|
|
176
|
+
};
|
|
177
|
+
const detectCursorIDE = () => {
|
|
178
|
+
if (process.env.CURSOR_IDE === "true") {
|
|
179
|
+
return true;
|
|
180
|
+
}
|
|
181
|
+
const possibleCursorSignals = [
|
|
182
|
+
process.env.CURSOR_EXTENSION === "true",
|
|
183
|
+
process.env.CURSOR_TERMINAL === "true",
|
|
184
|
+
process.env.CURSOR_APP === "true",
|
|
185
|
+
process.env.PATH?.includes("cursor"),
|
|
186
|
+
!!process.env.CURSOR_SESSION_ID
|
|
187
|
+
];
|
|
188
|
+
const isCursorIDE = possibleCursorSignals.some((signal) => signal);
|
|
189
|
+
if (isCursorIDE) {
|
|
190
|
+
process.env.CURSOR_IDE = "true";
|
|
191
|
+
}
|
|
192
|
+
return isCursorIDE;
|
|
193
|
+
};
|
|
194
|
+
const aiFunction = async (options) => {
|
|
195
|
+
try {
|
|
196
|
+
const config = LexConfig.config || {};
|
|
197
|
+
const aiConfig = config.ai || {};
|
|
198
|
+
const provider = options.provider || aiConfig.provider || "none";
|
|
199
|
+
if (provider === "none" && !process.env.CURSOR_EXTENSION) {
|
|
200
|
+
log(`${chalk.red("Error:")} No AI provider configured.`, "error");
|
|
201
|
+
return { error: "No AI provider configured" };
|
|
202
|
+
}
|
|
203
|
+
const task = options.task || "ask";
|
|
204
|
+
const validTasks = ["explain", "generate", "test", "analyze", "ask"];
|
|
205
|
+
if (!validTasks.includes(task)) {
|
|
206
|
+
log(`${chalk.red("Error:")} Invalid task "${task}". Valid tasks are: ${validTasks.join(", ")}`, "error");
|
|
207
|
+
return { error: `Invalid task "${task}"` };
|
|
208
|
+
}
|
|
209
|
+
const { prompt } = options;
|
|
210
|
+
if (!prompt) {
|
|
211
|
+
log(`${chalk.red("Error:")} No prompt provided. Use --prompt "Your prompt here"`, "error");
|
|
212
|
+
return { error: "No prompt provided" };
|
|
213
|
+
}
|
|
214
|
+
let context = "";
|
|
215
|
+
if (options.file) {
|
|
216
|
+
try {
|
|
217
|
+
const fs = await import("fs/promises");
|
|
218
|
+
const glob = await import("glob");
|
|
219
|
+
const files = await glob.glob(options.file);
|
|
220
|
+
if (files.length === 0) {
|
|
221
|
+
log(`${chalk.yellow("Warning:")} No files found matching "${options.file}"`, "warning");
|
|
222
|
+
} else {
|
|
223
|
+
for (const file of files) {
|
|
224
|
+
const content = await fs.readFile(file, "utf8");
|
|
225
|
+
context += `
|
|
226
|
+
===FILE: ${file}===
|
|
227
|
+
${content}
|
|
228
|
+
`;
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
} catch (error) {
|
|
232
|
+
log(`${chalk.yellow("Warning:")} Error reading file: ${error.message}`, "warning");
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
if (options.dir) {
|
|
236
|
+
try {
|
|
237
|
+
const { execaSync } = await import("execa");
|
|
238
|
+
const result = execaSync("find", [options.dir, "-type", "f", "|", "sort"]);
|
|
239
|
+
context += `
|
|
240
|
+
===Project structure:===
|
|
241
|
+
${result.stdout}
|
|
242
|
+
`;
|
|
243
|
+
} catch (error) {
|
|
244
|
+
log(`${chalk.yellow("Warning:")} Error reading directory: ${error.message}`, "warning");
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
let formattedPrompt = "";
|
|
248
|
+
switch (task) {
|
|
249
|
+
case "explain":
|
|
250
|
+
formattedPrompt = `Explain the following code:
|
|
251
|
+
${prompt}`;
|
|
252
|
+
break;
|
|
253
|
+
case "generate":
|
|
254
|
+
formattedPrompt = `Generate code according to the following request:
|
|
255
|
+
${prompt}`;
|
|
256
|
+
break;
|
|
257
|
+
case "test":
|
|
258
|
+
formattedPrompt = `Generate comprehensive unit tests:
|
|
259
|
+
${prompt}`;
|
|
260
|
+
break;
|
|
261
|
+
case "analyze":
|
|
262
|
+
formattedPrompt = `Analyze the following code:
|
|
263
|
+
${prompt}`;
|
|
264
|
+
break;
|
|
265
|
+
case "ask":
|
|
266
|
+
formattedPrompt = `Provide guidance on the following development question:
|
|
267
|
+
${prompt}`;
|
|
268
|
+
break;
|
|
269
|
+
}
|
|
270
|
+
if (context) {
|
|
271
|
+
formattedPrompt += `
|
|
272
|
+
===CONTEXT===
|
|
273
|
+
${context}`;
|
|
274
|
+
}
|
|
275
|
+
if ((provider === "cursor" || process.env.CURSOR_EXTENSION) && task === "generate") {
|
|
276
|
+
log("Using Cursor IDE for code generation...", "info");
|
|
277
|
+
log("Note: For full code generation capabilities, please use Cursor IDE directly with Cmd+L or Cmd+K.", "info");
|
|
278
|
+
log("The CLI integration has limited capabilities for code generation.", "warning");
|
|
279
|
+
} else if (provider === "cursor" || process.env.CURSOR_EXTENSION) {
|
|
280
|
+
log("Using Cursor IDE for AI assistance...", "info");
|
|
281
|
+
log("Note: This is a limited integration. For full AI capabilities, use Cursor IDE directly.", "info");
|
|
282
|
+
} else {
|
|
283
|
+
log(`Using ${provider} for AI assistance...`, "info");
|
|
284
|
+
}
|
|
285
|
+
const response = await callAIService(formattedPrompt, options.quiet || false);
|
|
286
|
+
log(`
|
|
287
|
+
${response}`, "success");
|
|
288
|
+
return { response };
|
|
289
|
+
} catch (error) {
|
|
290
|
+
log(`${chalk.red("Error:")} ${error.message}`, "error");
|
|
291
|
+
return { error: error.message };
|
|
292
|
+
}
|
|
293
|
+
};
|
|
294
|
+
const ai = new Command("ai").description("Use AI to help with development tasks").option("--provider <provider>", "AI provider to use (openai, anthropic, cursor)").option("--task <task>", "Task to perform (explain, generate, test, analyze, ask)").option("--prompt <prompt>", "Prompt to send to AI").option("--file <file>", "File to analyze").option("--dir <dir>", "Directory to analyze").action(async (options) => {
|
|
295
|
+
await aiFunction(options);
|
|
296
|
+
});
|
|
297
|
+
var ai_default = ai;
|
|
298
|
+
export {
|
|
299
|
+
ai,
|
|
300
|
+
aiFunction,
|
|
301
|
+
ai_default as default
|
|
302
|
+
};
|
|
303
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/commands/ai/ai.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 chalk from 'chalk';\nimport {Command} from 'commander';\nimport {readFileSync} from 'fs';\nimport {sync as globSync} from 'glob';\n\nimport {LexConfig} from '../../LexConfig.js';\nimport {callAIService} from '../../utils/aiService.js';\nimport {log} from '../../utils/log.js';\n\nif(process.env.CURSOR_EXTENSION === 'true' ||\n  process.env.CURSOR_TERMINAL === 'true' ||\n  process.env.CURSOR_APP === 'true' ||\n  process.env.PATH?.includes('cursor') ||\n  process.env.CURSOR_SESSION_ID) {\n  process.env.CURSOR_IDE = 'true';\n}\n\nexport interface AIOptions {\n  readonly cliName?: string;\n  readonly context?: boolean;\n  readonly file?: string;\n  readonly lexConfig?: string;\n  readonly model?: string;\n  readonly prompt?: string;\n  readonly quiet?: boolean;\n  readonly task?: 'generate' | 'explain' | 'test' | 'optimize' | 'help' | 'ask' | 'analyze';\n  readonly debug?: boolean;\n  readonly provider?: string;\n  readonly dir?: string;\n}\n\nconst getFileContext = (filePath: string): string => {\n  try {\n    const content = readFileSync(filePath, 'utf-8');\n    return `File: ${filePath}\\n\\n${content}`;\n  } catch (_error) {\n    return `Error reading file: ${filePath}`;\n  }\n};\n\nconst getProjectContext = async (options: AIOptions): Promise<string> => {\n  const {file, task, context} = options;\n\n  if(context === false) {\n    return '';\n  }\n\n  let projectContext = '';\n\n  if(file) {\n    projectContext += getFileContext(file);\n  }\n\n  switch(task) {\n    case 'generate':\n      const files = globSync('src/**/*.{ts,tsx,js,jsx}', {\n        cwd: process.cwd(),\n        ignore: ['**/node_modules/**', '**/dist/**', '**/*.test.*', '**/*.spec.*'],\n        maxDepth: 3\n      });\n      projectContext += `\\n\\nProject structure:\\n${files.join('\\n')}`;\n      break;\n\n    case 'test':\n      if(file) {\n        const testConfig = getFileContext('jest.config.js');\n        projectContext += `\\n\\nTest configuration:\\n${testConfig}`;\n      }\n      break;\n\n    case 'optimize':\n      const webpackConfig = getFileContext('webpack.config.js');\n      projectContext += `\\n\\nWebpack configuration:\\n${webpackConfig}`;\n      break;\n\n    default:\n      break;\n  }\n\n  return projectContext;\n};\n\nconst constructPrompt = (options: AIOptions, projectContext: string): string => {\n  const {task = 'help', prompt = ''} = options;\n\n  const taskInstructions: Record<string, string> = {\n    generate: 'Generate code according to the following request. Make sure it follows best practices and is well documented:',\n    explain: 'Explain the following code in detail, including any patterns, potential issues, and improvement suggestions:',\n    test: 'Generate comprehensive unit tests for the following code:',\n    optimize: 'Analyze the following code/configuration and suggest optimization improvements:',\n    help: 'Provide guidance on the following development question:',\n    ask: 'Provide guidance on the following development question:',\n    analyze: 'Analyze the following code:'\n  };\n\n  const taskInstruction = taskInstructions[task] || taskInstructions.help;\n\n  let fullPrompt = `${taskInstruction}\\n\\n${prompt}`;\n\n  if(projectContext) {\n    fullPrompt += `\\n\\n===CONTEXT===\\n${projectContext}`;\n  }\n\n  return fullPrompt;\n};\n\nconst displayResponse = (response: any, options: AIOptions): void => {\n  const {task = 'help', quiet = false} = options;\n\n  let content = '';\n\n  if(typeof response === 'string') {\n    content = response;\n  } else if(response.choices?.[0]?.message?.content) {\n    content = response.choices[0].message.content;\n  } else if(response.content) {\n    content = response.content;\n  } else {\n    content = 'No response received from AI model';\n  }\n\n  const cleanedContent = cleanResponse(content, options);\n\n  switch(task) {\n    case 'generate':\n      log('\\nGenerated Code:\\n', 'success', quiet);\n      log(cleanedContent, 'default', quiet);\n      break;\n\n    case 'explain':\n      log('\\nCode Explanation:\\n', 'success', quiet);\n      log(cleanedContent, 'default', quiet);\n      break;\n\n    case 'test':\n      log('\\nGenerated Tests:\\n', 'success', quiet);\n      log(cleanedContent, 'default', quiet);\n      break;\n\n    case 'optimize':\n      log('\\nOptimization Suggestions:\\n', 'success', quiet);\n      log(cleanedContent, 'default', quiet);\n      break;\n\n    default:\n      log('\\nAI Response:\\n', 'success', quiet);\n      log(cleanedContent, 'default', quiet);\n      break;\n  }\n};\n\nconst cleanResponse = (content: string, options: AIOptions): string => {\n  const {prompt = '', task = 'help'} = options;\n\n  if(!content) {\n    return content;\n  }\n\n  let cleanedContent = content;\n\n  const taskInstructions: Record<string, string> = {\n    generate: 'Generate code according to the following request. Make sure it follows best practices and is well documented:',\n    explain: 'Explain the following code in detail, including any patterns, potential issues, and improvement suggestions:',\n    test: 'Generate comprehensive unit tests for the following code:',\n    optimize: 'Analyze the following code/configuration and suggest optimization improvements:',\n    help: 'Provide guidance on the following development question:',\n    ask: 'Provide guidance on the following development question:',\n    analyze: 'Analyze the following code:'\n  };\n\n  const instruction = taskInstructions[task] || '';\n\n  if(instruction && cleanedContent.includes(instruction)) {\n    cleanedContent = cleanedContent.replace(instruction, '').trim();\n  }\n\n  if(prompt && cleanedContent.includes(prompt)) {\n    cleanedContent = cleanedContent.replace(prompt, '').trim();\n  }\n\n  if(cleanedContent.includes('===CONTEXT===')) {\n    cleanedContent = cleanedContent.split('===CONTEXT===')[0].trim();\n  }\n\n  if(!cleanedContent) {\n    return content;\n  }\n\n  return cleanedContent;\n};\n\nconst getProviderAuth = (provider: string): string | undefined => {\n  if(process.cwd().includes('reaktor')) {\n    return 'cursor-auth';\n  }\n\n  if(process.env.AI_API_KEY) {\n    return process.env.AI_API_KEY;\n  }\n\n  if(provider === 'none' && process.env.CURSOR_IDE === 'true') {\n    return 'cursor-auth';\n  }\n\n  switch(provider) {\n    case 'openai':\n      return process.env.OPENAI_API_KEY;\n    case 'anthropic':\n      return process.env.ANTHROPIC_API_KEY;\n    case 'cursor':\n      return 'cursor-auth';\n    case 'copilot':\n      return process.env.GITHUB_TOKEN;\n    case 'none':\n      return undefined;\n    default:\n      return undefined;\n  }\n};\n\nconst detectCursorIDE = (): boolean => {\n  if(process.env.CURSOR_IDE === 'true') {\n    return true;\n  }\n\n  const possibleCursorSignals = [\n    process.env.CURSOR_EXTENSION === 'true',\n    process.env.CURSOR_TERMINAL === 'true',\n    process.env.CURSOR_APP === 'true',\n    process.env.PATH?.includes('cursor'),\n    !!process.env.CURSOR_SESSION_ID\n  ];\n\n  const isCursorIDE = possibleCursorSignals.some((signal) => signal);\n\n  if(isCursorIDE) {\n    process.env.CURSOR_IDE = 'true';\n  }\n\n  return isCursorIDE;\n};\n\nexport const aiFunction = async (options: AIOptions): Promise<any> => {\n  try {\n    const config = LexConfig.config || {};\n    const aiConfig = config.ai || {};\n    const provider = options.provider || aiConfig.provider || 'none';\n\n    if(provider === 'none' && !process.env.CURSOR_EXTENSION) {\n      log(`${chalk.red('Error:')} No AI provider configured.`, 'error');\n      return {error: 'No AI provider configured'};\n    }\n\n    const task = options.task || 'ask';\n    const validTasks = ['explain', 'generate', 'test', 'analyze', 'ask'];\n\n    if(!validTasks.includes(task)) {\n      log(`${chalk.red('Error:')} Invalid task \"${task}\". Valid tasks are: ${validTasks.join(', ')}`, 'error');\n      return {error: `Invalid task \"${task}\"`};\n    }\n\n    const {prompt} = options;\n\n    if(!prompt) {\n      log(`${chalk.red('Error:')} No prompt provided. Use --prompt \"Your prompt here\"`, 'error');\n      return {error: 'No prompt provided'};\n    }\n\n    let context = '';\n\n    if(options.file) {\n      try {\n        const fs = await import('fs/promises');\n        const glob = await import('glob');\n        const files = await glob.glob(options.file);\n\n        if(files.length === 0) {\n          log(`${chalk.yellow('Warning:')} No files found matching \"${options.file}\"`, 'warning');\n        } else {\n          for(const file of files) {\n            const content = await fs.readFile(file, 'utf8');\n            context += `\\n===FILE: ${file}===\\n${content}\\n`;\n          }\n        }\n      } catch (error) {\n        log(`${chalk.yellow('Warning:')} Error reading file: ${error.message}`, 'warning');\n      }\n    }\n\n    if(options.dir) {\n      try {\n        const {execaSync} = await import('execa');\n        const result = execaSync('find', [options.dir, '-type', 'f', '|', 'sort']);\n        context += `\\n===Project structure:===\\n${result.stdout}\\n`;\n      } catch (error) {\n        log(`${chalk.yellow('Warning:')} Error reading directory: ${error.message}`, 'warning');\n      }\n    }\n\n    let formattedPrompt = '';\n\n    switch(task) {\n      case 'explain':\n        formattedPrompt = `Explain the following code:\\n${prompt}`;\n        break;\n      case 'generate':\n        formattedPrompt = `Generate code according to the following request:\\n${prompt}`;\n        break;\n      case 'test':\n        formattedPrompt = `Generate comprehensive unit tests:\\n${prompt}`;\n        break;\n      case 'analyze':\n        formattedPrompt = `Analyze the following code:\\n${prompt}`;\n        break;\n      case 'ask':\n        formattedPrompt = `Provide guidance on the following development question:\\n${prompt}`;\n        break;\n    }\n\n    if(context) {\n      formattedPrompt += `\\n===CONTEXT===\\n${context}`;\n    }\n\n    if((provider === 'cursor' || process.env.CURSOR_EXTENSION) && task === 'generate') {\n      log('Using Cursor IDE for code generation...', 'info');\n      log('Note: For full code generation capabilities, please use Cursor IDE directly with Cmd+L or Cmd+K.', 'info');\n      log('The CLI integration has limited capabilities for code generation.', 'warning');\n    } else if(provider === 'cursor' || process.env.CURSOR_EXTENSION) {\n      log('Using Cursor IDE for AI assistance...', 'info');\n      log('Note: This is a limited integration. For full AI capabilities, use Cursor IDE directly.', 'info');\n    } else {\n      log(`Using ${provider} for AI assistance...`, 'info');\n    }\n\n    const response = await callAIService(formattedPrompt, options.quiet || false);\n\n    log(`\\n${response}`, 'success');\n\n    return {response};\n  } catch (error) {\n    log(`${chalk.red('Error:')} ${error.message}`, 'error');\n    return {error: error.message};\n  }\n};\n\nexport const ai = new Command('ai')\n  .description('Use AI to help with development tasks')\n  .option('--provider <provider>', 'AI provider to use (openai, anthropic, cursor)')\n  .option('--task <task>', 'Task to perform (explain, generate, test, analyze, ask)')\n  .option('--prompt <prompt>', 'Prompt to send to AI')\n  .option('--file <file>', 'File to analyze')\n  .option('--dir <dir>', 'Directory to analyze')\n  .action(async (options: AIOptions) => {\n    await aiFunction(options);\n  });\n\nexport default ai;"],
  "mappings": "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",
  "names": []
}

|