Skip to content

Commit 158c4bb

Browse files
authoredJun 9, 2023
fix(mocker): respect namespace import when hoisting vi.mock (#3547)
1 parent bc283ae commit 158c4bb

File tree

2 files changed

+30
-9
lines changed

2 files changed

+30
-9
lines changed
 

‎packages/vitest/src/node/hoistMocks.ts

+17-9
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import MagicString from 'magic-string'
2-
import type { CallExpression, Identifier, ImportDeclaration, VariableDeclaration, Node as _Node } from 'estree'
2+
import type { CallExpression, Identifier, ImportDeclaration, ImportNamespaceSpecifier, VariableDeclaration, Node as _Node } from 'estree'
33
import { findNodeAround, simple as simpleWalk } from 'acorn-walk'
44
import type { AcornNode } from 'rollup'
55

@@ -24,11 +24,6 @@ function isIdentifier(node: any): node is Positioned<Identifier> {
2424
}
2525

2626
function transformImportSpecifiers(node: ImportDeclaration) {
27-
const specifiers = node.specifiers
28-
29-
if (specifiers.length === 1 && specifiers[0].type === 'ImportNamespaceSpecifier')
30-
return specifiers[0].local.name
31-
3227
const dynamicImports = node.specifiers.map((specifier) => {
3328
if (specifier.type === 'ImportDefaultSpecifier')
3429
return `default: ${specifier.local.name}`
@@ -86,12 +81,25 @@ export function hoistMocks(code: string, id: string, parse: (code: string, optio
8681
const transformImportDeclaration = (node: ImportDeclaration) => {
8782
const source = node.source.value as string
8883

84+
const namespace = node.specifiers.find(specifier => specifier.type === 'ImportNamespaceSpecifier') as ImportNamespaceSpecifier | undefined
85+
86+
let code = ''
87+
if (namespace)
88+
code += `const ${namespace.local.name} = await import('${source}')\n`
89+
8990
// if we don't hijack ESM and process this file, then we definetly have mocks,
9091
// so we need to transform imports into dynamic ones, so "vi.mock" can be executed before
9192
const specifiers = transformImportSpecifiers(node)
92-
const code = specifiers
93-
? `const ${specifiers} = await import('${source}')\n`
94-
: `await import('${source}')\n`
93+
94+
if (specifiers) {
95+
if (namespace)
96+
code += `const ${specifiers} = ${namespace.local.name}\n`
97+
else
98+
code += `const ${specifiers} = await import('${source}')\n`
99+
}
100+
else if (!namespace) {
101+
code += `await import('${source}')\n`
102+
}
95103
return code
96104
}
97105

‎test/core/test/injector-mock.test.ts

+13
Original file line numberDiff line numberDiff line change
@@ -58,3 +58,16 @@ test('always hoists all imports but they are under mocks', () => {
5858
const { someValue2 } = await import('./path2.js')"
5959
`)
6060
})
61+
62+
test('correctly mocks namespaced', () => {
63+
expect(hoistSimpleCode(`
64+
import { vi } from 'vitest'
65+
import add, * as AddModule from '../src/add'
66+
vi.mock('../src/add', () => {})
67+
`)).toMatchInlineSnapshot(`
68+
"const { vi } = await import('vitest')
69+
vi.mock('../src/add', () => {})
70+
const AddModule = await import('../src/add')
71+
const { default: add } = AddModule"
72+
`)
73+
})

0 commit comments

Comments
 (0)
Please sign in to comment.