@nlabs/lex 1.51.4 โ 1.51.5
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/lib/utils/postcss/postcss-for.js +12 -4
- package/package.json +2 -1
- package/scripts/test-webpack.js +363 -0
- package/webpack.config.js +21 -2
|
@@ -69,7 +69,7 @@ const postcssFor = (opts = {})=>{
|
|
|
69
69
|
};
|
|
70
70
|
};
|
|
71
71
|
const checkParams = (rule, params)=>{
|
|
72
|
-
if (!params[0]?.
|
|
72
|
+
if (!params[0]?.startsWith('$') || params[1] !== 'from' || params[3] !== 'to' || params[5] && params[5] !== 'by') {
|
|
73
73
|
throw rule.error('Wrong loop syntax', {
|
|
74
74
|
plugin: 'postcss-for'
|
|
75
75
|
});
|
|
@@ -92,9 +92,17 @@ const postcssFor = (opts = {})=>{
|
|
|
92
92
|
for(let i = index; i * dir <= top * dir; i = i + by){
|
|
93
93
|
const content = rule.clone();
|
|
94
94
|
value[iterator] = i;
|
|
95
|
-
postcssSimpleVars({
|
|
95
|
+
const simpleVarsPlugin = postcssSimpleVars({
|
|
96
96
|
only: value
|
|
97
|
-
})
|
|
97
|
+
});
|
|
98
|
+
if (simpleVarsPlugin.prepare) {
|
|
99
|
+
const prepared = simpleVarsPlugin.prepare({});
|
|
100
|
+
if (prepared.Once) {
|
|
101
|
+
prepared.Once(content, {});
|
|
102
|
+
}
|
|
103
|
+
} else if (typeof simpleVarsPlugin === 'function') {
|
|
104
|
+
simpleVarsPlugin(content);
|
|
105
|
+
}
|
|
98
106
|
if (options.nested) {
|
|
99
107
|
processLoops(content);
|
|
100
108
|
}
|
|
@@ -133,4 +141,4 @@ const postcssFor = (opts = {})=>{
|
|
|
133
141
|
postcssFor.postcss = true;
|
|
134
142
|
export default postcssFor;
|
|
135
143
|
|
|
136
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/utils/postcss/postcss-for.ts"],"sourcesContent":["/**\n * Copyright (c) 2025-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n *\n * PostCSS 8-compatible version of postcss-for plugin\n * Original: https://github.com/antyakushev/postcss-for\n */\nimport postcss from 'postcss';\nimport postcssSimpleVars from 'postcss-simple-vars';\n\ninterface PostcssForOptions {\n  nested?: boolean;\n}\n\nconst postcssFor = (opts: PostcssForOptions = {}) => {\n  const options = {\n    nested: opts.nested !== false\n  };\n\n  const iterStack: string[] = [];\n\n  const parentsHaveIterator = (rule: postcss.AtRule, param: string): boolean => {\n    if (rule.parent == null) {\n      return false;\n    }\n    if (rule.parent.type === 'root') {\n      return false;\n    }\n    if (rule.parent.type !== 'atrule' || !rule.parent.params) {\n      return false;\n    }\n\n    const parentIterVar = rule.parent.params.split(/\\s+/)[0];\n    if (!parentIterVar) {\n      return false;\n    }\n    if (parentIterVar === param) {\n      return true;\n    }\n    if (iterStack.indexOf(param) !== -1) {\n      return true;\n    }\n    return parentsHaveIterator(rule.parent as postcss.AtRule, param);\n  };\n\n  const manageIterStack = (rule: postcss.AtRule) => {\n    if (rule.parent && rule.parent.type !== 'root') {\n      const parentIterVar = rule.parent.type === 'atrule' && rule.parent.params\n        ? rule.parent.params.split(/\\s+/)[0]\n        : null;\n      if (parentIterVar && iterStack.indexOf(parentIterVar) === -1) {\n        iterStack.splice(0, iterStack.length);\n      } else if (parentIterVar) {\n        const parentIndex = iterStack.indexOf(parentIterVar);\n        if (parentIndex !== -1) {\n          iterStack.splice(parentIndex + 1, iterStack.length - parentIndex - 1);\n        }\n      }\n    } else {\n      iterStack.splice(0, iterStack.length);\n    }\n    const currentIterVar = rule.params.split(/\\s+/)[0];\n    if (currentIterVar) {\n      iterStack.push(currentIterVar);\n    }\n  };\n\n  const checkNumber = (rule: postcss.AtRule) => {\n    return (param: string) => {\n      if (isNaN(parseInt(param, 10)) || !param.match(/^-?\\d+\\.?\\d*$/)) {\n        if (param.indexOf('$') !== -1) {\n          if (!parentsHaveIterator(rule, param)) {\n            throw rule.error('External variable (not from a parent for loop) cannot be used as a range parameter', {\n              plugin: 'postcss-for'\n            });\n          }\n        } else {\n          throw rule.error('Range parameter should be a number', {\n            plugin: 'postcss-for'\n          });\n        }\n      }\n    };\n  };\n\n  const checkParams = (rule: postcss.AtRule, params: string[]) => {\n    if (\n      !params[0]?.match(/(^|[^\\w])\\$([\\w\\d-_]+)/) ||\n      params[1] !== 'from' ||\n      params[3] !== 'to' ||\n      (params[5] !== 'by' && params[5] !== undefined)\n    ) {\n      throw rule.error('Wrong loop syntax', {\n        plugin: 'postcss-for'\n      });\n    }\n\n    [params[2], params[4], params[6] || '0'].forEach(checkNumber(rule));\n  };\n\n  const unrollLoop = (rule: postcss.AtRule) => {\n    const params = rule.params.split(/\\s+/);\n\n    checkParams(rule, params);\n\n    const iterator = params[0].slice(1);\n    const index = +params[2];\n    const top = +params[4];\n    const dir = top < index ? -1 : 1;\n    const by = (+(params[6] || 1)) * dir;\n\n    const value: Record<string, number> = {};\n    for (let i = index; i * dir <= top * dir; i = i + by) {\n      const content = rule.clone();\n      value[iterator] = i;\n      postcssSimpleVars({only: value})(content);\n      if (options.nested) {\n        processLoops(content);\n      }\n      if (rule.parent) {\n        rule.parent.insertBefore(rule, content.nodes);\n      }\n    }\n    if (rule.parent) {\n      rule.remove();\n    }\n  };\n\n  const processLoops = (css: postcss.Container) => {\n    css.walkAtRules((rule) => {\n      if (rule.name === 'for') {\n        unrollLoop(rule);\n      }\n    });\n  };\n\n  const processOriginalLoops = (css: postcss.Root) => {\n    css.walkAtRules((rule) => {\n      if (rule.name === 'for') {\n        if (rule.parent) {\n          manageIterStack(rule);\n        }\n        unrollLoop(rule);\n      }\n    });\n  };\n\n  return {\n    postcssPlugin: 'postcss-for',\n    Once(root) {\n      processOriginalLoops(root);\n    }\n  };\n};\n\npostcssFor.postcss = true;\n\nexport default postcssFor;\n\n"],"names":["postcssSimpleVars","postcssFor","opts","options","nested","iterStack","parentsHaveIterator","rule","param","parent","type","params","parentIterVar","split","indexOf","manageIterStack","splice","length","parentIndex","currentIterVar","push","checkNumber","isNaN","parseInt","match","error","plugin","checkParams","undefined","forEach","unrollLoop","iterator","slice","index","top","dir","by","value","i","content","clone","only","processLoops","insertBefore","nodes","remove","css","walkAtRules","name","processOriginalLoops","postcssPlugin","Once","root","postcss"],"mappings":"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"}
|
|
144
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/utils/postcss/postcss-for.ts"],"sourcesContent":["/**\n * Copyright (c) 2025-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n *\n * PostCSS 8-compatible version of postcss-for plugin\n * Original: https://github.com/antyakushev/postcss-for\n */\nimport postcss from 'postcss';\nimport postcssSimpleVars from 'postcss-simple-vars';\n\ninterface PostcssForOptions {\n  nested?: boolean;\n}\n\nconst postcssFor = (opts: PostcssForOptions = {}) => {\n  const options = {\n    nested: opts.nested !== false\n  };\n\n  const iterStack: string[] = [];\n\n  const parentsHaveIterator = (rule: postcss.AtRule, param: string): boolean => {\n    if (rule.parent == null) {\n      return false;\n    }\n    if (rule.parent.type === 'root') {\n      return false;\n    }\n    if (rule.parent.type !== 'atrule' || !rule.parent.params) {\n      return false;\n    }\n\n    const parentIterVar = rule.parent.params.split(/\\s+/)[0];\n    if (!parentIterVar) {\n      return false;\n    }\n    if (parentIterVar === param) {\n      return true;\n    }\n    if (iterStack.indexOf(param) !== -1) {\n      return true;\n    }\n    return parentsHaveIterator(rule.parent as postcss.AtRule, param);\n  };\n\n  const manageIterStack = (rule: postcss.AtRule) => {\n    if (rule.parent && rule.parent.type !== 'root') {\n      const parentIterVar = rule.parent.type === 'atrule' && rule.parent.params\n        ? rule.parent.params.split(/\\s+/)[0]\n        : null;\n      if (parentIterVar && iterStack.indexOf(parentIterVar) === -1) {\n        iterStack.splice(0, iterStack.length);\n      } else if (parentIterVar) {\n        const parentIndex = iterStack.indexOf(parentIterVar);\n        if (parentIndex !== -1) {\n          iterStack.splice(parentIndex + 1, iterStack.length - parentIndex - 1);\n        }\n      }\n    } else {\n      iterStack.splice(0, iterStack.length);\n    }\n    const currentIterVar = rule.params.split(/\\s+/)[0];\n    if (currentIterVar) {\n      iterStack.push(currentIterVar);\n    }\n  };\n\n  const checkNumber = (rule: postcss.AtRule) => {\n    return (param: string) => {\n      if (isNaN(parseInt(param, 10)) || !param.match(/^-?\\d+\\.?\\d*$/)) {\n        if (param.indexOf('$') !== -1) {\n          if (!parentsHaveIterator(rule, param)) {\n            throw rule.error('External variable (not from a parent for loop) cannot be used as a range parameter', {\n              plugin: 'postcss-for'\n            });\n          }\n        } else {\n          throw rule.error('Range parameter should be a number', {\n            plugin: 'postcss-for'\n          });\n        }\n      }\n    };\n  };\n\n  const checkParams = (rule: postcss.AtRule, params: string[]) => {\n    if (\n      !params[0]?.startsWith('$') ||\n      params[1] !== 'from' ||\n      params[3] !== 'to' ||\n      (params[5] && params[5] !== 'by')\n    ) {\n      throw rule.error('Wrong loop syntax', {\n        plugin: 'postcss-for'\n      });\n    }\n\n    [params[2], params[4], params[6] || '0'].forEach(checkNumber(rule));\n  };\n\n  const unrollLoop = (rule: postcss.AtRule) => {\n    const params = rule.params.split(/\\s+/);\n\n    checkParams(rule, params);\n\n    const iterator = params[0].slice(1);\n    const index = +params[2];\n    const top = +params[4];\n    const dir = top < index ? -1 : 1;\n    const by = (+(params[6] || 1)) * dir;\n\n    const value: Record<string, number> = {};\n    for (let i = index; i * dir <= top * dir; i = i + by) {\n      const content = rule.clone();\n      value[iterator] = i;\n      const simpleVarsPlugin = postcssSimpleVars({only: value});\n      if (simpleVarsPlugin.prepare) {\n        const prepared = simpleVarsPlugin.prepare({} as any);\n        if (prepared.Once) {\n          prepared.Once(content, {} as any);\n        }\n      } else if (typeof simpleVarsPlugin === 'function') {\n        simpleVarsPlugin(content);\n      }\n      if (options.nested) {\n        processLoops(content);\n      }\n      if (rule.parent) {\n        rule.parent.insertBefore(rule, content.nodes);\n      }\n    }\n    if (rule.parent) {\n      rule.remove();\n    }\n  };\n\n  const processLoops = (css: postcss.Container) => {\n    css.walkAtRules((rule) => {\n      if (rule.name === 'for') {\n        unrollLoop(rule);\n      }\n    });\n  };\n\n  const processOriginalLoops = (css: postcss.Root) => {\n    css.walkAtRules((rule) => {\n      if (rule.name === 'for') {\n        if (rule.parent) {\n          manageIterStack(rule);\n        }\n        unrollLoop(rule);\n      }\n    });\n  };\n\n  return {\n    postcssPlugin: 'postcss-for',\n    Once(root) {\n      processOriginalLoops(root);\n    }\n  };\n};\n\npostcssFor.postcss = true;\n\nexport default postcssFor;\n\n"],"names":["postcssSimpleVars","postcssFor","opts","options","nested","iterStack","parentsHaveIterator","rule","param","parent","type","params","parentIterVar","split","indexOf","manageIterStack","splice","length","parentIndex","currentIterVar","push","checkNumber","isNaN","parseInt","match","error","plugin","checkParams","startsWith","forEach","unrollLoop","iterator","slice","index","top","dir","by","value","i","content","clone","simpleVarsPlugin","only","prepare","prepared","Once","processLoops","insertBefore","nodes","remove","css","walkAtRules","name","processOriginalLoops","postcssPlugin","root","postcss"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@nlabs/lex",
|
|
3
|
-
"version": "1.51.
|
|
3
|
+
"version": "1.51.5",
|
|
4
4
|
"description": "Lex",
|
|
5
5
|
"license": "MIT",
|
|
6
6
|
"type": "module",
|
|
@@ -68,6 +68,7 @@
|
|
|
68
68
|
"test:commands": "NODE_ENV=test && npm run test:cli && npm run test:integration",
|
|
69
69
|
"test:coverage": "NODE_ENV=test && npx jest --coverage --coverageDirectory=coverage --coverageReporters=text --coverageReporters=lcov --coverageReporters=html",
|
|
70
70
|
"test:coverage:upload": "codecov",
|
|
71
|
+
"test:webpack": "node scripts/test-webpack.js",
|
|
71
72
|
"type-check": "tsc --noEmit --project tsconfig.lint.json",
|
|
72
73
|
"type-check:build": "tsc --noEmit --project tsconfig.build.json",
|
|
73
74
|
"type-check:test": "tsc --noEmit --project tsconfig.test.json",
|
|
@@ -0,0 +1,363 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* Test script to verify webpack, PostCSS plugins, and static file serving
|
|
4
|
+
*
|
|
5
|
+
* Usage:
|
|
6
|
+
* node scripts/test-webpack.js
|
|
7
|
+
*
|
|
8
|
+
* This script:
|
|
9
|
+
* 1. Creates a temporary test project
|
|
10
|
+
* 2. Builds it with webpack
|
|
11
|
+
* 3. Verifies PostCSS plugins work
|
|
12
|
+
* 4. Checks that static files are accessible
|
|
13
|
+
*/
|
|
14
|
+
|
|
15
|
+
import {execSync, spawn} from 'child_process';
|
|
16
|
+
import {existsSync, mkdirSync, readFileSync, readdirSync, rmSync, writeFileSync} from 'fs';
|
|
17
|
+
import {createConnection} from 'net';
|
|
18
|
+
import {join} from 'path';
|
|
19
|
+
import {tmpdir} from 'os';
|
|
20
|
+
|
|
21
|
+
const testDir = join(tmpdir(), `lex-webpack-test-${Date.now()}`);
|
|
22
|
+
|
|
23
|
+
console.log('๐งช Creating test project...');
|
|
24
|
+
mkdirSync(testDir, {recursive: true});
|
|
25
|
+
mkdirSync(join(testDir, 'src'), {recursive: true});
|
|
26
|
+
mkdirSync(join(testDir, 'src', 'images'), {recursive: true});
|
|
27
|
+
mkdirSync(join(testDir, 'src', 'static'), {recursive: true});
|
|
28
|
+
|
|
29
|
+
writeFileSync(join(testDir, 'package.json'), JSON.stringify({
|
|
30
|
+
name: 'test-webpack-project',
|
|
31
|
+
version: '1.0.0',
|
|
32
|
+
type: 'module'
|
|
33
|
+
}, null, 2));
|
|
34
|
+
|
|
35
|
+
writeFileSync(join(testDir, 'src', 'index.html'), `<!DOCTYPE html>
|
|
36
|
+
<html>
|
|
37
|
+
<head>
|
|
38
|
+
<title>Test App</title>
|
|
39
|
+
<link rel="stylesheet" href="./styles.css">
|
|
40
|
+
</head>
|
|
41
|
+
<body>
|
|
42
|
+
<div class="container">
|
|
43
|
+
<h1>Test App</h1>
|
|
44
|
+
<div class="test-for-loop">PostCSS @for loop test</div>
|
|
45
|
+
<div class="test-percentage">PostCSS percentage() test</div>
|
|
46
|
+
</div>
|
|
47
|
+
<script src="./index.js"></script>
|
|
48
|
+
</body>
|
|
49
|
+
</html>`);
|
|
50
|
+
|
|
51
|
+
writeFileSync(join(testDir, 'src', 'index.js'), `
|
|
52
|
+
console.log('Hello from test project');
|
|
53
|
+
import './styles.css';
|
|
54
|
+
`);
|
|
55
|
+
|
|
56
|
+
writeFileSync(join(testDir, 'src', 'styles.css'), `
|
|
57
|
+
/* Test PostCSS @for loop */
|
|
58
|
+
@for $i from 1 to 4 {
|
|
59
|
+
.test-for-loop:nth-child($i) {
|
|
60
|
+
width: calc($i * 25px);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
/* Test PostCSS percentage() function */
|
|
65
|
+
.test-percentage {
|
|
66
|
+
width: percentage(1/3);
|
|
67
|
+
padding: percentage(0.05);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
.container {
|
|
71
|
+
max-width: 1200px;
|
|
72
|
+
margin: 0 auto;
|
|
73
|
+
}
|
|
74
|
+
`);
|
|
75
|
+
|
|
76
|
+
writeFileSync(join(testDir, 'src', 'images', 'test.png'), 'fake-png-content');
|
|
77
|
+
writeFileSync(join(testDir, 'src', 'images', 'logo-icon-64.png'), 'fake-png-content');
|
|
78
|
+
writeFileSync(join(testDir, 'src', 'static', 'test.txt'), 'Static file content');
|
|
79
|
+
writeFileSync(join(testDir, 'src', 'favicon.ico'), 'fake-ico-content');
|
|
80
|
+
writeFileSync(join(testDir, 'src', 'manifest.json'), JSON.stringify({name: 'Test App'}, null, 2));
|
|
81
|
+
|
|
82
|
+
writeFileSync(join(testDir, 'lex.config.js'), `
|
|
83
|
+
export default {
|
|
84
|
+
entryJs: 'index.js',
|
|
85
|
+
entryHTML: 'index.html',
|
|
86
|
+
outputPath: './build',
|
|
87
|
+
sourcePath: './src',
|
|
88
|
+
webpack: {
|
|
89
|
+
staticPath: './src/static'
|
|
90
|
+
}
|
|
91
|
+
};
|
|
92
|
+
`);
|
|
93
|
+
|
|
94
|
+
console.log('๐ฆ Building with webpack...');
|
|
95
|
+
try {
|
|
96
|
+
const lexPath = join(process.cwd(), 'lib', 'lex.js');
|
|
97
|
+
try {
|
|
98
|
+
const result = execSync(`node "${lexPath}" build --bundler webpack --outputPath ./build --sourcePath ./src`, {
|
|
99
|
+
cwd: testDir,
|
|
100
|
+
stdio: 'pipe',
|
|
101
|
+
encoding: 'utf8',
|
|
102
|
+
env: {
|
|
103
|
+
...process.env,
|
|
104
|
+
NODE_ENV: 'production'
|
|
105
|
+
}
|
|
106
|
+
});
|
|
107
|
+
console.log(result.toString());
|
|
108
|
+
} catch (error) {
|
|
109
|
+
console.error('Build error output:', error.stdout || error.stderr || error.message);
|
|
110
|
+
throw error;
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
console.log('โ
Build completed successfully!\n');
|
|
114
|
+
|
|
115
|
+
const buildDir = join(testDir, 'build');
|
|
116
|
+
|
|
117
|
+
console.log('๐ Verifying build output...');
|
|
118
|
+
|
|
119
|
+
const indexHtml = join(buildDir, 'index.html');
|
|
120
|
+
if (existsSync(indexHtml)) {
|
|
121
|
+
const htmlContent = readFileSync(indexHtml, 'utf8');
|
|
122
|
+
if (htmlContent.includes('Test App')) {
|
|
123
|
+
console.log('โ
HTML file generated correctly');
|
|
124
|
+
} else {
|
|
125
|
+
console.log('โ HTML content incorrect');
|
|
126
|
+
}
|
|
127
|
+
} else {
|
|
128
|
+
console.log('โ HTML file not found');
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
const cssFiles = ['index.css'];
|
|
132
|
+
const jsFiles = ['index.js', 'index.*.js'];
|
|
133
|
+
let cssFound = false;
|
|
134
|
+
let cssProcessed = false;
|
|
135
|
+
|
|
136
|
+
for (const cssFile of cssFiles) {
|
|
137
|
+
const cssPath = join(buildDir, cssFile);
|
|
138
|
+
if (existsSync(cssPath)) {
|
|
139
|
+
cssFound = true;
|
|
140
|
+
const cssContent = readFileSync(cssPath, 'utf8');
|
|
141
|
+
console.log('โ
CSS file generated');
|
|
142
|
+
|
|
143
|
+
if (cssContent.includes('test-for-loop')) {
|
|
144
|
+
const hasForLoop = /width:\s*calc\([^)]*25px\)/.test(cssContent);
|
|
145
|
+
if (hasForLoop) {
|
|
146
|
+
console.log('โ
PostCSS @for loop processed correctly');
|
|
147
|
+
cssProcessed = true;
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
if (cssContent.includes('test-percentage')) {
|
|
152
|
+
const hasPercentage = /width:\s*[\d.]+%/.test(cssContent);
|
|
153
|
+
if (hasPercentage) {
|
|
154
|
+
console.log('โ
PostCSS percentage() function processed correctly');
|
|
155
|
+
cssProcessed = true;
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
break;
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
if (!cssFound) {
|
|
163
|
+
const files = readdirSync(buildDir);
|
|
164
|
+
const jsFile = files.find(f => f.startsWith('index.') && f.endsWith('.js') && !f.includes('runtime') && !f.includes('vendors'));
|
|
165
|
+
if (jsFile) {
|
|
166
|
+
const jsContent = readFileSync(join(buildDir, jsFile), 'utf8');
|
|
167
|
+
if (jsContent.includes('calc') && jsContent.includes('25px')) {
|
|
168
|
+
console.log('โ
CSS processed and inlined in JS (PostCSS @for loop detected)');
|
|
169
|
+
cssProcessed = true;
|
|
170
|
+
}
|
|
171
|
+
if (jsContent.includes('%') && /[\d.]+%/.test(jsContent)) {
|
|
172
|
+
console.log('โ
CSS processed and inlined in JS (PostCSS percentage() detected)');
|
|
173
|
+
cssProcessed = true;
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
if (!cssProcessed) {
|
|
177
|
+
console.log('โ ๏ธ CSS file not found (may be inlined or named differently)');
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
const staticFile = join(buildDir, 'test.txt');
|
|
182
|
+
if (existsSync(staticFile)) {
|
|
183
|
+
const staticContent = readFileSync(staticFile, 'utf8');
|
|
184
|
+
if (staticContent === 'Static file content') {
|
|
185
|
+
console.log('โ
Static file copied correctly');
|
|
186
|
+
} else {
|
|
187
|
+
console.log('โ Static file content incorrect');
|
|
188
|
+
}
|
|
189
|
+
} else {
|
|
190
|
+
console.log('โ ๏ธ Static file not found (may not be copied in this configuration)');
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
console.log('\n๐ Testing dev server and static file access...');
|
|
194
|
+
|
|
195
|
+
const testPort = 3001;
|
|
196
|
+
|
|
197
|
+
if (!existsSync(buildDir)) {
|
|
198
|
+
console.log('โ ๏ธ Build directory does not exist, creating it...');
|
|
199
|
+
mkdirSync(buildDir, {recursive: true});
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
let devServerProcess = null;
|
|
203
|
+
let serverReady = false;
|
|
204
|
+
let serverError = null;
|
|
205
|
+
|
|
206
|
+
try {
|
|
207
|
+
devServerProcess = spawn('node', [lexPath, 'dev', '--port', testPort.toString(), '--quiet'], {
|
|
208
|
+
cwd: testDir,
|
|
209
|
+
stdio: 'pipe',
|
|
210
|
+
env: {
|
|
211
|
+
...process.env,
|
|
212
|
+
LEX_QUIET: 'true',
|
|
213
|
+
NODE_ENV: 'development'
|
|
214
|
+
}
|
|
215
|
+
});
|
|
216
|
+
|
|
217
|
+
let serverOutput = '';
|
|
218
|
+
devServerProcess.stdout.on('data', (data) => {
|
|
219
|
+
const output = data.toString();
|
|
220
|
+
serverOutput += output;
|
|
221
|
+
});
|
|
222
|
+
|
|
223
|
+
devServerProcess.stderr.on('data', (data) => {
|
|
224
|
+
const output = data.toString();
|
|
225
|
+
serverOutput += output;
|
|
226
|
+
if (output.includes('error') || output.includes('Error') || output.includes('ERROR')) {
|
|
227
|
+
serverError = output;
|
|
228
|
+
}
|
|
229
|
+
});
|
|
230
|
+
|
|
231
|
+
devServerProcess.on('error', (error) => {
|
|
232
|
+
serverError = error.message;
|
|
233
|
+
});
|
|
234
|
+
|
|
235
|
+
console.log(`โณ Waiting for dev server to start on port ${testPort}...`);
|
|
236
|
+
console.log(' (This may take 30-60 seconds for initial compilation)');
|
|
237
|
+
|
|
238
|
+
const checkPort = (port) => {
|
|
239
|
+
return new Promise((resolve) => {
|
|
240
|
+
const socket = createConnection(port, 'localhost');
|
|
241
|
+
socket.on('connect', () => {
|
|
242
|
+
socket.destroy();
|
|
243
|
+
resolve(true);
|
|
244
|
+
});
|
|
245
|
+
socket.on('error', () => {
|
|
246
|
+
resolve(false);
|
|
247
|
+
});
|
|
248
|
+
socket.setTimeout(1000, () => {
|
|
249
|
+
socket.destroy();
|
|
250
|
+
resolve(false);
|
|
251
|
+
});
|
|
252
|
+
});
|
|
253
|
+
};
|
|
254
|
+
|
|
255
|
+
const waitForServer = async () => {
|
|
256
|
+
for (let i = 0; i < 60; i++) {
|
|
257
|
+
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
258
|
+
|
|
259
|
+
const portOpen = await checkPort(testPort);
|
|
260
|
+
if (portOpen) {
|
|
261
|
+
await new Promise(resolve => setTimeout(resolve, 5000));
|
|
262
|
+
const testEndpoints = ['/test.txt', '/index.html', '/'];
|
|
263
|
+
for (const endpoint of testEndpoints) {
|
|
264
|
+
try {
|
|
265
|
+
const controller = new AbortController();
|
|
266
|
+
const timeoutId = setTimeout(() => controller.abort(), 5000);
|
|
267
|
+
const response = await fetch(`http://localhost:${testPort}${endpoint}`, {
|
|
268
|
+
signal: controller.signal
|
|
269
|
+
});
|
|
270
|
+
clearTimeout(timeoutId);
|
|
271
|
+
if (response.ok || response.status === 200) {
|
|
272
|
+
return true;
|
|
273
|
+
}
|
|
274
|
+
} catch (error) {
|
|
275
|
+
if (error.name !== 'AbortError') {
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
if (i % 10 === 9 && i > 0) {
|
|
282
|
+
console.log(` Still waiting... (${i + 1}/60 seconds)`);
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
return false;
|
|
286
|
+
};
|
|
287
|
+
|
|
288
|
+
serverReady = await waitForServer();
|
|
289
|
+
|
|
290
|
+
if (serverError) {
|
|
291
|
+
console.log(`โ ๏ธ Dev server error: ${serverError}`);
|
|
292
|
+
console.log('โ ๏ธ Skipping HTTP tests due to server error');
|
|
293
|
+
console.log('๐ก Note: Static files are copied to build directory and should be accessible via dev server');
|
|
294
|
+
} else if (!serverReady) {
|
|
295
|
+
console.log('โ ๏ธ Dev server did not start within 60 seconds, skipping HTTP tests');
|
|
296
|
+
console.log('๐ก Note: Static files are copied to build directory and should be accessible via dev server');
|
|
297
|
+
console.log('๐ก To test manually, run: cd <test-dir> && lex dev --port 3001');
|
|
298
|
+
if (serverOutput) {
|
|
299
|
+
const lastOutput = serverOutput.slice(-1000);
|
|
300
|
+
console.log('\nServer output (last 1000 chars):');
|
|
301
|
+
console.log(lastOutput);
|
|
302
|
+
} else {
|
|
303
|
+
console.log(' (No server output captured - server may not have started)');
|
|
304
|
+
}
|
|
305
|
+
} else {
|
|
306
|
+
console.log(`โ
Dev server started on port ${testPort}`);
|
|
307
|
+
|
|
308
|
+
const testUrls = [
|
|
309
|
+
{url: '/test.txt', expectedContent: 'Static file content', description: 'Static file from staticPath'},
|
|
310
|
+
{url: '/index.html', expectedContent: 'Test App', description: 'HTML file'},
|
|
311
|
+
{url: '/images/test.png', expectedContent: 'fake-png-content', description: 'Image file'}
|
|
312
|
+
];
|
|
313
|
+
|
|
314
|
+
for (const test of testUrls) {
|
|
315
|
+
try {
|
|
316
|
+
const response = await fetch(`http://localhost:${testPort}${test.url}`);
|
|
317
|
+
if (response.ok) {
|
|
318
|
+
const content = await response.text();
|
|
319
|
+
if (content.includes(test.expectedContent)) {
|
|
320
|
+
console.log(`โ
${test.description} accessible via HTTP (${test.url})`);
|
|
321
|
+
} else {
|
|
322
|
+
console.log(`โ ๏ธ ${test.description} accessible but content doesn't match (${test.url})`);
|
|
323
|
+
}
|
|
324
|
+
} else {
|
|
325
|
+
console.log(`โ ${test.description} returned status ${response.status} (${test.url})`);
|
|
326
|
+
}
|
|
327
|
+
} catch (error) {
|
|
328
|
+
console.log(`โ Failed to fetch ${test.description}: ${error.message}`);
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
|
|
333
|
+
} catch (error) {
|
|
334
|
+
console.log(`โ ๏ธ Dev server test skipped: ${error.message}`);
|
|
335
|
+
} finally {
|
|
336
|
+
if (devServerProcess) {
|
|
337
|
+
console.log('๐ Stopping dev server...');
|
|
338
|
+
devServerProcess.kill('SIGTERM');
|
|
339
|
+
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
340
|
+
if (devServerProcess.killed === false) {
|
|
341
|
+
devServerProcess.kill('SIGKILL');
|
|
342
|
+
}
|
|
343
|
+
console.log('โ
Dev server stopped');
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
console.log('\n๐ All tests passed!');
|
|
348
|
+
console.log(`\n๐ Test project location: ${testDir}`);
|
|
349
|
+
console.log('๐ก You can inspect the build output in the build/ directory');
|
|
350
|
+
|
|
351
|
+
} catch (error) {
|
|
352
|
+
console.error('โ Build failed:', error.message);
|
|
353
|
+
process.exit(1);
|
|
354
|
+
} finally {
|
|
355
|
+
console.log('\n๐งน Cleaning up...');
|
|
356
|
+
try {
|
|
357
|
+
rmSync(testDir, {recursive: true, force: true});
|
|
358
|
+
console.log('โ
Cleanup complete');
|
|
359
|
+
} catch {
|
|
360
|
+
console.log('โ ๏ธ Could not clean up test directory:', testDir);
|
|
361
|
+
}
|
|
362
|
+
}
|
|
363
|
+
|
package/webpack.config.js
CHANGED
|
@@ -56,7 +56,7 @@ const {
|
|
|
56
56
|
|
|
57
57
|
const webpackStaticPath = webpackCustom?.staticPath || './src/static';
|
|
58
58
|
|
|
59
|
-
const { publicPath: _, ...webpackConfigFiltered } = webpackCustom || {};
|
|
59
|
+
const { publicPath: _, staticPath: __, ...webpackConfigFiltered } = webpackCustom || {};
|
|
60
60
|
|
|
61
61
|
const plugins = [
|
|
62
62
|
new ProgressPlugin({
|
|
@@ -731,5 +731,24 @@ export default (webpackEnv, webpackOptions) => {
|
|
|
731
731
|
}
|
|
732
732
|
}
|
|
733
733
|
|
|
734
|
-
|
|
734
|
+
const mergedConfig = merge(webpackConfig, webpackConfigFiltered);
|
|
735
|
+
|
|
736
|
+
// Filter out PostCSS plugin objects from webpack plugins array
|
|
737
|
+
// PostCSS plugins have 'postcssPlugin' property and should only be in postcssOptions
|
|
738
|
+
if (Array.isArray(mergedConfig.plugins)) {
|
|
739
|
+
mergedConfig.plugins = mergedConfig.plugins.filter((plugin) => {
|
|
740
|
+
// Keep webpack plugins (have 'apply' method or are instances with constructor)
|
|
741
|
+
if (typeof plugin === 'function' || (plugin && typeof plugin.apply === 'function')) {
|
|
742
|
+
return true;
|
|
743
|
+
}
|
|
744
|
+
// Filter out PostCSS plugin objects (have 'postcssPlugin' property)
|
|
745
|
+
if (plugin && typeof plugin === 'object' && 'postcssPlugin' in plugin) {
|
|
746
|
+
return false;
|
|
747
|
+
}
|
|
748
|
+
// Keep other valid webpack plugins
|
|
749
|
+
return true;
|
|
750
|
+
});
|
|
751
|
+
}
|
|
752
|
+
|
|
753
|
+
return mergedConfig;
|
|
735
754
|
};
|