Skip to content

Commit 36b5ace

Browse files
authoredSep 13, 2024··
fix(workspace): resolve glob pattern once to avoid name collision (#6489)
1 parent 16aa76c commit 36b5ace

File tree

8 files changed

+70
-32
lines changed

8 files changed

+70
-32
lines changed
 

‎docs/guide/workspace.md

+2-5
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ export default [
2626
```
2727
:::
2828

29-
Vitest will consider every folder in `packages` as a separate project even if it doesn't have a config file inside.
29+
Vitest will consider every folder in `packages` as a separate project even if it doesn't have a config file inside. Since Vitest 2.1, if this glob pattern matches any file it will be considered a Vitest config even if it doesn't have a `vitest` in its name.
3030

3131
::: warning
3232
Vitest will not consider the root config as a workspace project (so it will not run tests specified in `include`) unless it is specified in this config.
@@ -44,10 +44,6 @@ export default [
4444

4545
This pattern will only include projects with `vitest.config` file that includes `e2e` and `unit` before the extension.
4646

47-
::: warning
48-
If you are referencing filenames with glob pattern, make sure your config file starts with `vite.config` or `vitest.config`. Otherwise Vitest will skip it.
49-
:::
50-
5147
You can also define projects with inline config. Workspace file supports using both syntaxes at the same time.
5248

5349
:::code-group
@@ -56,6 +52,7 @@ import { defineWorkspace } from 'vitest/config'
5652

5753
// defineWorkspace provides a nice type hinting DX
5854
export default defineWorkspace([
55+
// matches every folder and file inside the `packages` folder
5956
'packages/*',
6057
{
6158
// add "extends" to merge two configs together

‎packages/vitest/src/node/workspace/resolveWorkspace.ts

+35-26
Original file line numberDiff line numberDiff line change
@@ -51,10 +51,11 @@ export async function resolveWorkspace(
5151
const cwd = process.cwd()
5252

5353
const projects: WorkspaceProject[] = []
54+
const fileProjects = [...configFiles, ...nonConfigDirectories]
5455

5556
try {
5657
// we have to resolve them one by one because CWD should depend on the project
57-
for (const filepath of [...configFiles, ...nonConfigDirectories]) {
58+
for (const filepath of fileProjects) {
5859
// if file leads to the root config, then we can just reuse it because we already initialized it
5960
if (vitest.server.config.configFile === filepath) {
6061
const project = await vitest._createCoreProject()
@@ -111,12 +112,20 @@ export async function resolveWorkspace(
111112
const name = project.getName()
112113
if (names.has(name)) {
113114
const duplicate = resolvedProjects.find(p => p.getName() === name && p !== project)!
115+
const filesError = fileProjects.length
116+
? [
117+
'\n\nYour config matched these files:\n',
118+
fileProjects.map(p => ` - ${relative(vitest.config.root, p)}`).join('\n'),
119+
'\n\n',
120+
].join('')
121+
: [' ']
114122
throw new Error([
115123
`Project name "${name}"`,
116124
project.server.config.configFile ? ` from "${relative(vitest.config.root, project.server.config.configFile)}"` : '',
117125
' is not unique.',
118126
duplicate?.server.config.configFile ? ` The project is already defined by "${relative(vitest.config.root, duplicate.server.config.configFile)}".` : '',
119-
' All projects in a workspace should have unique names. Make sure your configuration is correct.',
127+
filesError,
128+
'All projects in a workspace should have unique names. Make sure your configuration is correct.',
120129
].join(''))
121130
}
122131
names.add(name)
@@ -196,36 +205,36 @@ async function resolveWorkspaceProjectConfigs(
196205
else {
197206
projectsOptions.push(await definition)
198207
}
208+
}
199209

200-
if (workspaceGlobMatches.length) {
201-
const globOptions: GlobOptions = {
202-
absolute: true,
203-
dot: true,
204-
onlyFiles: false,
205-
cwd: vitest.config.root,
206-
expandDirectories: false,
207-
ignore: ['**/node_modules/**', '**/*.timestamp-*'],
208-
}
210+
if (workspaceGlobMatches.length) {
211+
const globOptions: GlobOptions = {
212+
absolute: true,
213+
dot: true,
214+
onlyFiles: false,
215+
cwd: vitest.config.root,
216+
expandDirectories: false,
217+
ignore: ['**/node_modules/**', '**/*.timestamp-*'],
218+
}
209219

210-
const workspacesFs = await glob(workspaceGlobMatches, globOptions)
220+
const workspacesFs = await glob(workspaceGlobMatches, globOptions)
211221

212-
await Promise.all(workspacesFs.map(async (filepath) => {
213-
// directories are allowed with a glob like `packages/*`
214-
// in this case every directory is treated as a project
215-
if (filepath.endsWith('/')) {
216-
const configFile = await resolveDirectoryConfig(filepath)
217-
if (configFile) {
218-
workspaceConfigFiles.push(configFile)
219-
}
220-
else {
221-
nonConfigProjectDirectories.push(filepath)
222-
}
222+
await Promise.all(workspacesFs.map(async (filepath) => {
223+
// directories are allowed with a glob like `packages/*`
224+
// in this case every directory is treated as a project
225+
if (filepath.endsWith('/')) {
226+
const configFile = await resolveDirectoryConfig(filepath)
227+
if (configFile) {
228+
workspaceConfigFiles.push(configFile)
223229
}
224230
else {
225-
workspaceConfigFiles.push(filepath)
231+
nonConfigProjectDirectories.push(filepath)
226232
}
227-
}))
228-
}
233+
}
234+
else {
235+
workspaceConfigFiles.push(filepath)
236+
}
237+
}))
229238
}
230239

231240
const projectConfigFiles = Array.from(new Set(workspaceConfigFiles))
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"name": "b"
3+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
import { test } from 'vitest';
2+
3+
test('test - b')
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"name": "a"
3+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
import { test } from 'vitest';
2+
3+
test('test - a')
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
export default [
2+
'projects/*',
3+
'apps/*'
4+
]

‎test/config/test/workspace.test.ts

+17-1
Original file line numberDiff line numberDiff line change
@@ -25,13 +25,29 @@ it('runs the workspace if there are several vitest config files', async () => {
2525
expect(stdout).toContain('2 + 2 = 4')
2626
})
2727

28+
it('correctly resolves workspace projects with a several folder globs', async () => {
29+
const { stderr, stdout } = await runVitest({
30+
root: 'fixtures/workspace/several-folders',
31+
workspace: './fixtures/workspace/several-folders/vitest.workspace.ts',
32+
})
33+
expect(stderr).toBe('')
34+
expect(stdout).toContain('test - a')
35+
expect(stdout).toContain('test - b')
36+
})
37+
2838
it('fails if project names are identical with a nice error message', async () => {
2939
const { stderr } = await runVitest({
3040
root: 'fixtures/workspace/invalid-duplicate-configs',
3141
workspace: './fixtures/workspace/invalid-duplicate-configs/vitest.workspace.ts',
3242
}, [], 'test', {}, { fails: true })
3343
expect(stderr).toContain(
34-
'Project name "test" from "vitest2.config.js" is not unique. The project is already defined by "vitest1.config.js". All projects in a workspace should have unique names. Make sure your configuration is correct.',
44+
`Project name "test" from "vitest2.config.js" is not unique. The project is already defined by "vitest1.config.js".
45+
46+
Your config matched these files:
47+
- vitest1.config.js
48+
- vitest2.config.js
49+
50+
All projects in a workspace should have unique names. Make sure your configuration is correct.`,
3551
)
3652
})
3753

0 commit comments

Comments
 (0)
Please sign in to comment.