diff --git a/docs/generated/packages/react/generators/application.json b/docs/generated/packages/react/generators/application.json index 58aae1b3fe8a2b..9e5a6b52e5ef3c 100644 --- a/docs/generated/packages/react/generators/application.json +++ b/docs/generated/packages/react/generators/application.json @@ -111,7 +111,7 @@ }, "e2eTestRunner": { "type": "string", - "enum": ["cypress", "none"], + "enum": ["cypress", "playwright", "none"], "description": "Test runner to use for end to end (E2E) tests.", "default": "cypress" }, diff --git a/docs/generated/packages/react/generators/init.json b/docs/generated/packages/react/generators/init.json index 9c4e5d9f3f6987..63380111587a48 100644 --- a/docs/generated/packages/react/generators/init.json +++ b/docs/generated/packages/react/generators/init.json @@ -18,7 +18,7 @@ "e2eTestRunner": { "description": "Adds the specified E2E test runner.", "type": "string", - "enum": ["cypress", "none"], + "enum": ["cypress", "playwright", "none"], "default": "cypress" }, "skipFormat": { diff --git a/packages/react/src/generators/application/application.spec.ts b/packages/react/src/generators/application/application.spec.ts index d51711c0a48ea3..b4b2a392b728d1 100644 --- a/packages/react/src/generators/application/application.spec.ts +++ b/packages/react/src/generators/application/application.spec.ts @@ -1,3 +1,4 @@ +// eslint-disable-next-line @nx/enforce-module-boundaries import { installedCypressVersion } from '@nx/cypress/src/utils/cypress-version'; import { getProjects, @@ -267,6 +268,24 @@ describe('app', () => { }, ].forEach(hasJsonValue); }); + + it('should setup playwright', async () => { + await applicationGenerator(appTree, { + ...schema, + directory: 'myDir', + e2eTestRunner: 'playwright', + }); + + expect( + appTree.exists('apps/my-dir/my-app-e2e/playwright.config.ts') + ).toBeTruthy(); + expect( + appTree.exists('apps/my-dir/my-app-e2e/src/example.spec.ts') + ).toBeTruthy(); + expect( + readProjectConfiguration(appTree, 'my-app-e2e')?.targets?.e2e?.executor + ).toEqual('@nx/playwright:playwright'); + }); }); it('should create Nx specific template', async () => { @@ -332,7 +351,7 @@ describe('app', () => { ); }); - it('should setup the nrwl web build builder', async () => { + it('should setup the nx web build builder', async () => { await applicationGenerator(appTree, { ...schema, name: 'my-app', @@ -372,7 +391,7 @@ describe('app', () => { }); }); - it('should setup the nrwl vite builder if bundler is vite', async () => { + it('should setup the nx vite builder if bundler is vite', async () => { await applicationGenerator(appTree, { ...schema, name: 'my-app', @@ -394,7 +413,7 @@ describe('app', () => { ).toBeFalsy(); }); - it('should setup the nrwl web dev server builder', async () => { + it('should setup the nx web dev server builder', async () => { await applicationGenerator(appTree, { ...schema, name: 'my-app', @@ -414,7 +433,7 @@ describe('app', () => { }); }); - it('should setup the nrwl vite dev server builder if bundler is vite', async () => { + it('should setup the nx vite dev server builder if bundler is vite', async () => { await applicationGenerator(appTree, { ...schema, name: 'my-app', @@ -486,6 +505,25 @@ describe('app', () => { }); }); + describe('--e2e-test-runner playwright', () => { + it('should setup playwright', async () => { + await applicationGenerator(appTree, { + ...schema, + e2eTestRunner: 'playwright', + }); + + expect( + appTree.exists('apps/my-app-e2e/playwright.config.ts') + ).toBeTruthy(); + expect( + appTree.exists('apps/my-app-e2e/src/example.spec.ts') + ).toBeTruthy(); + expect( + readProjectConfiguration(appTree, 'my-app-e2e')?.targets?.e2e?.executor + ).toEqual('@nx/playwright:playwright'); + }); + }); + describe('--pascalCaseFiles', () => { it('should use upper case app file', async () => { await applicationGenerator(appTree, { ...schema, pascalCaseFiles: true }); @@ -950,6 +988,21 @@ describe('app', () => { ] ).toEqual('dist/my-app2'); }); + + it('should setup playwright', async () => { + await applicationGenerator(appTree, { + ...schema, + name: 'my-app3', + rootProject: true, + e2eTestRunner: 'playwright', + }); + + expect(appTree.exists('e2e/playwright.config.ts')).toBeTruthy(); + expect(appTree.exists('e2e/src/example.spec.ts')).toBeTruthy(); + expect( + readProjectConfiguration(appTree, 'e2e')?.targets?.e2e?.executor + ).toEqual('@nx/playwright:playwright'); + }); }); describe('setup React app with --bundler=vite', () => { diff --git a/packages/react/src/generators/application/application.ts b/packages/react/src/generators/application/application.ts index ab1eec504ba5e5..8cab9eb89d851f 100644 --- a/packages/react/src/generators/application/application.ts +++ b/packages/react/src/generators/application/application.ts @@ -7,7 +7,6 @@ import { createApplicationFiles } from './lib/create-application-files'; import { updateSpecConfig } from './lib/update-jest-config'; import { normalizeOptions } from './lib/normalize-options'; import { addProject, maybeJs } from './lib/add-project'; -import { addCypress } from './lib/add-cypress'; import { addJest } from './lib/add-jest'; import { addRouting } from './lib/add-routing'; import { setDefaults } from './lib/set-defaults'; @@ -40,6 +39,7 @@ import { extractTsConfigBase } from '../../utils/create-ts-config'; import { addSwcDependencies } from '@nx/js/src/utils/swc/add-swc-dependencies'; import * as chalk from 'chalk'; import { showPossibleWarnings } from './lib/show-possible-warnings'; +import { addE2e } from './lib/add-e2e'; async function addLinting(host: Tree, options: NormalizedSchema) { const tasks: GeneratorCallback[] = []; @@ -71,7 +71,7 @@ async function addLinting(host: Tree, options: NormalizedSchema) { ); if (!options.skipPackageJson) { - const installTask = await addDependenciesToPackageJson( + const installTask = addDependenciesToPackageJson( host, extraEslintDependencies.dependencies, extraEslintDependencies.devDependencies @@ -190,8 +190,8 @@ export async function applicationGenerator( const lintTask = await addLinting(host, options); tasks.push(lintTask); - const cypressTask = await addCypress(host, options); - tasks.push(cypressTask); + const e2eTask = await addE2e(host, options); + tasks.push(e2eTask); if (options.unitTestRunner === 'jest') { const jestTask = await addJest(host, options); diff --git a/packages/react/src/generators/application/lib/add-cypress.ts b/packages/react/src/generators/application/lib/add-cypress.ts deleted file mode 100644 index cb65f18050c6ee..00000000000000 --- a/packages/react/src/generators/application/lib/add-cypress.ts +++ /dev/null @@ -1,28 +0,0 @@ -import { ensurePackage, Tree } from '@nx/devkit'; -import { webStaticServeGenerator } from '@nx/web'; -import { nxVersion } from '../../../utils/versions'; -import { NormalizedSchema } from '../schema'; - -export async function addCypress(host: Tree, options: NormalizedSchema) { - if (options.e2eTestRunner !== 'cypress') { - return () => {}; - } - - await webStaticServeGenerator(host, { - buildTarget: `${options.projectName}:build`, - targetName: 'serve-static', - }); - - const { cypressProjectGenerator } = ensurePackage< - typeof import('@nx/cypress') - >('@nx/cypress', nxVersion); - - return await cypressProjectGenerator(host, { - ...options, - name: options.e2eProjectName, - directory: options.directory, - project: options.projectName, - bundler: options.bundler === 'rspack' ? 'webpack' : options.bundler, - skipFormat: true, - }); -} diff --git a/packages/react/src/generators/application/lib/add-e2e.ts b/packages/react/src/generators/application/lib/add-e2e.ts new file mode 100644 index 00000000000000..f5de4f15221117 --- /dev/null +++ b/packages/react/src/generators/application/lib/add-e2e.ts @@ -0,0 +1,59 @@ +import type { GeneratorCallback, Tree } from '@nx/devkit'; +import { + addProjectConfiguration, + ensurePackage, + getPackageManagerCommand, + joinPathFragments, + readProjectConfiguration, +} from '@nx/devkit'; +import { webStaticServeGenerator } from '@nx/web'; + +import { nxVersion } from '../../../utils/versions'; +import { NormalizedSchema } from '../schema'; + +export async function addE2e( + tree: Tree, + options: NormalizedSchema +): Promise { + if (options.e2eTestRunner === 'none') { + return () => {}; + } else if (options.e2eTestRunner === 'cypress') { + webStaticServeGenerator(tree, { + buildTarget: `${options.projectName}:build`, + targetName: 'serve-static', + }); + + const { cypressProjectGenerator } = ensurePackage< + typeof import('@nx/cypress') + >('@nx/cypress', nxVersion); + + return await cypressProjectGenerator(tree, { + ...options, + name: options.e2eProjectName, + directory: options.directory, + project: options.projectName, + bundler: options.bundler === 'rspack' ? 'webpack' : options.bundler, + skipFormat: true, + }); + } else if (options.e2eTestRunner === 'playwright') { + const { configurationGenerator } = ensurePackage< + typeof import('@nx/playwright') + >('@nx/playwright', nxVersion); + addProjectConfiguration(tree, options.e2eProjectName, { + root: options.e2eProjectRoot, + sourceRoot: joinPathFragments(options.e2eProjectRoot, 'src'), + targets: {}, + implicitDependencies: [options.projectName], + }); + return configurationGenerator(tree, { + project: options.e2eProjectName, + skipFormat: true, + skipPackageJson: options.skipPackageJson, + directory: 'src', + js: false, + webServerCommand: `${getPackageManagerCommand().exec} nx serve ${ + options.name + }`, + }); + } +} diff --git a/packages/react/src/generators/application/lib/normalize-options.ts b/packages/react/src/generators/application/lib/normalize-options.ts index 461bf3a0b182bd..7ab1ef94542b47 100644 --- a/packages/react/src/generators/application/lib/normalize-options.ts +++ b/packages/react/src/generators/application/lib/normalize-options.ts @@ -3,6 +3,7 @@ import { assertValidStyle } from '../../../utils/assertion'; import { extractLayoutDirectory, getWorkspaceLayout, + joinPathFragments, names, normalizePath, Tree, @@ -36,6 +37,9 @@ export function normalizeOptions( const appProjectRoot = options.rootProject ? '.' : normalizePath(`${appsDir}/${appDirectory}`); + const e2eProjectRoot = options.rootProject + ? 'e2e' + : joinPathFragments(appsDir, `${appDirectory}-e2e`); const parsedTags = options.tags ? options.tags.split(',').map((s) => s.trim()) @@ -59,6 +63,7 @@ export function normalizeOptions( projectName: appProjectName, appProjectRoot, e2eProjectName, + e2eProjectRoot, parsedTags, fileName, styledModule, diff --git a/packages/react/src/generators/application/schema.d.ts b/packages/react/src/generators/application/schema.d.ts index 7b8ae279ddbac1..a742bcf57a1904 100644 --- a/packages/react/src/generators/application/schema.d.ts +++ b/packages/react/src/generators/application/schema.d.ts @@ -9,7 +9,7 @@ export interface Schema { tags?: string; unitTestRunner?: 'jest' | 'vitest' | 'none'; inSourceTests?: boolean; - e2eTestRunner: 'cypress' | 'none'; + e2eTestRunner: 'cypress' | 'playwright' | 'none'; linter: Linter; pascalCaseFiles?: boolean; classComponent?: boolean; @@ -32,6 +32,7 @@ export interface NormalizedSchema extends T { projectName: string; appProjectRoot: string; e2eProjectName: string; + e2eProjectRoot: string; parsedTags: string[]; fileName: string; styledModule: null | SupportedStyles; diff --git a/packages/react/src/generators/application/schema.json b/packages/react/src/generators/application/schema.json index f0dc9d5d152195..a28d626fe79fdf 100644 --- a/packages/react/src/generators/application/schema.json +++ b/packages/react/src/generators/application/schema.json @@ -117,7 +117,7 @@ }, "e2eTestRunner": { "type": "string", - "enum": ["cypress", "none"], + "enum": ["cypress", "playwright", "none"], "description": "Test runner to use for end to end (E2E) tests.", "default": "cypress" }, diff --git a/packages/react/src/generators/init/schema.d.ts b/packages/react/src/generators/init/schema.d.ts index 9892a322415a93..7146fd7a3610e6 100644 --- a/packages/react/src/generators/init/schema.d.ts +++ b/packages/react/src/generators/init/schema.d.ts @@ -1,6 +1,6 @@ export interface InitSchema { unitTestRunner?: 'jest' | 'vitest' | 'none'; - e2eTestRunner?: 'cypress' | 'none'; + e2eTestRunner?: 'cypress' | 'playwright' | 'none'; skipFormat?: boolean; skipPackageJson?: boolean; skipHelperLibs?: boolean; diff --git a/packages/react/src/generators/init/schema.json b/packages/react/src/generators/init/schema.json index 78b86fa7791cd2..56e9c902989bae 100644 --- a/packages/react/src/generators/init/schema.json +++ b/packages/react/src/generators/init/schema.json @@ -15,7 +15,7 @@ "e2eTestRunner": { "description": "Adds the specified E2E test runner.", "type": "string", - "enum": ["cypress", "none"], + "enum": ["cypress", "playwright", "none"], "default": "cypress" }, "skipFormat": { diff --git a/packages/web/src/generators/application/application.ts b/packages/web/src/generators/application/application.ts index d839f2893cf3eb..3a854b7f11d083 100644 --- a/packages/web/src/generators/application/application.ts +++ b/packages/web/src/generators/application/application.ts @@ -260,7 +260,7 @@ export async function applicationGenerator(host: Tree, schema: Schema) { } if (options.e2eTestRunner === 'cypress') { - const { cypressProjectGenerator } = await ensurePackage< + const { cypressProjectGenerator } = ensurePackage< typeof import('@nx/cypress') >('@nx/cypress', nxVersion); const cypressTask = await cypressProjectGenerator(host, { @@ -273,9 +273,10 @@ export async function applicationGenerator(host: Tree, schema: Schema) { tasks.push(cypressTask); } if (options.unitTestRunner === 'jest') { - const { configurationGenerator } = await ensurePackage< - typeof import('@nx/jest') - >('@nx/jest', nxVersion); + const { configurationGenerator } = ensurePackage( + '@nx/jest', + nxVersion + ); const jestTask = await configurationGenerator(host, { project: options.projectName, skipSerializers: true,