diff --git a/e2e/vite/src/vite.test.ts b/e2e/vite/src/vite.test.ts index 49d3a24912f483..2dab72a8f67352 100644 --- a/e2e/vite/src/vite.test.ts +++ b/e2e/vite/src/vite.test.ts @@ -5,6 +5,7 @@ import { directoryExists, exists, fileExists, + getPackageManagerCommand, killPorts, listFiles, newProject, @@ -14,12 +15,14 @@ import { removeFile, rmDist, runCLI, + runCommand, runCLIAsync, runCommandUntil, tmpProjPath, uniq, updateFile, updateJson, + checkFilesExist, } from '@nx/e2e/utils'; import { join } from 'path'; @@ -282,7 +285,7 @@ export function App() { <${buildableLibCmp} /> <${nonBuildableLibCmp} />

{${buildableJsLibFn}()}

- + ); } @@ -354,7 +357,7 @@ export default App; it('should collect coverage', () => { runCLI(`generate @nx/react:lib ${lib} --unitTestRunner=vitest`); updateFile(`libs/${lib}/vite.config.ts`, () => { - return `/// + return `/// import { defineConfig } from 'vite'; import react from '@vitejs/plugin-react'; import { nxViteTsPaths } from '@nx/vite/plugins/nx-tsconfig-paths.plugin'; @@ -475,4 +478,71 @@ export default defineConfig({ expect(result.combinedOutput).toContain(`1 passed`); }, 100_000); }); + + describe('ESM-only apps', () => { + beforeAll(() => { + newProject({ unsetProjectNameAndRootFormat: false }); + }); + + it('should support ESM-only plugins in vite.config.ts for root apps (#NXP-168)', () => { + // ESM-only plugin to test with + updateFile( + 'foo/package.json', + JSON.stringify({ + name: '@acme/foo', + type: 'module', + version: '1.0.0', + main: 'index.js', + }) + ); + updateFile( + 'foo/index.js', + ` + export default function fooPlugin() { + return { + name: 'foo-plugin', + configResolved() { + console.log('Foo plugin'); + } + } + }` + ); + updateJson('package.json', (json) => { + json.devDependencies['@acme/foo'] = 'file:./foo'; + return json; + }); + runCommand(getPackageManagerCommand().install); + + const rootApp = uniq('root'); + runCLI( + `generate @nx/react:app ${rootApp} --rootProject --bundler=vite --unitTestRunner=none --e2eTestRunner=none --style=css --no-interactive` + ); + updateJson(`package.json`, (json) => { + // This allows us to use ESM-only packages in vite.config.ts. + json.type = 'module'; + return json; + }); + updateFile( + `vite.config.ts`, + ` + import fooPlugin from '@acme/foo'; + import { defineConfig } from 'vite'; + import react from '@vitejs/plugin-react'; + import { nxViteTsPaths } from '@nx/vite/plugins/nx-tsconfig-paths.plugin'; + + export default defineConfig({ + cacheDir: './node_modules/.vite/root-app', + server: { + port: 4200, + host: 'localhost', + }, + plugins: [react(), nxViteTsPaths(), fooPlugin()], + });` + ); + + runCLI(`nx build`); + + checkFilesExist(`dist/${rootApp}/index.html`); + }); + }); }); diff --git a/packages/vite/src/utils/options-utils.ts b/packages/vite/src/utils/options-utils.ts index 90808de828f807..63502e87e4c5a4 100644 --- a/packages/vite/src/utils/options-utils.ts +++ b/packages/vite/src/utils/options-utils.ts @@ -89,10 +89,10 @@ export function getViteSharedConfig( const projectRoot = context.projectsConfigurations.projects[context.projectName].root; - const root = relative( - context.cwd, - joinPathFragments(context.root, projectRoot) - ); + const root = + projectRoot === '.' + ? process.cwd() + : relative(context.cwd, joinPathFragments(context.root, projectRoot)); return { mode: options.mode,