import { checkFilesDoNotExist, checkFilesExist, cleanupProject, e2eCwd, expectNoAngularDevkit, expectNoTsJestInJestConfig, getSelectedPackageManager, packageManagerLockFile, readJson, runCreateWorkspace, uniq, } from '@nrwl/e2e/utils'; import { existsSync, mkdirSync } from 'fs-extra'; describe('create-nx-workspace', () => { const packageManager = getSelectedPackageManager() || 'pnpm'; afterEach(() => cleanupProject()); it('should create a workspace with a single angular app at the root with routing', () => { const wsName = uniq('angular'); runCreateWorkspace(wsName, { preset: 'angular-standalone', appName: wsName, style: 'css', packageManager, routing: true, }); checkFilesExist('package.json'); checkFilesExist('src/app/app.routes.ts'); checkFilesExist('project.json'); }); it('should create a workspace with a single angular app at the root without routing', () => { const wsName = uniq('angular'); runCreateWorkspace(wsName, { preset: 'angular-standalone', appName: wsName, style: 'css', packageManager, routing: false, }); checkFilesExist('package.json'); checkFilesExist('project.json'); }); it('should create a workspace with a single react app with vite at the root', () => { const wsName = uniq('react'); runCreateWorkspace(wsName, { preset: 'react-standalone', appName: wsName, style: 'css', packageManager, bundler: 'vite', }); checkFilesExist('package.json'); checkFilesExist('project.json'); checkFilesExist('vite.config.ts'); }); it('should create a workspace with a single react app with webpack at the root', () => { const wsName = uniq('react'); runCreateWorkspace(wsName, { preset: 'react-standalone', appName: wsName, style: 'css', packageManager, bundler: 'webpack', }); checkFilesExist('package.json'); checkFilesExist('project.json'); checkFilesExist('webpack.config.js'); }); it('should be able to create an empty workspace built for apps', () => { const wsName = uniq('apps'); runCreateWorkspace(wsName, { preset: 'empty', packageManager, }); checkFilesExist( 'package.json', packageManagerLockFile[packageManager], 'apps/.gitkeep', 'libs/.gitkeep' ); const foreignLockFiles = Object.keys(packageManagerLockFile) .filter((pm) => pm !== packageManager) .map((pm) => packageManagerLockFile[pm]); expectNoAngularDevkit(); }); it('should be able to create an empty workspace with npm capabilities', () => { const wsName = uniq('npm'); runCreateWorkspace(wsName, { preset: 'npm', packageManager, }); expectNoAngularDevkit(); }); it('should be able to create an empty workspace with ts/js capabilities', () => { const wsName = uniq('ts'); runCreateWorkspace(wsName, { preset: 'npm', packageManager, }); expectNoAngularDevkit(); }); it('should be able to create an angular workspace', () => { const wsName = uniq('angular'); const appName = uniq('app'); runCreateWorkspace(wsName, { preset: 'angular-monorepo', style: 'css', appName, packageManager, routing: true, }); }); it('should fail correctly when preset errors', () => { // Using Angular Preset as the example here to test // It will error when npmScope is of form `--` // Due to a validation error Angular will throw. const wsName = uniq('angular-1-test'); const appName = uniq('app'); try { runCreateWorkspace(wsName, { preset: 'angular-monorepo', style: 'css', appName, packageManager, routing: true, }); } catch (e) { expect(e).toBeTruthy(); } }); it('should be able to create a react workspace with webpack', () => { const wsName = uniq('react'); const appName = uniq('app'); runCreateWorkspace(wsName, { preset: 'react-monorepo', style: 'css', appName, packageManager, bundler: 'webpack', }); expectNoAngularDevkit(); expectNoTsJestInJestConfig(appName); const packageJson = readJson('package.json'); expect(packageJson.devDependencies['@nrwl/webpack']).toBeDefined(); }); it('should be able to create a react workspace with vite', () => { const wsName = uniq('react'); const appName = uniq('app'); runCreateWorkspace(wsName, { preset: 'react-monorepo', style: 'css', appName, packageManager, bundler: 'vite', }); expectNoAngularDevkit(); const packageJson = readJson('package.json'); expect(packageJson.devDependencies['@nrwl/webpack']).not.toBeDefined(); expect(packageJson.devDependencies['@nrwl/vite']).toBeDefined(); }); it('should be able to create an next workspace', () => { const wsName = uniq('next'); const appName = uniq('app'); runCreateWorkspace(wsName, { preset: 'next', style: 'css', appName, packageManager, }); expectNoAngularDevkit(); }); it('should be able to create an web-components workspace', () => { const wsName = uniq('web-components'); const appName = uniq('app'); runCreateWorkspace(wsName, { preset: 'web-components', style: 'css', appName, packageManager, }); expectNoAngularDevkit(); }); it('should be able to create an express workspace', () => { const wsName = uniq('express'); const appName = uniq('app'); runCreateWorkspace(wsName, { preset: 'express', style: 'css', appName, packageManager, }); expectNoAngularDevkit(); }); it('should be able to create react-native workspace', () => { const wsName = uniq('react-native'); const appName = uniq('app'); runCreateWorkspace(wsName, { preset: 'react-native', appName, packageManager: 'npm', }); expectNoAngularDevkit(); }); it('should be able to create an expo workspace', () => { const wsName = uniq('expo'); const appName = uniq('app'); runCreateWorkspace(wsName, { preset: 'expo', appName, packageManager: 'npm', }); expectNoAngularDevkit(); }); it('should be able to create a workspace with a custom base branch and HEAD', () => { const wsName = uniq('branch'); runCreateWorkspace(wsName, { preset: 'empty', base: 'main', packageManager, }); }); it('should be able to create a workspace with custom commit information', () => { const wsName = uniq('branch'); runCreateWorkspace(wsName, { preset: 'empty', extraArgs: '--commit.name="John Doe" --commit.email="myemail@test.com" --commit.message="Custom commit message!"', packageManager, }); }); it('should be able to create a nest workspace', () => { const wsName = uniq('nest'); const appName = uniq('app'); runCreateWorkspace(wsName, { preset: 'nest', appName, packageManager, }); }); it('should respect package manager preference', () => { const wsName = uniq('pm'); const appName = uniq('app'); process.env.YARN_REGISTRY = `http://localhost:4872`; process.env.SELECTED_PM = 'npm'; runCreateWorkspace(wsName, { preset: 'react-monorepo', style: 'css', appName, packageManager: 'npm', bundler: 'webpack', }); checkFilesDoNotExist('yarn.lock'); checkFilesExist('package-lock.json'); process.env.SELECTED_PM = packageManager; }); it('should store package manager preference for angular cli', () => { const wsName = uniq('pm'); const appName = uniq('app'); process.env.YARN_REGISTRY = `http://localhost:4872`; process.env.SELECTED_PM = 'npm'; runCreateWorkspace(wsName, { preset: 'angular-monorepo', appName, style: 'css', packageManager: 'npm', routing: true, }); checkFilesDoNotExist('yarn.lock'); checkFilesExist('package-lock.json'); process.env.SELECTED_PM = packageManager; }); it('should return error when ci workflow is selected but no cloud is set up', () => { const wsName = uniq('github'); const create = runCreateWorkspace(wsName, { preset: 'npm', packageManager, ci: 'circleci', }); checkFilesExist('package.json'); checkFilesDoNotExist('.circleci/config.yml'); }); describe('Use detected package manager', () => { function setupProject(envPm: 'npm' | 'yarn' | 'pnpm') { process.env.SELECTED_PM = envPm; runCreateWorkspace(uniq('pm'), { preset: 'empty', packageManager: envPm, useDetectedPm: true, }); } if (packageManager === 'npm') { it('should use npm when invoked with npx', () => { setupProject('npm'); checkFilesExist(packageManagerLockFile['npm']); checkFilesDoNotExist( packageManagerLockFile['yarn'], packageManagerLockFile['pnpm'] ); process.env.SELECTED_PM = packageManager; }, 90000); } if (packageManager === 'pnpm') { it('should use pnpm when invoked with pnpx', () => { setupProject('pnpm'); checkFilesExist(packageManagerLockFile['pnpm']); checkFilesDoNotExist( packageManagerLockFile['yarn'], packageManagerLockFile['npm'] ); process.env.SELECTED_PM = packageManager; }, 90000); } // skipping due to packageManagerCommand for createWorkspace not using yarn create nx-workspace if (packageManager === 'yarn') { xit('should use yarn when invoked with yarn create', () => { setupProject('yarn'); checkFilesExist(packageManagerLockFile['yarn']); checkFilesDoNotExist( packageManagerLockFile['pnpm'], packageManagerLockFile['npm'] ); process.env.SELECTED_PM = packageManager; }, 90000); } }); }); describe('create-nx-workspace custom parent folder', () => { const tmpDir = `${e2eCwd}/${uniq('with space')}`; const wsName = uniq('empty'); const packageManager = getSelectedPackageManager() || 'pnpm'; afterEach(() => cleanupProject({ cwd: `${tmpDir}/${wsName}` })); it('should handle spaces in workspace path', () => { mkdirSync(tmpDir, { recursive: true }); runCreateWorkspace(wsName, { preset: 'apps', packageManager, cwd: tmpDir, }); expect(existsSync(`${tmpDir}/${wsName}/package.json`)).toBeTruthy(); }); });