Skip to content

Commit 6b56094

Browse files
MarabyteArnaudBarre
andauthoredJul 24, 2023
fix: lightningcss fails with html-proxy (#13776)
Co-authored-by: Arnaud Barré <arnaud.barre@carbometrix.com>
1 parent 883089c commit 6b56094

File tree

8 files changed

+183
-8
lines changed

8 files changed

+183
-8
lines changed
 

‎packages/vite/src/node/plugins/css.ts

+4
Original file line numberDiff line numberDiff line change
@@ -2183,6 +2183,10 @@ async function compileLightningCSS(
21832183
if (filePath === filename) {
21842184
return src
21852185
}
2186+
// This happens with html-proxy (#13776)
2187+
if (!filePath.endsWith('.css')) {
2188+
return src
2189+
}
21862190
return fs.readFileSync(toAbsolute(filePath), 'utf-8')
21872191
},
21882192
async resolve(id, from) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
import { describe, expect, test } from 'vitest'
2+
import { port } from './serve'
3+
import { getColor, page } from '~utils'
4+
5+
const url = `http://localhost:${port}`
6+
7+
describe('injected inline style', () => {
8+
test('injected inline style is present', async () => {
9+
await page.goto(url)
10+
const el = await page.$('.ssr-proxy')
11+
expect(await getColor(el)).toBe('coral')
12+
})
13+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
// this is automatically detected by playground/vitestSetup.ts and will replace
2+
// the default e2e test serve behavior
3+
4+
import path from 'node:path'
5+
import kill from 'kill-port'
6+
import { hmrPorts, ports, rootDir } from '~utils'
7+
8+
export const port = ports['css/lightningcss-proxy']
9+
10+
export async function serve(): Promise<{ close(): Promise<void> }> {
11+
await kill(port)
12+
13+
const { createServer } = await import(path.resolve(rootDir, 'server.js'))
14+
const { app, vite } = await createServer(
15+
rootDir,
16+
hmrPorts['css/lightningcss-proxy'],
17+
)
18+
19+
return new Promise((resolve, reject) => {
20+
try {
21+
const server = app.listen(port, () => {
22+
resolve({
23+
// for test teardown
24+
async close() {
25+
await new Promise((resolve) => {
26+
server.close(resolve)
27+
})
28+
if (vite) {
29+
await vite.close()
30+
}
31+
},
32+
})
33+
})
34+
} catch (e) {
35+
reject(e)
36+
}
37+
})
38+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
<!--[inline-css]-->
2+
<div class="wrapper">
3+
<p>Injected inline style with SSR Proxy</p>
4+
<p class="ssr-proxy">This should be coral</p>
5+
</div>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
"name": "@vitejs/test-css-lightningcss-proxy",
3+
"private": true,
4+
"type": "module",
5+
"scripts": {
6+
"dev": "node server",
7+
"serve": "NODE_ENV=production node server",
8+
"debug": "node --inspect-brk server",
9+
"preview": "vite preview"
10+
},
11+
"devDependencies": {
12+
"lightningcss": "^1.21.5",
13+
"express": "^4.18.2"
14+
}
15+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
import fs from 'node:fs'
2+
import path from 'node:path'
3+
import { fileURLToPath } from 'node:url'
4+
import express from 'express'
5+
6+
const __dirname = path.dirname(fileURLToPath(import.meta.url))
7+
const isTest = process.env.VITEST
8+
9+
const DYNAMIC_STYLES = `
10+
<style>
11+
.ssr-proxy {
12+
color: coral;
13+
}
14+
</style>
15+
`
16+
17+
export async function createServer(root = process.cwd(), hmrPort) {
18+
const resolve = (p) => path.resolve(__dirname, p)
19+
20+
const app = express()
21+
22+
/**
23+
* @type {import('vite').ViteDevServer}
24+
*/
25+
const vite = await (
26+
await import('vite')
27+
).createServer({
28+
root,
29+
logLevel: isTest ? 'error' : 'info',
30+
css: {
31+
transformer: 'lightningcss',
32+
lightningcss: {
33+
drafts: { nesting: true },
34+
},
35+
},
36+
server: {
37+
middlewareMode: true,
38+
watch: {
39+
// During tests we edit the files too fast and sometimes chokidar
40+
// misses change events, so enforce polling for consistency
41+
usePolling: true,
42+
interval: 100,
43+
},
44+
hmr: {
45+
port: hmrPort,
46+
},
47+
},
48+
appType: 'custom',
49+
})
50+
// use vite's connect instance as middleware
51+
app.use(vite.middlewares)
52+
53+
app.use('*', async (req, res, next) => {
54+
try {
55+
let [url] = req.originalUrl.split('?')
56+
if (url.endsWith('/')) url += 'index.html'
57+
58+
if (url.startsWith('/favicon.ico')) {
59+
return res.status(404).end('404')
60+
}
61+
62+
const htmlLoc = resolve(`.${url}`)
63+
let template = fs.readFileSync(htmlLoc, 'utf-8')
64+
65+
template = template.replace('<!--[inline-css]-->', DYNAMIC_STYLES)
66+
67+
// Force calling transformIndexHtml with url === '/', to simulate
68+
// usage by ecosystem that was recommended in the SSR documentation
69+
// as `const url = req.originalUrl`
70+
const html = await vite.transformIndexHtml('/', template)
71+
72+
res.status(200).set({ 'Content-Type': 'text/html' }).end(html)
73+
} catch (e) {
74+
vite && vite.ssrFixStacktrace(e)
75+
console.log(e.stack)
76+
res.status(500).end(e.stack)
77+
}
78+
})
79+
80+
return { app, vite }
81+
}
82+
83+
if (!isTest) {
84+
createServer().then(({ app }) =>
85+
app.listen(5173, () => {
86+
console.log('http://localhost:5173')
87+
}),
88+
)
89+
}

‎playground/test-utils.ts

+2
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@ export const ports = {
3535
'css/postcss-caching': 5005,
3636
'css/postcss-plugins-different-dir': 5006,
3737
'css/dynamic-import': 5007,
38+
'css/lightningcss-proxy': 5008,
3839
}
3940
export const hmrPorts = {
4041
'optimize-missing-deps': 24680,
@@ -43,6 +44,7 @@ export const hmrPorts = {
4344
'ssr-html': 24683,
4445
'ssr-noexternal': 24684,
4546
'ssr-pug': 24685,
47+
'css/lightningcss-proxy': 24686,
4648
}
4749

4850
const hexToNameMap: Record<string, string> = {}

‎pnpm-lock.yaml

+17-8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)
Please sign in to comment.