@nlabs/lex 1.49.4 → 1.50.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/.swcrc +35 -0
- package/README.md +43 -59
- package/__mocks__/chalk.js +19 -17
- package/config.json +32 -8
- package/examples/lex.config.js +110 -10
- package/index.cjs +1 -5
- package/lex.config.js +34 -7
- package/lib/Button.stories.js +99 -0
- package/lib/LexConfig.d.ts +60 -22
- package/lib/LexConfig.js +285 -244
- package/lib/commands/ai/ai.js +287 -288
- package/lib/commands/ai/index.js +8 -7
- package/lib/commands/build/build.d.ts +2 -2
- package/lib/commands/build/build.js +349 -458
- package/lib/commands/clean/clean.js +45 -33
- package/lib/commands/compile/compile.js +214 -227
- package/lib/commands/config/config.js +46 -42
- package/lib/commands/copy/copy.js +36 -35
- package/lib/commands/create/create.js +200 -121
- package/lib/commands/dev/dev.d.ts +2 -0
- package/lib/commands/dev/dev.js +259 -263
- package/lib/commands/init/init.js +108 -88
- package/lib/commands/link/link.js +18 -14
- package/lib/commands/lint/lint.js +735 -742
- package/lib/commands/migrate/migrate.js +49 -36
- package/lib/commands/publish/publish.js +116 -96
- package/lib/commands/serverless/serverless.js +611 -585
- package/lib/commands/storybook/storybook.js +242 -238
- package/lib/commands/test/test.d.ts +1 -1
- package/lib/commands/test/test.js +382 -394
- package/lib/commands/update/update.js +141 -120
- package/lib/commands/upgrade/upgrade.js +51 -44
- package/lib/commands/versions/versions.d.ts +1 -1
- package/lib/commands/versions/versions.js +36 -38
- package/lib/create/changelog.js +136 -125
- package/lib/index.js +40 -38
- package/lib/lex.js +95 -68
- package/lib/storybook/index.js +6 -1
- package/lib/test-react/index.js +7 -84
- package/lib/types.d.ts +1 -1
- package/lib/types.js +7 -1
- package/lib/utils/aiService.js +240 -227
- package/lib/utils/app.js +274 -273
- package/lib/utils/deepMerge.js +37 -23
- package/lib/utils/file.js +218 -215
- package/lib/utils/log.js +29 -27
- package/lib/utils/reactShim.js +7 -85
- package/lib/utils/translations.js +91 -65
- package/package.json +63 -64
- package/templates/typescript/DataLayer.js.txt +218 -0
- package/templates/typescript/DataLayer.test.js.txt +268 -0
- package/templates/typescript/DataLayer.test.ts.txt +269 -0
- package/templates/typescript/DataLayer.ts.txt +227 -0
- package/webpack.config.js +53 -26
- package/lib/commands/lint/autofix.d.ts +0 -2
package/lib/commands/dev/dev.js
CHANGED
|
@@ -1,286 +1,282 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
import
|
|
6
|
-
import {
|
|
7
|
-
import {
|
|
8
|
-
import
|
|
9
|
-
import {
|
|
10
|
-
import {
|
|
11
|
-
import {
|
|
12
|
-
import {
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) 2018-Present, Nitrogen Labs, Inc.
|
|
3
|
+
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
|
|
4
|
+
*/ import boxen from 'boxen';
|
|
5
|
+
import chalk from 'chalk';
|
|
6
|
+
import { execa } from 'execa';
|
|
7
|
+
import { existsSync, readFileSync, writeFileSync, mkdirSync } from 'fs';
|
|
8
|
+
import https from 'https';
|
|
9
|
+
import { networkInterfaces, homedir } from 'os';
|
|
10
|
+
import { dirname, resolve as pathResolve, join } from 'path';
|
|
11
|
+
import { LexConfig } from '../../LexConfig.js';
|
|
12
|
+
import { createSpinner, handleWebpackProgress, removeFiles } from '../../utils/app.js';
|
|
13
|
+
import { resolveWebpackPaths } from '../../utils/file.js';
|
|
14
|
+
import { log } from '../../utils/log.js';
|
|
15
|
+
import { processTranslations } from '../../utils/translations.js';
|
|
13
16
|
let currentFilename;
|
|
14
17
|
let currentDirname;
|
|
15
18
|
try {
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
19
|
+
// eslint-disable-next-line no-eval
|
|
20
|
+
currentFilename = eval('require("url").fileURLToPath(import.meta.url)');
|
|
21
|
+
currentDirname = dirname(currentFilename);
|
|
22
|
+
} catch {
|
|
23
|
+
currentFilename = process.cwd();
|
|
24
|
+
currentDirname = process.cwd();
|
|
21
25
|
}
|
|
22
|
-
const getCacheDir = ()
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
26
|
+
const getCacheDir = ()=>{
|
|
27
|
+
const cacheDir = join(homedir(), '.lex-cache');
|
|
28
|
+
if (!existsSync(cacheDir)) {
|
|
29
|
+
mkdirSync(cacheDir, {
|
|
30
|
+
recursive: true
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
return cacheDir;
|
|
28
34
|
};
|
|
29
|
-
const getCachePath = ()
|
|
30
|
-
const readPublicIpCache = ()
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
35
|
+
const getCachePath = ()=>join(getCacheDir(), 'public-ip.json');
|
|
36
|
+
const readPublicIpCache = ()=>{
|
|
37
|
+
const cachePath = getCachePath();
|
|
38
|
+
if (!existsSync(cachePath)) {
|
|
39
|
+
return null;
|
|
40
|
+
}
|
|
41
|
+
try {
|
|
42
|
+
const cacheData = readFileSync(cachePath, 'utf8');
|
|
43
|
+
const cache = JSON.parse(cacheData);
|
|
44
|
+
const oneWeekMs = 7 * 24 * 60 * 60 * 1000;
|
|
45
|
+
if (Date.now() - cache.timestamp > oneWeekMs) {
|
|
46
|
+
return null;
|
|
47
|
+
}
|
|
48
|
+
return cache;
|
|
49
|
+
} catch {
|
|
50
|
+
return null;
|
|
41
51
|
}
|
|
42
|
-
return cache;
|
|
43
|
-
} catch {
|
|
44
|
-
return null;
|
|
45
|
-
}
|
|
46
52
|
};
|
|
47
|
-
const writePublicIpCache = (ip)
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
53
|
+
const writePublicIpCache = (ip)=>{
|
|
54
|
+
const cachePath = getCachePath();
|
|
55
|
+
const cache = {
|
|
56
|
+
ip,
|
|
57
|
+
timestamp: Date.now()
|
|
58
|
+
};
|
|
59
|
+
writeFileSync(cachePath, JSON.stringify(cache, null, 2));
|
|
54
60
|
};
|
|
55
|
-
const fetchPublicIp = (forceRefresh = false)
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
61
|
+
const fetchPublicIp = (forceRefresh = false)=>new Promise((resolve)=>{
|
|
62
|
+
if (!forceRefresh) {
|
|
63
|
+
const cached = readPublicIpCache();
|
|
64
|
+
if (cached) {
|
|
65
|
+
resolve(cached.ip);
|
|
66
|
+
return;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
https.get('https://api.ipify.org', (res)=>{
|
|
70
|
+
let data = '';
|
|
71
|
+
res.on('data', (chunk)=>data += chunk);
|
|
72
|
+
res.on('end', ()=>{
|
|
73
|
+
const ip = data.trim();
|
|
74
|
+
if (ip) {
|
|
75
|
+
writePublicIpCache(ip);
|
|
76
|
+
}
|
|
77
|
+
resolve(ip);
|
|
78
|
+
});
|
|
79
|
+
}).on('error', ()=>resolve(undefined));
|
|
72
80
|
});
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
const
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
81
|
+
const getNetworkAddresses = ()=>{
|
|
82
|
+
const interfaces = networkInterfaces();
|
|
83
|
+
const addresses = {
|
|
84
|
+
local: 'localhost',
|
|
85
|
+
private: null,
|
|
86
|
+
public: null
|
|
87
|
+
};
|
|
88
|
+
for (const name of Object.keys(interfaces)){
|
|
89
|
+
const networkInterface = interfaces[name];
|
|
90
|
+
if (!networkInterface) {
|
|
91
|
+
continue;
|
|
92
|
+
}
|
|
93
|
+
for (const iface of networkInterface){
|
|
94
|
+
if (iface.family === 'IPv4' && !iface.internal) {
|
|
95
|
+
const ip = iface.address;
|
|
96
|
+
if (ip.startsWith('10.') || ip.startsWith('192.168.') || ip.startsWith('172.')) {
|
|
97
|
+
if (!addresses.private) {
|
|
98
|
+
addresses.private = ip;
|
|
99
|
+
}
|
|
100
|
+
} else {
|
|
101
|
+
if (!addresses.public) {
|
|
102
|
+
addresses.public = ip;
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
98
106
|
}
|
|
99
|
-
}
|
|
100
107
|
}
|
|
101
|
-
|
|
102
|
-
return addresses;
|
|
108
|
+
return addresses;
|
|
103
109
|
};
|
|
104
|
-
const displayServerStatus = (port =
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
}
|
|
108
|
-
const addresses = getNetworkAddresses();
|
|
109
|
-
const localUrl = `http://localhost:${port}`;
|
|
110
|
-
const privateUrl = addresses.private ? `http://${addresses.private}:${port}` : null;
|
|
111
|
-
let publicUrl = null;
|
|
112
|
-
if (publicIp) {
|
|
113
|
-
publicUrl = `http://${publicIp}:${port}`;
|
|
114
|
-
} else if (addresses.public) {
|
|
115
|
-
publicUrl = `http://${addresses.public}:${port}`;
|
|
116
|
-
}
|
|
117
|
-
let urlLines = `${chalk.green("Local:")} ${chalk.underline(localUrl)}
|
|
118
|
-
`;
|
|
119
|
-
if (privateUrl) {
|
|
120
|
-
urlLines += `${chalk.green("Private:")} ${chalk.underline(privateUrl)}
|
|
121
|
-
`;
|
|
122
|
-
}
|
|
123
|
-
if (publicUrl) {
|
|
124
|
-
urlLines += `${chalk.green("Public:")} ${chalk.underline(publicUrl)}
|
|
125
|
-
`;
|
|
126
|
-
}
|
|
127
|
-
const statusBox = boxen(
|
|
128
|
-
`${chalk.cyan.bold("\u{1F680} Development Server Running")}
|
|
129
|
-
|
|
130
|
-
${urlLines}
|
|
131
|
-
${chalk.yellow("Press Ctrl+C to stop the server")}`,
|
|
132
|
-
{
|
|
133
|
-
backgroundColor: "#1a1a1a",
|
|
134
|
-
borderColor: "cyan",
|
|
135
|
-
borderStyle: "round",
|
|
136
|
-
margin: 1,
|
|
137
|
-
padding: 1
|
|
110
|
+
const displayServerStatus = (port = 3000, quiet, publicIp)=>{
|
|
111
|
+
if (quiet) {
|
|
112
|
+
return;
|
|
138
113
|
}
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
log(`${cliName} start development server...`, "info", quiet);
|
|
148
|
-
await LexConfig.parseConfig(cmd);
|
|
149
|
-
const { outputFullPath, useTypescript } = LexConfig.config;
|
|
150
|
-
let variablesObj = { NODE_ENV: "development" };
|
|
151
|
-
if (variables) {
|
|
152
|
-
try {
|
|
153
|
-
variablesObj = JSON.parse(variables);
|
|
154
|
-
} catch (_error) {
|
|
155
|
-
log(`
|
|
156
|
-
${cliName} Error: Environment variables option is not a valid JSON object.`, "error", quiet);
|
|
157
|
-
callback(1);
|
|
158
|
-
return 1;
|
|
114
|
+
const addresses = getNetworkAddresses();
|
|
115
|
+
const localUrl = `http://localhost:${port}`;
|
|
116
|
+
const privateUrl = addresses.private ? `http://${addresses.private}:${port}` : null;
|
|
117
|
+
let publicUrl = null;
|
|
118
|
+
if (publicIp) {
|
|
119
|
+
publicUrl = `http://${publicIp}:${port}`;
|
|
120
|
+
} else if (addresses.public) {
|
|
121
|
+
publicUrl = `http://${addresses.public}:${port}`;
|
|
159
122
|
}
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
spinner.start("Cleaning output directory...");
|
|
167
|
-
await removeFiles(outputFullPath || "");
|
|
168
|
-
spinner.succeed("Successfully cleaned output directory!");
|
|
169
|
-
}
|
|
170
|
-
if (translations) {
|
|
171
|
-
spinner.start("Processing translations...");
|
|
172
|
-
try {
|
|
173
|
-
const sourcePath = LexConfig.config.sourceFullPath || process.cwd();
|
|
174
|
-
const outputPath = LexConfig.config.outputFullPath || "lib";
|
|
175
|
-
await processTranslations(sourcePath, outputPath, quiet);
|
|
176
|
-
spinner.succeed("Translations processed successfully!");
|
|
177
|
-
} catch (translationError) {
|
|
178
|
-
log(`
|
|
179
|
-
${cliName} Error: Failed to process translations: ${translationError.message}`, "error", quiet);
|
|
180
|
-
spinner.fail("Failed to process translations.");
|
|
181
|
-
callback(1);
|
|
182
|
-
return 1;
|
|
123
|
+
let urlLines = `${chalk.green('Local:')} ${chalk.underline(localUrl)}\n`;
|
|
124
|
+
if (privateUrl) {
|
|
125
|
+
urlLines += `${chalk.green('Private:')} ${chalk.underline(privateUrl)}\n`;
|
|
126
|
+
}
|
|
127
|
+
if (publicUrl) {
|
|
128
|
+
urlLines += `${chalk.green('Public:')} ${chalk.underline(publicUrl)}\n`;
|
|
183
129
|
}
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
const { webpackConfig: resolvedConfig } = resolveWebpackPaths(currentDirname);
|
|
191
|
-
webpackConfig = resolvedConfig;
|
|
192
|
-
}
|
|
193
|
-
const { webpackPath } = resolveWebpackPaths(currentDirname);
|
|
194
|
-
const webpackOptions = [
|
|
195
|
-
"--color",
|
|
196
|
-
"--watch",
|
|
197
|
-
"--config",
|
|
198
|
-
webpackConfig
|
|
199
|
-
];
|
|
200
|
-
if (bundleAnalyzer) {
|
|
201
|
-
webpackOptions.push("--bundleAnalyzer");
|
|
202
|
-
}
|
|
203
|
-
try {
|
|
204
|
-
const finalWebpackOptions = webpackPath === "npx" ? ["webpack", ...webpackOptions] : webpackOptions;
|
|
205
|
-
spinner.start("Starting development server...");
|
|
206
|
-
const childProcess = execa(webpackPath, finalWebpackOptions, {
|
|
207
|
-
encoding: "utf8",
|
|
208
|
-
env: {
|
|
209
|
-
LEX_QUIET: quiet,
|
|
210
|
-
WEBPACK_DEV_OPEN: open
|
|
211
|
-
},
|
|
212
|
-
stdio: "pipe"
|
|
130
|
+
const statusBox = boxen(`${chalk.cyan.bold('🚀 Development Server Running')}\n\n${urlLines}\n` + `${chalk.yellow('Press Ctrl+C to stop the server')}`, {
|
|
131
|
+
backgroundColor: '#1a1a1a',
|
|
132
|
+
borderColor: 'cyan',
|
|
133
|
+
borderStyle: 'round',
|
|
134
|
+
margin: 1,
|
|
135
|
+
padding: 1
|
|
213
136
|
});
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
137
|
+
// eslint-disable-next-line no-console
|
|
138
|
+
console.log(`\n${statusBox}\n`);
|
|
139
|
+
};
|
|
140
|
+
export const dev = async (cmd, callback = ()=>({}))=>{
|
|
141
|
+
const { bundleAnalyzer, cliName = 'Lex', config, format = 'esm', open = false, port = 3000, quiet, remove, translations = false, usePublicIp, variables } = cmd;
|
|
142
|
+
const spinner = createSpinner(quiet);
|
|
143
|
+
log(`${cliName} start development server...`, 'info', quiet);
|
|
144
|
+
await LexConfig.parseConfig(cmd);
|
|
145
|
+
const { outputFullPath, useTypescript } = LexConfig.config;
|
|
146
|
+
let variablesObj = {
|
|
147
|
+
NODE_ENV: 'development'
|
|
148
|
+
};
|
|
149
|
+
if (variables) {
|
|
150
|
+
try {
|
|
151
|
+
variablesObj = JSON.parse(variables);
|
|
152
|
+
} catch (_error) {
|
|
153
|
+
log(`\n${cliName} Error: Environment variables option is not a valid JSON object.`, 'error', quiet);
|
|
154
|
+
callback(1);
|
|
155
|
+
return 1;
|
|
225
156
|
}
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
157
|
+
}
|
|
158
|
+
process.env = {
|
|
159
|
+
...process.env,
|
|
160
|
+
...variablesObj
|
|
161
|
+
};
|
|
162
|
+
if (useTypescript) {
|
|
163
|
+
LexConfig.checkTypescriptConfig();
|
|
164
|
+
}
|
|
165
|
+
if (remove) {
|
|
166
|
+
spinner.start('Cleaning output directory...');
|
|
167
|
+
await removeFiles(outputFullPath || '');
|
|
168
|
+
spinner.succeed('Successfully cleaned output directory!');
|
|
169
|
+
}
|
|
170
|
+
if (translations) {
|
|
171
|
+
spinner.start('Processing translations...');
|
|
172
|
+
try {
|
|
173
|
+
const sourcePath = LexConfig.config.sourceFullPath || process.cwd();
|
|
174
|
+
const outputPath = LexConfig.config.outputFullPath || 'lib';
|
|
175
|
+
await processTranslations(sourcePath, outputPath, quiet);
|
|
176
|
+
spinner.succeed('Translations processed successfully!');
|
|
177
|
+
} catch (translationError) {
|
|
178
|
+
log(`\n${cliName} Error: Failed to process translations: ${translationError.message}`, 'error', quiet);
|
|
179
|
+
spinner.fail('Failed to process translations.');
|
|
180
|
+
callback(1);
|
|
181
|
+
return 1;
|
|
243
182
|
}
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
183
|
+
}
|
|
184
|
+
let webpackConfig;
|
|
185
|
+
if (config) {
|
|
186
|
+
const isRelativeConfig = config.substr(0, 2) === './';
|
|
187
|
+
webpackConfig = isRelativeConfig ? pathResolve(process.cwd(), config) : config;
|
|
188
|
+
} else {
|
|
189
|
+
const { webpackConfig: resolvedConfig } = resolveWebpackPaths(currentDirname);
|
|
190
|
+
webpackConfig = resolvedConfig;
|
|
191
|
+
}
|
|
192
|
+
const { webpackPath } = resolveWebpackPaths(currentDirname);
|
|
193
|
+
const webpackOptions = [
|
|
194
|
+
'--color',
|
|
195
|
+
'--watch',
|
|
196
|
+
'--config',
|
|
197
|
+
webpackConfig
|
|
198
|
+
];
|
|
199
|
+
if (bundleAnalyzer) {
|
|
200
|
+
webpackOptions.push('--bundleAnalyzer');
|
|
201
|
+
}
|
|
202
|
+
if (port !== 3000) {
|
|
203
|
+
webpackOptions.push('--port', port.toString());
|
|
204
|
+
}
|
|
205
|
+
try {
|
|
206
|
+
const finalWebpackOptions = webpackPath === 'npx' ? [
|
|
207
|
+
'webpack',
|
|
208
|
+
...webpackOptions
|
|
209
|
+
] : webpackOptions;
|
|
210
|
+
spinner.start('Starting development server...');
|
|
211
|
+
const childProcess = execa(webpackPath, finalWebpackOptions, {
|
|
212
|
+
encoding: 'utf8',
|
|
213
|
+
env: {
|
|
214
|
+
LEX_QUIET: quiet,
|
|
215
|
+
WEBPACK_DEV_OPEN: open
|
|
216
|
+
},
|
|
217
|
+
stdio: 'pipe'
|
|
249
218
|
});
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
219
|
+
let serverStarted = false;
|
|
220
|
+
let statusShown = false;
|
|
221
|
+
const showStatusOnce = (portToShow)=>{
|
|
222
|
+
if (statusShown) {
|
|
223
|
+
return;
|
|
224
|
+
}
|
|
225
|
+
statusShown = true;
|
|
226
|
+
if (usePublicIp) {
|
|
227
|
+
fetchPublicIp(usePublicIp).then((publicIp)=>{
|
|
228
|
+
displayServerStatus(portToShow, quiet, publicIp);
|
|
229
|
+
});
|
|
230
|
+
} else {
|
|
231
|
+
displayServerStatus(portToShow, quiet);
|
|
232
|
+
}
|
|
233
|
+
};
|
|
234
|
+
let detectedPort = 3000;
|
|
235
|
+
childProcess.stdout?.on('data', (data)=>{
|
|
236
|
+
const output = data.toString();
|
|
237
|
+
handleWebpackProgress(output, spinner, quiet, '🚀', 'Webpack Building');
|
|
238
|
+
if (!serverStarted && (output.includes('Local:') || output.includes('webpack compiled') || output.includes('webpack-plugin-serve') || output.includes('http://localhost') || output.includes('listening on port'))) {
|
|
239
|
+
serverStarted = true;
|
|
240
|
+
spinner.succeed('Development server started.');
|
|
241
|
+
const portMatch = output.match(/Local:\s*http:\/\/[^:]+:(\d+)/) || output.match(/http:\/\/localhost:(\d+)/) || output.match(/port:\s*(\d+)/) || output.match(/listening on port (\d+)/) || output.match(/WebpackPluginServe listening on port (\d+)/);
|
|
242
|
+
if (portMatch) {
|
|
243
|
+
detectedPort = parseInt(portMatch[1]);
|
|
244
|
+
}
|
|
245
|
+
showStatusOnce(detectedPort);
|
|
246
|
+
}
|
|
260
247
|
});
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
248
|
+
childProcess.stderr?.on('data', (data)=>{
|
|
249
|
+
const output = data.toString();
|
|
250
|
+
handleWebpackProgress(output, spinner, quiet, '🚀', 'Webpack Building');
|
|
251
|
+
if (!serverStarted && (output.includes('Local:') || output.includes('webpack compiled') || output.includes('webpack-plugin-serve') || output.includes('http://localhost') || output.includes('listening on port'))) {
|
|
252
|
+
serverStarted = true;
|
|
253
|
+
spinner.succeed('Development server started.');
|
|
254
|
+
const portMatch = output.match(/Local:\s*http:\/\/[^:]+:(\d+)/) || output.match(/http:\/\/localhost:(\d+)/) || output.match(/port:\s*(\d+)/) || output.match(/listening on port (\d+)/) || output.match(/WebpackPluginServe listening on port (\d+)/);
|
|
255
|
+
if (portMatch) {
|
|
256
|
+
detectedPort = parseInt(portMatch[1]);
|
|
257
|
+
}
|
|
258
|
+
showStatusOnce(detectedPort);
|
|
259
|
+
}
|
|
260
|
+
});
|
|
261
|
+
setTimeout(()=>{
|
|
262
|
+
if (!serverStarted) {
|
|
263
|
+
spinner.succeed('Development server started.');
|
|
264
|
+
showStatusOnce(detectedPort);
|
|
265
|
+
}
|
|
266
|
+
}, 5000);
|
|
267
|
+
await childProcess;
|
|
268
|
+
if (!serverStarted) {
|
|
269
|
+
spinner.succeed('Development server started.');
|
|
270
|
+
showStatusOnce(detectedPort);
|
|
270
271
|
}
|
|
271
|
-
|
|
272
|
+
callback(0);
|
|
273
|
+
return 0;
|
|
274
|
+
} catch (error) {
|
|
275
|
+
log(`\n${cliName} Error: ${error.message}`, 'error', quiet);
|
|
276
|
+
spinner.fail('There was an error while running Webpack.');
|
|
277
|
+
callback(1);
|
|
278
|
+
return 1;
|
|
272
279
|
}
|
|
273
|
-
callback(0);
|
|
274
|
-
return 0;
|
|
275
|
-
} catch (error) {
|
|
276
|
-
log(`
|
|
277
|
-
${cliName} Error: ${error.message}`, "error", quiet);
|
|
278
|
-
spinner.fail("There was an error while running Webpack.");
|
|
279
|
-
callback(1);
|
|
280
|
-
return 1;
|
|
281
|
-
}
|
|
282
|
-
};
|
|
283
|
-
export {
|
|
284
|
-
dev
|
|
285
280
|
};
|
|
286
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/commands/dev/dev.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 boxen from 'boxen';\nimport chalk from 'chalk';\nimport {execa} from 'execa';\nimport {existsSync, readFileSync, writeFileSync, mkdirSync} from 'fs';\nimport https from 'https';\nimport {networkInterfaces, homedir} from 'os';\nimport {dirname, resolve as pathResolve, join} from 'path';\n\nimport {LexConfig} from '../../LexConfig.js';\nimport {createSpinner, handleWebpackProgress, removeFiles} from '../../utils/app.js';\nimport {resolveWebpackPaths} from '../../utils/file.js';\nimport {log} from '../../utils/log.js';\nimport {processTranslations} from '../../utils/translations.js';\n\nlet currentFilename: string;\nlet currentDirname: string;\n\ntry {\n  // eslint-disable-next-line no-eval\n  currentFilename = eval('require(\"url\").fileURLToPath(import.meta.url)');\n  currentDirname = dirname(currentFilename);\n} catch {\n  currentFilename = process.cwd();\n  currentDirname = process.cwd();\n}\n\nexport interface DevOptions {\n  readonly bundleAnalyzer?: boolean;\n  readonly cliName?: string;\n  readonly config?: string;\n  readonly open?: boolean;\n  readonly quiet?: boolean;\n  readonly remove?: boolean;\n  readonly translations?: boolean;\n  readonly usePublicIp?: boolean;\n  readonly variables?: string;\n}\n\nexport type DevCallback = (status: number) => void;\n\ninterface PublicIpCache {\n  ip: string;\n  timestamp: number;\n}\n\nconst getCacheDir = (): string => {\n  const cacheDir = join(homedir(), '.lex-cache');\n  if(!existsSync(cacheDir)) {\n    mkdirSync(cacheDir, {recursive: true});\n  }\n  return cacheDir;\n};\n\nconst getCachePath = (): string => join(getCacheDir(), 'public-ip.json');\n\nconst readPublicIpCache = (): PublicIpCache | null => {\n  const cachePath = getCachePath();\n  if(!existsSync(cachePath)) {\n    return null;\n  }\n\n  try {\n    const cacheData = readFileSync(cachePath, 'utf8');\n    const cache: PublicIpCache = JSON.parse(cacheData);\n\n    // Check if cache is older than 1 week (7 days * 24 hours * 60 minutes * 60 seconds * 1000 milliseconds)\n    const oneWeekMs = 7 * 24 * 60 * 60 * 1000;\n    if(Date.now() - cache.timestamp > oneWeekMs) {\n      return null;\n    }\n\n    return cache;\n  } catch {\n    return null;\n  }\n};\n\nconst writePublicIpCache = (ip: string): void => {\n  const cachePath = getCachePath();\n  const cache: PublicIpCache = {\n    ip,\n    timestamp: Date.now()\n  };\n  writeFileSync(cachePath, JSON.stringify(cache, null, 2));\n};\n\nconst fetchPublicIp = (forceRefresh: boolean = false): Promise<string | undefined> => new Promise((resolve) => {\n  // Check cache first unless force refresh is requested\n  if(!forceRefresh) {\n    const cached = readPublicIpCache();\n    if(cached) {\n      resolve(cached.ip);\n      return;\n    }\n  }\n\n  https.get('https://api.ipify.org', (res) => {\n    let data = '';\n    res.on('data', (chunk) => (data += chunk));\n    res.on('end', () => {\n      const ip = data.trim();\n      if(ip) {\n        writePublicIpCache(ip);\n      }\n      resolve(ip);\n    });\n  }).on('error', () => resolve(undefined));\n});\n\nconst getNetworkAddresses = () => {\n  const interfaces = networkInterfaces();\n  const addresses = {\n    local: 'localhost',\n    private: null,\n    public: null\n  };\n\n  for(const name of Object.keys(interfaces)) {\n    const networkInterface = interfaces[name];\n    if(!networkInterface) {\n      continue;\n    }\n\n    for(const iface of networkInterface) {\n      if(iface.family === 'IPv4' && !iface.internal) {\n        const ip = iface.address;\n\n        // Private IP ranges\n        if(ip.startsWith('10.') || ip.startsWith('192.168.') || ip.startsWith('172.')) {\n          if(!addresses.private) {\n            addresses.private = ip;\n          }\n        } else {\n          // Public IP (not in private ranges)\n          if(!addresses.public) {\n            addresses.public = ip;\n          }\n        }\n      }\n    }\n  }\n\n  return addresses;\n};\n\nconst displayServerStatus = (port: number = 7001, quiet: boolean, publicIp?: string) => {\n  if(quiet) {\n    return;\n  }\n\n  const addresses = getNetworkAddresses();\n  const localUrl = `http://localhost:${port}`;\n  const privateUrl = addresses.private ? `http://${addresses.private}:${port}` : null;\n  let publicUrl = null;\n  if(publicIp) {\n    publicUrl = `http://${publicIp}:${port}`;\n  } else if(addresses.public) {\n    publicUrl = `http://${addresses.public}:${port}`;\n  }\n\n  let urlLines = `${chalk.green('Local:')}     ${chalk.underline(localUrl)}\\n`;\n\n  if(privateUrl) {\n    urlLines += `${chalk.green('Private:')}   ${chalk.underline(privateUrl)}\\n`;\n  }\n\n  if(publicUrl) {\n    urlLines += `${chalk.green('Public:')}    ${chalk.underline(publicUrl)}\\n`;\n  }\n\n  const statusBox = boxen(\n    `${chalk.cyan.bold('\uD83D\uDE80 Development Server Running')}\\n\\n${urlLines}\\n` +\n    `${chalk.yellow('Press Ctrl+C to stop the server')}`,\n    {\n      backgroundColor: '#1a1a1a',\n      borderColor: 'cyan',\n      borderStyle: 'round',\n      margin: 1,\n      padding: 1\n    }\n  );\n\n  // eslint-disable-next-line no-console\n  console.log(`\\n${statusBox}\\n`);\n};\n\nexport const dev = async (cmd: DevOptions, callback: DevCallback = () => ({})): Promise<number> => {\n  const {bundleAnalyzer, cliName = 'Lex', config, open = false, quiet, remove, translations = false, usePublicIp, variables} = cmd;\n\n  const spinner = createSpinner(quiet);\n\n  log(`${cliName} start development server...`, 'info', quiet);\n\n  await LexConfig.parseConfig(cmd);\n\n  const {outputFullPath, useTypescript} = LexConfig.config;\n\n  let variablesObj: object = {NODE_ENV: 'development'};\n\n  if(variables) {\n    try {\n      variablesObj = JSON.parse(variables);\n    } catch (_error) {\n      log(`\\n${cliName} Error: Environment variables option is not a valid JSON object.`, 'error', quiet);\n      callback(1);\n      return 1;\n    }\n  }\n\n  process.env = {...process.env, ...variablesObj};\n\n  if(useTypescript) {\n    LexConfig.checkTypescriptConfig();\n  }\n\n  if(remove) {\n    spinner.start('Cleaning output directory...');\n\n    await removeFiles(outputFullPath || '');\n\n    spinner.succeed('Successfully cleaned output directory!');\n  }\n\n  // Process translations if flag is enabled (before starting dev server)\n  if(translations) {\n    spinner.start('Processing translations...');\n\n    try {\n      const sourcePath = LexConfig.config.sourceFullPath || process.cwd();\n      const outputPath = LexConfig.config.outputFullPath || 'lib';\n\n      await processTranslations(sourcePath, outputPath, quiet);\n      spinner.succeed('Translations processed successfully!');\n    } catch (translationError) {\n      log(`\\n${cliName} Error: Failed to process translations: ${translationError.message}`, 'error', quiet);\n      spinner.fail('Failed to process translations.');\n      callback(1);\n      return 1;\n    }\n  }\n\n  let webpackConfig: string;\n\n  if(config) {\n    const isRelativeConfig: boolean = config.substr(0, 2) === './';\n    webpackConfig = isRelativeConfig ? pathResolve(process.cwd(), config) : config;\n  } else {\n    const {webpackConfig: resolvedConfig} = resolveWebpackPaths(currentDirname);\n    webpackConfig = resolvedConfig;\n  }\n\n  const {webpackPath} = resolveWebpackPaths(currentDirname);\n\n  const webpackOptions: string[] = [\n    '--color',\n    '--watch',\n    '--config', webpackConfig\n  ];\n\n  if(bundleAnalyzer) {\n    webpackOptions.push('--bundleAnalyzer');\n  }\n\n  try {\n    const finalWebpackOptions = webpackPath === 'npx' ? ['webpack', ...webpackOptions] : webpackOptions;\n\n    spinner.start('Starting development server...');\n\n    const childProcess = execa(webpackPath, finalWebpackOptions, {\n      encoding: 'utf8',\n      env: {\n        LEX_QUIET: quiet,\n        WEBPACK_DEV_OPEN: open\n      },\n      stdio: 'pipe'\n    } as any);\n\n    let serverStarted = false;\n    let detectedPort = 7001;\n\n    childProcess.stdout?.on('data', (data: Buffer) => {\n      const output = data.toString();\n\n      handleWebpackProgress(output, spinner, quiet, '\uD83D\uDE80', 'Webpack Building');\n\n      if(!serverStarted && (output.includes('Local:') || output.includes('webpack compiled') || output.includes('webpack-plugin-serve') || output.includes('http://localhost') || output.includes('listening on port'))) {\n        serverStarted = true;\n        spinner.succeed('Development server started.');\n\n        // Try multiple patterns to detect the port\n        const portMatch = output.match(/Local:\\s*http:\\/\\/[^:]+:(\\d+)/) ||\n          output.match(/http:\\/\\/localhost:(\\d+)/) ||\n          output.match(/port:\\s*(\\d+)/) ||\n          output.match(/listening on port (\\d+)/) ||\n          output.match(/WebpackPluginServe listening on port (\\d+)/);\n        if(portMatch) {\n          detectedPort = parseInt(portMatch[1]);\n        }\n\n        displayServerStatus(detectedPort, quiet);\n        fetchPublicIp(usePublicIp).then((publicIp) => {\n          if(publicIp) {\n            displayServerStatus(detectedPort, quiet, publicIp);\n          }\n        });\n      }\n    });\n\n    childProcess.stderr?.on('data', (data: Buffer) => {\n      const output = data.toString();\n\n      handleWebpackProgress(output, spinner, quiet, '\uD83D\uDE80', 'Webpack Building');\n\n      if(!serverStarted && (output.includes('Local:') || output.includes('webpack compiled') || output.includes('webpack-plugin-serve') || output.includes('http://localhost') || output.includes('listening on port'))) {\n        serverStarted = true;\n        spinner.succeed('Development server started.');\n\n        // Try multiple patterns to detect the port\n        const portMatch = output.match(/Local:\\s*http:\\/\\/[^:]+:(\\d+)/) ||\n          output.match(/http:\\/\\/localhost:(\\d+)/) ||\n          output.match(/port:\\s*(\\d+)/) ||\n          output.match(/listening on port (\\d+)/) ||\n          output.match(/WebpackPluginServe listening on port (\\d+)/);\n        if(portMatch) {\n          detectedPort = parseInt(portMatch[1]);\n        }\n\n        displayServerStatus(detectedPort, quiet);\n        fetchPublicIp(usePublicIp).then((publicIp) => {\n          if(publicIp) {\n            displayServerStatus(detectedPort, quiet, publicIp);\n          }\n        });\n      }\n    });\n\n    setTimeout(() => {\n      if(!serverStarted) {\n        spinner.succeed('Development server started.');\n        displayServerStatus(detectedPort, quiet);\n        fetchPublicIp(usePublicIp).then((publicIp) => {\n          if(publicIp) {\n            displayServerStatus(detectedPort, quiet, publicIp);\n          }\n        });\n      }\n    }, 5000);\n\n    await childProcess;\n\n    if(!serverStarted) {\n      spinner.succeed('Development server started.');\n      displayServerStatus(detectedPort, quiet);\n      fetchPublicIp(usePublicIp).then((publicIp) => {\n        if(publicIp) {\n          displayServerStatus(detectedPort, quiet, publicIp);\n        }\n      });\n    }\n\n    callback(0);\n    return 0;\n  } catch (error) {\n    log(`\\n${cliName} Error: ${error.message}`, 'error', quiet);\n\n    spinner.fail('There was an error while running Webpack.');\n\n    callback(1);\n    return 1;\n  }\n};"],
  "mappings": "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",
  "names": []
}

|
|
281
|
+
|
|
282
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/commands/dev/dev.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 boxen from 'boxen';\nimport chalk from 'chalk';\nimport {execa} from 'execa';\nimport {existsSync, readFileSync, writeFileSync, mkdirSync} from 'fs';\nimport https from 'https';\nimport {networkInterfaces, homedir} from 'os';\nimport {dirname, resolve as pathResolve, join} from 'path';\n\nimport {LexConfig} from '../../LexConfig.js';\nimport {createSpinner, handleWebpackProgress, removeFiles} from '../../utils/app.js';\nimport {resolveWebpackPaths} from '../../utils/file.js';\nimport {log} from '../../utils/log.js';\nimport {processTranslations} from '../../utils/translations.js';\n\nlet currentFilename: string;\nlet currentDirname: string;\n\ntry {\n  // eslint-disable-next-line no-eval\n  currentFilename = eval('require(\"url\").fileURLToPath(import.meta.url)');\n  currentDirname = dirname(currentFilename);\n} catch{\n  currentFilename = process.cwd();\n  currentDirname = process.cwd();\n}\n\nexport interface DevOptions {\n  readonly bundleAnalyzer?: boolean;\n  readonly cliName?: string;\n  readonly config?: string;\n  readonly format?: string;\n  readonly open?: boolean;\n  readonly port?: number;\n  readonly quiet?: boolean;\n  readonly remove?: boolean;\n  readonly translations?: boolean;\n  readonly usePublicIp?: boolean;\n  readonly variables?: string;\n}\n\nexport type DevCallback = (status: number) => void;\n\ninterface PublicIpCache {\n  ip: string;\n  timestamp: number;\n}\n\nconst getCacheDir = (): string => {\n  const cacheDir = join(homedir(), '.lex-cache');\n  if(!existsSync(cacheDir)) {\n    mkdirSync(cacheDir, {recursive: true});\n  }\n  return cacheDir;\n};\n\nconst getCachePath = (): string => join(getCacheDir(), 'public-ip.json');\n\nconst readPublicIpCache = (): PublicIpCache | null => {\n  const cachePath = getCachePath();\n  if(!existsSync(cachePath)) {\n    return null;\n  }\n\n  try {\n    const cacheData = readFileSync(cachePath, 'utf8');\n    const cache: PublicIpCache = JSON.parse(cacheData);\n    const oneWeekMs = 7 * 24 * 60 * 60 * 1000;\n\n    if(Date.now() - cache.timestamp > oneWeekMs) {\n      return null;\n    }\n\n    return cache;\n  } catch{\n    return null;\n  }\n};\n\nconst writePublicIpCache = (ip: string): void => {\n  const cachePath = getCachePath();\n  const cache: PublicIpCache = {\n    ip,\n    timestamp: Date.now()\n  };\n  writeFileSync(cachePath, JSON.stringify(cache, null, 2));\n};\n\nconst fetchPublicIp = (forceRefresh: boolean = false): Promise<string | undefined> => new Promise((resolve) => {\n  if(!forceRefresh) {\n    const cached = readPublicIpCache();\n    if(cached) {\n      resolve(cached.ip);\n      return;\n    }\n  }\n\n  https.get('https://api.ipify.org', (res) => {\n    let data = '';\n    res.on('data', (chunk) => (data += chunk));\n    res.on('end', () => {\n      const ip = data.trim();\n      if(ip) {\n        writePublicIpCache(ip);\n      }\n      resolve(ip);\n    });\n  }).on('error', () => resolve(undefined));\n});\n\nconst getNetworkAddresses = () => {\n  const interfaces = networkInterfaces();\n  const addresses = {\n    local: 'localhost',\n    private: null,\n    public: null\n  };\n\n  for(const name of Object.keys(interfaces)) {\n    const networkInterface = interfaces[name];\n    if(!networkInterface) {\n      continue;\n    }\n\n    for(const iface of networkInterface) {\n      if(iface.family === 'IPv4' && !iface.internal) {\n        const ip = iface.address;\n\n        if(ip.startsWith('10.') || ip.startsWith('192.168.') || ip.startsWith('172.')) {\n          if(!addresses.private) {\n            addresses.private = ip;\n          }\n        } else {\n          if(!addresses.public) {\n            addresses.public = ip;\n          }\n        }\n      }\n    }\n  }\n\n  return addresses;\n};\n\nconst displayServerStatus = (port: number = 3000, quiet: boolean, publicIp?: string) => {\n  if(quiet) {\n    return;\n  }\n\n  const addresses = getNetworkAddresses();\n  const localUrl = `http://localhost:${port}`;\n  const privateUrl = addresses.private ? `http://${addresses.private}:${port}` : null;\n  let publicUrl = null;\n  if(publicIp) {\n    publicUrl = `http://${publicIp}:${port}`;\n  } else if(addresses.public) {\n    publicUrl = `http://${addresses.public}:${port}`;\n  }\n\n  let urlLines = `${chalk.green('Local:')}     ${chalk.underline(localUrl)}\\n`;\n\n  if(privateUrl) {\n    urlLines += `${chalk.green('Private:')}   ${chalk.underline(privateUrl)}\\n`;\n  }\n\n  if(publicUrl) {\n    urlLines += `${chalk.green('Public:')}    ${chalk.underline(publicUrl)}\\n`;\n  }\n\n  const statusBox = boxen(\n    `${chalk.cyan.bold('🚀 Development Server Running')}\\n\\n${urlLines}\\n` +\n    `${chalk.yellow('Press Ctrl+C to stop the server')}`,\n    {\n      backgroundColor: '#1a1a1a',\n      borderColor: 'cyan',\n      borderStyle: 'round',\n      margin: 1,\n      padding: 1\n    }\n  );\n\n  // eslint-disable-next-line no-console\n  console.log(`\\n${statusBox}\\n`);\n};\n\nexport const dev = async (cmd: DevOptions, callback: DevCallback = () => ({})): Promise<number> => {\n  const {bundleAnalyzer, cliName = 'Lex', config, format = 'esm', open = false, port = 3000, quiet, remove, translations = false, usePublicIp, variables} = cmd;\n\n  const spinner = createSpinner(quiet);\n\n  log(`${cliName} start development server...`, 'info', quiet);\n\n  await LexConfig.parseConfig(cmd);\n\n  const {outputFullPath, useTypescript} = LexConfig.config;\n\n  let variablesObj: object = {NODE_ENV: 'development'};\n\n  if(variables) {\n    try {\n      variablesObj = JSON.parse(variables);\n    } catch(_error) {\n      log(`\\n${cliName} Error: Environment variables option is not a valid JSON object.`, 'error', quiet);\n      callback(1);\n      return 1;\n    }\n  }\n\n  process.env = {...process.env, ...variablesObj};\n\n  if(useTypescript) {\n    LexConfig.checkTypescriptConfig();\n  }\n\n  if(remove) {\n    spinner.start('Cleaning output directory...');\n\n    await removeFiles(outputFullPath || '');\n\n    spinner.succeed('Successfully cleaned output directory!');\n  }\n\n  if(translations) {\n    spinner.start('Processing translations...');\n\n    try {\n      const sourcePath = LexConfig.config.sourceFullPath || process.cwd();\n      const outputPath = LexConfig.config.outputFullPath || 'lib';\n\n      await processTranslations(sourcePath, outputPath, quiet);\n      spinner.succeed('Translations processed successfully!');\n    } catch(translationError) {\n      log(`\\n${cliName} Error: Failed to process translations: ${translationError.message}`, 'error', quiet);\n      spinner.fail('Failed to process translations.');\n      callback(1);\n      return 1;\n    }\n  }\n\n  let webpackConfig: string;\n\n  if(config) {\n    const isRelativeConfig: boolean = config.substr(0, 2) === './';\n    webpackConfig = isRelativeConfig ? pathResolve(process.cwd(), config) : config;\n  } else {\n    const {webpackConfig: resolvedConfig} = resolveWebpackPaths(currentDirname);\n    webpackConfig = resolvedConfig;\n  }\n\n  const {webpackPath} = resolveWebpackPaths(currentDirname);\n\n  const webpackOptions: string[] = [\n    '--color',\n    '--watch',\n    '--config', webpackConfig\n  ];\n\n  if(bundleAnalyzer) {\n    webpackOptions.push('--bundleAnalyzer');\n  }\n\n  if(port !== 3000) {\n    webpackOptions.push('--port', port.toString());\n  }\n\n  try {\n    const finalWebpackOptions = webpackPath === 'npx' ? ['webpack', ...webpackOptions] : webpackOptions;\n\n    spinner.start('Starting development server...');\n\n    const childProcess = execa(webpackPath, finalWebpackOptions, {\n      encoding: 'utf8',\n      env: {\n        LEX_QUIET: quiet,\n        WEBPACK_DEV_OPEN: open\n      },\n      stdio: 'pipe'\n    } as any);\n\n    let serverStarted = false;\n    let statusShown = false;\n    const showStatusOnce = (portToShow: number) => {\n      if(statusShown) {\n        return;\n      }\n      statusShown = true;\n      if(usePublicIp) {\n        fetchPublicIp(usePublicIp).then((publicIp) => {\n          displayServerStatus(portToShow, quiet, publicIp);\n        });\n      } else {\n        displayServerStatus(portToShow, quiet);\n      }\n    };\n    let detectedPort = 3000;\n\n    childProcess.stdout?.on('data', (data: Buffer) => {\n      const output = data.toString();\n\n      handleWebpackProgress(output, spinner, quiet, '🚀', 'Webpack Building');\n\n      if(!serverStarted && (output.includes('Local:') || output.includes('webpack compiled') || output.includes('webpack-plugin-serve') || output.includes('http://localhost') || output.includes('listening on port'))) {\n        serverStarted = true;\n        spinner.succeed('Development server started.');\n\n        const portMatch = output.match(/Local:\\s*http:\\/\\/[^:]+:(\\d+)/) ||\n          output.match(/http:\\/\\/localhost:(\\d+)/) ||\n          output.match(/port:\\s*(\\d+)/) ||\n          output.match(/listening on port (\\d+)/) ||\n          output.match(/WebpackPluginServe listening on port (\\d+)/);\n        if(portMatch) {\n          detectedPort = parseInt(portMatch[1]);\n        }\n\n        showStatusOnce(detectedPort);\n      }\n    });\n\n    childProcess.stderr?.on('data', (data: Buffer) => {\n      const output = data.toString();\n\n      handleWebpackProgress(output, spinner, quiet, '🚀', 'Webpack Building');\n\n      if(!serverStarted && (output.includes('Local:') || output.includes('webpack compiled') || output.includes('webpack-plugin-serve') || output.includes('http://localhost') || output.includes('listening on port'))) {\n        serverStarted = true;\n        spinner.succeed('Development server started.');\n\n        const portMatch = output.match(/Local:\\s*http:\\/\\/[^:]+:(\\d+)/) ||\n          output.match(/http:\\/\\/localhost:(\\d+)/) ||\n          output.match(/port:\\s*(\\d+)/) ||\n          output.match(/listening on port (\\d+)/) ||\n          output.match(/WebpackPluginServe listening on port (\\d+)/);\n        if(portMatch) {\n          detectedPort = parseInt(portMatch[1]);\n        }\n\n        showStatusOnce(detectedPort);\n      }\n    });\n\n    setTimeout(() => {\n      if(!serverStarted) {\n        spinner.succeed('Development server started.');\n        showStatusOnce(detectedPort);\n      }\n    }, 5000);\n\n    await childProcess;\n\n    if(!serverStarted) {\n      spinner.succeed('Development server started.');\n      showStatusOnce(detectedPort);\n    }\n\n    callback(0);\n    return 0;\n  } catch(error) {\n    log(`\\n${cliName} Error: ${error.message}`, 'error', quiet);\n\n    spinner.fail('There was an error while running Webpack.');\n\n    callback(1);\n    return 1;\n  }\n};"],"names":["boxen","chalk","execa","existsSync","readFileSync","writeFileSync","mkdirSync","https","networkInterfaces","homedir","dirname","resolve","pathResolve","join","LexConfig","createSpinner","handleWebpackProgress","removeFiles","resolveWebpackPaths","log","processTranslations","currentFilename","currentDirname","eval","process","cwd","getCacheDir","cacheDir","recursive","getCachePath","readPublicIpCache","cachePath","cacheData","cache","JSON","parse","oneWeekMs","Date","now","timestamp","writePublicIpCache","ip","stringify","fetchPublicIp","forceRefresh","Promise","cached","get","res","data","on","chunk","trim","undefined","getNetworkAddresses","interfaces","addresses","local","private","public","name","Object","keys","networkInterface","iface","family","internal","address","startsWith","displayServerStatus","port","quiet","publicIp","localUrl","privateUrl","publicUrl","urlLines","green","underline","statusBox","cyan","bold","yellow","backgroundColor","borderColor","borderStyle","margin","padding","console","dev","cmd","callback","bundleAnalyzer","cliName","config","format","open","remove","translations","usePublicIp","variables","spinner","parseConfig","outputFullPath","useTypescript","variablesObj","NODE_ENV","_error","env","checkTypescriptConfig","start","succeed","sourcePath","sourceFullPath","outputPath","translationError","message","fail","webpackConfig","isRelativeConfig","substr","resolvedConfig","webpackPath","webpackOptions","push","toString","finalWebpackOptions","childProcess","encoding","LEX_QUIET","WEBPACK_DEV_OPEN","stdio","serverStarted","statusShown","showStatusOnce","portToShow","then","detectedPort","stdout","output","includes","portMatch","match","parseInt","stderr","setTimeout","error"],"mappings":"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"}
|