Skip to content

Commit 67e7bf2

Browse files
authoredOct 20, 2022
fix(css): remove ?direct in id for postcss process (#10514)
1 parent 7562014 commit 67e7bf2

File tree

3 files changed

+91
-53
lines changed

3 files changed

+91
-53
lines changed
 

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

+83-51
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import fs from 'node:fs'
22
import path from 'node:path'
33
import { describe, expect, test, vi } from 'vitest'
44
import { resolveConfig } from '../../config'
5+
import type { InlineConfig } from '../../config'
56
import { cssPlugin, cssUrlRE, hoistAtRules } from '../../plugins/css'
67

78
describe('search css url function', () => {
@@ -46,13 +47,17 @@ describe('search css url function', () => {
4647
})
4748
})
4849

49-
describe('css path resolutions', () => {
50-
const mockedProjectPath = path.join(process.cwd(), '/foo/bar/project')
51-
const mockedBarCssRelativePath = '/css/bar.module.css'
52-
const mockedFooCssRelativePath = '/css/foo.module.css'
53-
54-
test('cssmodule compose/from path resolutions', async () => {
55-
const config = await resolveConfig(
50+
describe('css modules', () => {
51+
test('css module compose/from path resolutions', async () => {
52+
const mockedProjectPath = path.join(process.cwd(), '/foo/bar/project')
53+
const { transform, resetMock } = await createCssPluginTransform(
54+
{
55+
[path.join(mockedProjectPath, '/css/bar.module.css')]: `\
56+
.bar {
57+
display: block;
58+
background: #f0f;
59+
}`
60+
},
5661
{
5762
resolve: {
5863
alias: [
@@ -62,57 +67,48 @@ describe('css path resolutions', () => {
6267
}
6368
]
6469
}
65-
},
66-
'serve'
70+
}
6771
)
6872

69-
const { transform, buildStart } = cssPlugin(config)
70-
71-
await buildStart.call({})
72-
73-
const mockFs = vi
74-
.spyOn(fs, 'readFile')
75-
// @ts-ignore vi.spyOn not recognize override `fs.readFile` definition.
76-
.mockImplementationOnce((p, encoding, callback) => {
77-
expect(p).toBe(path.join(mockedProjectPath, mockedBarCssRelativePath))
78-
expect(encoding).toBe('utf-8')
79-
callback(
80-
null,
81-
Buffer.from(`
82-
.bar {
83-
display: block;
84-
background: #f0f;
85-
}
86-
`)
87-
)
88-
})
89-
90-
const { code } = await transform.call(
91-
{
92-
addWatchFile() {
93-
return
94-
}
95-
},
96-
`
73+
const result = await transform(
74+
`\
9775
.foo {
98-
position: fixed;
99-
composes: bar from '@${mockedBarCssRelativePath}';
100-
}
101-
`,
102-
path.join(mockedProjectPath, mockedFooCssRelativePath)
76+
position: fixed;
77+
composes: bar from '@/css/bar.module.css';
78+
}`,
79+
'/css/foo.module.css'
10380
)
10481

105-
expect(code).toBe(`
106-
._bar_soicv_2 {
107-
display: block;
108-
background: #f0f;
109-
}
110-
._foo_sctn3_2 {
111-
position: fixed;
82+
expect(result.code).toBe(
83+
`\
84+
._bar_1csqm_1 {
85+
display: block;
86+
background: #f0f;
11287
}
113-
`)
88+
._foo_86148_1 {
89+
position: fixed;
90+
}`
91+
)
92+
93+
resetMock()
94+
})
11495

115-
mockFs.mockReset()
96+
test('custom generateScopedName', async () => {
97+
const { transform, resetMock } = await createCssPluginTransform(undefined, {
98+
css: {
99+
modules: {
100+
generateScopedName: 'custom__[hash:base64:5]'
101+
}
102+
}
103+
})
104+
const css = `\
105+
.foo {
106+
color: red;
107+
}`
108+
const result1 = await transform(css, '/foo.module.css') // server
109+
const result2 = await transform(css, '/foo.module.css?direct') // client
110+
expect(result1.code).toBe(result2.code)
111+
resetMock()
116112
})
117113
})
118114

@@ -205,3 +201,39 @@ describe('hoist @ rules', () => {
205201
`)
206202
})
207203
})
204+
205+
async function createCssPluginTransform(
206+
files?: Record<string, string>,
207+
inlineConfig: InlineConfig = {}
208+
) {
209+
const config = await resolveConfig(inlineConfig, 'serve')
210+
const { transform, buildStart } = cssPlugin(config)
211+
212+
// @ts-expect-error
213+
await buildStart.call({})
214+
215+
const mockFs = vi
216+
.spyOn(fs, 'readFile')
217+
// @ts-expect-error vi.spyOn not recognize override `fs.readFile` definition.
218+
.mockImplementationOnce((p, encoding, callback) => {
219+
callback(null, Buffer.from(files?.[p] ?? ''))
220+
})
221+
222+
return {
223+
async transform(code: string, id: string) {
224+
// @ts-expect-error
225+
return await transform.call(
226+
{
227+
addWatchFile() {
228+
return
229+
}
230+
},
231+
code,
232+
id
233+
)
234+
},
235+
resetMock() {
236+
mockFs.mockReset()
237+
}
238+
}
239+
}

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

+4-2
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@ import {
4242
normalizePath,
4343
parseRequest,
4444
processSrcSet,
45+
removeDirectQuery,
4546
requireResolveFromRootWithFallback
4647
} from '../utils'
4748
import type { Logger } from '../logger'
@@ -914,13 +915,14 @@ async function compileCSS(
914915

915916
let postcssResult: PostCSS.Result
916917
try {
918+
const source = removeDirectQuery(id)
917919
// postcss is an unbundled dep and should be lazy imported
918920
postcssResult = await (await import('postcss'))
919921
.default(postcssPlugins)
920922
.process(code, {
921923
...postcssOptions,
922-
to: id,
923-
from: id,
924+
to: source,
925+
from: source,
924926
...(devSourcemap
925927
? {
926928
map: {

‎packages/vite/src/node/utils.ts

+4
Original file line numberDiff line numberDiff line change
@@ -303,6 +303,7 @@ export function getPotentialTsSrcPaths(filePath: string): string[] {
303303
}
304304

305305
const importQueryRE = /(\?|&)import=?(?:&|$)/
306+
const directRequestRE = /(\?|&)direct=?(?:&|$)/
306307
const internalPrefixes = [
307308
FS_PREFIX,
308309
VALID_ID_PREFIX,
@@ -318,6 +319,9 @@ export const isInternalRequest = (url: string): boolean =>
318319
export function removeImportQuery(url: string): string {
319320
return url.replace(importQueryRE, '$1').replace(trailingSeparatorRE, '')
320321
}
322+
export function removeDirectQuery(url: string): string {
323+
return url.replace(directRequestRE, '$1').replace(trailingSeparatorRE, '')
324+
}
321325

322326
export function injectQuery(url: string, queryToInject: string): string {
323327
// encode percents for consistent behavior with pathToFileURL

0 commit comments

Comments
 (0)
Please sign in to comment.