From 64d03a666174f0f4075598edb9df29da818aae6d Mon Sep 17 00:00:00 2001 From: Craigory Coppola Date: Thu, 30 May 2024 13:27:07 -0400 Subject: [PATCH] feat(repo): spin up .ts daemon when running tests that use daemon --- jest.preset.js | 1 + .../executors/cypress/cypress.impl.spec.ts | 5 +++ packages/cypress/src/plugins/plugin.ts | 1 + .../rename-workspace-rules.spec.ts | 2 ++ .../eslint/src/generators/init/init.spec.ts | 2 ++ packages/eslint/src/plugins/plugin.spec.ts | 3 +- packages/eslint/src/plugins/plugin.ts | 2 +- packages/gradle/src/plugin/nodes.ts | 4 +-- .../gradle/src/utils/get-gradle-report.ts | 6 ++-- packages/jest/src/plugins/plugin.spec.ts | 3 ++ packages/nest/jest.config.ts | 1 + .../nest/src/generators/filter/filter.spec.ts | 4 +-- packages/nest/test-setup.ts | 12 +++++++ packages/nx/src/adapter/ngcli-adapter.spec.ts | 8 +++++ packages/nx/src/daemon/client/client.ts | 19 ++++------- .../message-types/get-context-file-data.ts | 2 -- .../daemon/server/handle-context-file-data.ts | 2 +- .../server/handle-get-files-in-directory.ts | 2 +- .../daemon/server/handle-request-file-data.ts | 10 ------ packages/nx/src/daemon/server/server.ts | 26 ++++++++------- ...explicit-package-json-dependencies.spec.ts | 2 +- .../plugins/isolation/plugin-pool.ts | 7 ++-- .../utils/retrieve-workspace-files.spec.ts | 2 +- packages/nx/src/utils/all-file-data.ts | 7 +--- packages/nx/src/utils/workspace-context.ts | 6 ++-- packages/playwright/src/plugins/plugin.ts | 6 ++-- .../src/utils/create-watch-paths.spec.ts | 33 +++++++++++++++++-- packages/vite/src/plugins/plugin.spec.ts | 5 ++- packages/vite/src/plugins/plugin.ts | 10 ++++-- scripts/unit-test-setup.js | 14 ++++++++ typedoc-theme/jest.config.ts | 10 +++++- 31 files changed, 146 insertions(+), 71 deletions(-) create mode 100644 packages/nest/test-setup.ts delete mode 100644 packages/nx/src/daemon/server/handle-request-file-data.ts create mode 100644 scripts/unit-test-setup.js diff --git a/jest.preset.js b/jest.preset.js index 28490c0c213ff3..28be337c9faf31 100644 --- a/jest.preset.js +++ b/jest.preset.js @@ -12,4 +12,5 @@ module.exports = { coverageReporters: ['html'], maxWorkers: 1, testEnvironment: 'node', + setupFiles: ['../../scripts/unit-test-setup.js'], }; diff --git a/packages/cypress/src/executors/cypress/cypress.impl.spec.ts b/packages/cypress/src/executors/cypress/cypress.impl.spec.ts index f60956aa88ca01..7d75a6a7bafa8b 100644 --- a/packages/cypress/src/executors/cypress/cypress.impl.spec.ts +++ b/packages/cypress/src/executors/cypress/cypress.impl.spec.ts @@ -75,6 +75,11 @@ describe('Cypress builder', () => { configuration, }; }; + (devkit as any).logger = { + warn: jest.fn(), + log: jest.fn(), + info: jest.fn(), + }; cypressRun = jest .spyOn(Cypress, 'run') .mockReturnValue(Promise.resolve({})); diff --git a/packages/cypress/src/plugins/plugin.ts b/packages/cypress/src/plugins/plugin.ts index 5313637d35abcd..7634b7f7695f03 100644 --- a/packages/cypress/src/plugins/plugin.ts +++ b/packages/cypress/src/plugins/plugin.ts @@ -210,6 +210,7 @@ async function buildCypressTargets( ? cypressConfig.e2e.excludeSpecPattern.map((p) => join(projectRoot, p)) : [join(projectRoot, cypressConfig.e2e.excludeSpecPattern)]; const specFiles = await globWithWorkspaceContext( + context.workspaceRoot, specPatterns, excludeSpecPatterns ); diff --git a/packages/eslint-plugin/src/migrations/update-17-2-6-rename-workspace-rules/rename-workspace-rules.spec.ts b/packages/eslint-plugin/src/migrations/update-17-2-6-rename-workspace-rules/rename-workspace-rules.spec.ts index c00e45fe28f70e..57d2ba436d87b9 100644 --- a/packages/eslint-plugin/src/migrations/update-17-2-6-rename-workspace-rules/rename-workspace-rules.spec.ts +++ b/packages/eslint-plugin/src/migrations/update-17-2-6-rename-workspace-rules/rename-workspace-rules.spec.ts @@ -3,6 +3,8 @@ import { createTreeWithEmptyWorkspace } from 'nx/src/devkit-testing-exports'; import { WORKSPACE_PLUGIN_DIR } from '../../constants'; import update from './rename-workspace-rules'; +import 'nx/src/internal-testing-utils/mock-project-graph'; + const rule1Name = 'test-rule'; const rule2Name = 'my-rule'; diff --git a/packages/eslint/src/generators/init/init.spec.ts b/packages/eslint/src/generators/init/init.spec.ts index bcf161b287aefb..140bb2aeb7b309 100644 --- a/packages/eslint/src/generators/init/init.spec.ts +++ b/packages/eslint/src/generators/init/init.spec.ts @@ -3,6 +3,7 @@ import 'nx/src/internal-testing-utils/mock-project-graph'; import { NxJsonConfiguration, readJson, Tree, updateJson } from '@nx/devkit'; import { createTreeWithEmptyWorkspace } from '@nx/devkit/testing'; import { LinterInitOptions, lintInitGenerator } from './init'; +import { setWorkspaceRoot } from 'nx/src/utils/workspace-root'; describe('@nx/eslint:init', () => { let tree: Tree; @@ -10,6 +11,7 @@ describe('@nx/eslint:init', () => { beforeEach(() => { tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' }); + setWorkspaceRoot(tree.root); options = { addPlugin: true, }; diff --git a/packages/eslint/src/plugins/plugin.spec.ts b/packages/eslint/src/plugins/plugin.spec.ts index 744fede327aa2b..95a1a953ff6af1 100644 --- a/packages/eslint/src/plugins/plugin.spec.ts +++ b/packages/eslint/src/plugins/plugin.spec.ts @@ -1,5 +1,3 @@ -import 'nx/src/internal-testing-utils/mock-fs'; - let fsRoot: string = ''; jest.mock( @@ -44,6 +42,7 @@ describe('@nx/eslint/plugin', () => { workspaceRoot: tempFs.tempDir, configFiles: [], }; + fsRoot = tempFs.tempDir; }); afterEach(() => { diff --git a/packages/eslint/src/plugins/plugin.ts b/packages/eslint/src/plugins/plugin.ts index 40ba619c1653aa..95b3a805157745 100644 --- a/packages/eslint/src/plugins/plugin.ts +++ b/packages/eslint/src/plugins/plugin.ts @@ -7,6 +7,7 @@ import { import { existsSync } from 'node:fs'; import { dirname, join, normalize, sep } from 'node:path'; import { combineGlobPatterns } from 'nx/src/utils/globs'; +import { globWithWorkspaceContext } from 'nx/src/utils/workspace-context'; import { ESLINT_CONFIG_FILENAMES, baseEsLintConfigFile, @@ -15,7 +16,6 @@ import { } from '../utils/config-file'; import { resolveESLintClass } from '../utils/resolve-eslint-class'; import { gte } from 'semver'; -import { globWithWorkspaceContext } from 'nx/src/utils/workspace-context'; export interface EslintPluginOptions { targetName?: string; diff --git a/packages/gradle/src/plugin/nodes.ts b/packages/gradle/src/plugin/nodes.ts index 69ffc11cf9eb77..953eabcf8cd61d 100644 --- a/packages/gradle/src/plugin/nodes.ts +++ b/packages/gradle/src/plugin/nodes.ts @@ -72,7 +72,7 @@ export const createNodesV2: CreateNodesV2 = [ ); const targetsCache = readTargetsCache(cachePath); - populateGradleReport(context.workspaceRoot); + await populateGradleReport(context.workspaceRoot); const gradleReport = getCurrentGradleReport(); try { @@ -131,7 +131,7 @@ export const createNodes: CreateNodes = [ logger.warn( '`createNodes` is deprecated. Update your plugin to utilize createNodesV2 instead. In Nx 20, this will error.' ); - populateGradleReport(context.workspaceRoot); + await populateGradleReport(context.workspaceRoot); const gradleReport = getCurrentGradleReport(); const internalCreateNodes = makeCreateNodes(gradleReport, {}); return await internalCreateNodes(configFile, options, context); diff --git a/packages/gradle/src/utils/get-gradle-report.ts b/packages/gradle/src/utils/get-gradle-report.ts index 38c00fb9d8487a..76919c29e264c2 100644 --- a/packages/gradle/src/utils/get-gradle-report.ts +++ b/packages/gradle/src/utils/get-gradle-report.ts @@ -36,8 +36,10 @@ export function getCurrentGradleReport() { return gradleReportCache; } -export function populateGradleReport(workspaceRoot: string): void { - const gradleConfigHash = hashWithWorkspaceContext(workspaceRoot, [ +export async function populateGradleReport( + workspaceRoot: string +): Promise { + const gradleConfigHash = await hashWithWorkspaceContext(workspaceRoot, [ gradleConfigGlob, ]); if (gradleReportCache && gradleConfigHash === gradleCurrentConfigHash) { diff --git a/packages/jest/src/plugins/plugin.spec.ts b/packages/jest/src/plugins/plugin.spec.ts index f2e4a177a1cc3a..deb05e3817085c 100644 --- a/packages/jest/src/plugins/plugin.spec.ts +++ b/packages/jest/src/plugins/plugin.spec.ts @@ -3,6 +3,7 @@ import { join } from 'path'; import { createNodes } from './plugin'; import { TempFs } from 'nx/src/internal-testing-utils/temp-fs'; +import { setWorkspaceRoot } from 'nx/src/utils/workspace-root'; describe('@nx/jest/plugin', () => { let createNodesFunction = createNodes[1]; @@ -25,6 +26,8 @@ describe('@nx/jest/plugin', () => { configFiles: [], }; + setWorkspaceRoot(tempFs.tempDir); + await tempFs.createFiles({ 'proj/jest.config.js': `module.exports = {}`, 'proj/src/unit.spec.ts': '', diff --git a/packages/nest/jest.config.ts b/packages/nest/jest.config.ts index 8075e6159956b3..9b464c25a7445f 100644 --- a/packages/nest/jest.config.ts +++ b/packages/nest/jest.config.ts @@ -7,4 +7,5 @@ export default { globals: {}, displayName: 'nest', preset: '../../jest.preset.js', + setupFilesAfterEnv: ['/test-setup.ts'], }; diff --git a/packages/nest/src/generators/filter/filter.spec.ts b/packages/nest/src/generators/filter/filter.spec.ts index fff38d07e21540..44b2fec0d9242c 100644 --- a/packages/nest/src/generators/filter/filter.spec.ts +++ b/packages/nest/src/generators/filter/filter.spec.ts @@ -14,10 +14,10 @@ describe('filter generator', () => { beforeEach(() => { tree = createTreeWithNestApplication(project); - jest.clearAllMocks(); }); it('should run successfully', async () => { - await expect(filterGenerator(tree, options)).resolves.not.toThrowError(); + await filterGenerator(tree, options); + expect(true).toBe(true); }); }); diff --git a/packages/nest/test-setup.ts b/packages/nest/test-setup.ts new file mode 100644 index 00000000000000..5126f328f3f8f3 --- /dev/null +++ b/packages/nest/test-setup.ts @@ -0,0 +1,12 @@ +// If a test uses a util from devkit, but that util +// lives in the Nx package and creates the project graph, +// we need to mock the resolved value inside the Nx package +jest + .spyOn( + require('nx/src/project-graph/project-graph'), + 'createProjectGraphAsync' + ) + .mockResolvedValue({ + nodes: {}, + dependencies: {}, + }); diff --git a/packages/nx/src/adapter/ngcli-adapter.spec.ts b/packages/nx/src/adapter/ngcli-adapter.spec.ts index cb055967a3ead7..2e1946031d09f7 100644 --- a/packages/nx/src/adapter/ngcli-adapter.spec.ts +++ b/packages/nx/src/adapter/ngcli-adapter.spec.ts @@ -5,6 +5,14 @@ import { import { createTreeWithEmptyWorkspace } from '../generators/testing-utils/create-tree-with-empty-workspace'; import { addProjectConfiguration } from '../generators/utils/project-configuration'; +jest.mock('../project-graph/project-graph', () => ({ + ...jest.requireActual('../project-graph/project-graph'), + createProjectGraphAsync: () => ({ + nodes: {}, + externalNodes: {}, + }), +})); + describe('ngcli-adapter', () => { it('arrayBufferToString should support large buffers', () => { const largeString = 'a'.repeat(1000000); diff --git a/packages/nx/src/daemon/client/client.ts b/packages/nx/src/daemon/client/client.ts index 0fb3fa08804af3..29f6524592002f 100644 --- a/packages/nx/src/daemon/client/client.ts +++ b/packages/nx/src/daemon/client/client.ts @@ -4,7 +4,7 @@ import { readFileSync, statSync } from 'fs'; import { FileHandle, open } from 'fs/promises'; import { ensureDirSync, ensureFileSync } from 'fs-extra'; import { connect } from 'net'; -import { join } from 'path'; +import { extname, join } from 'path'; import { performance } from 'perf_hooks'; import { output } from '../../utils/output'; import { getFullOsSocketPath, killSocketOrPath } from '../socket-utils'; @@ -49,9 +49,6 @@ import { NxWorkspaceFiles } from '../../native'; const DAEMON_ENV_SETTINGS = { NX_PROJECT_GLOB_CACHE: 'false', NX_CACHE_PROJECTS_CONFIG: 'false', - - // Used to identify that the code is running in the daemon process. - NX_ON_DAEMON_PROCESS: 'true', }; export type UnregisterCallback = () => void; @@ -283,14 +280,9 @@ export class DaemonClient { return this.sendToDaemonViaQueue(message); } - getWorkspaceContextFileData( - globs: string[], - exclude?: string[] - ): Promise { + getWorkspaceContextFileData(): Promise { const message: HandleContextFileDataMessage = { type: GET_CONTEXT_FILE_DATA, - globs, - exclude, }; return this.sendToDaemonViaQueue(message); } @@ -480,14 +472,17 @@ export class DaemonClient { const backgroundProcess = spawn( process.execPath, - [join(__dirname, '../server/start.js')], + [join(__dirname, `../server/start.js`)], { cwd: workspaceRoot, stdio: ['ignore', this._out.fd, this._err.fd], detached: true, windowsHide: true, shell: false, - env: { ...process.env, ...DAEMON_ENV_SETTINGS }, + env: { + ...process.env, + ...DAEMON_ENV_SETTINGS, + }, } ); backgroundProcess.unref(); diff --git a/packages/nx/src/daemon/message-types/get-context-file-data.ts b/packages/nx/src/daemon/message-types/get-context-file-data.ts index c804b93eaf340a..935f1cbb876091 100644 --- a/packages/nx/src/daemon/message-types/get-context-file-data.ts +++ b/packages/nx/src/daemon/message-types/get-context-file-data.ts @@ -2,8 +2,6 @@ export const GET_CONTEXT_FILE_DATA = 'GET_CONTEXT_FILE_DATA' as const; export type HandleContextFileDataMessage = { type: typeof GET_CONTEXT_FILE_DATA; - globs: string[]; - exclude?: string[]; }; export function isHandleContextFileDataMessage( diff --git a/packages/nx/src/daemon/server/handle-context-file-data.ts b/packages/nx/src/daemon/server/handle-context-file-data.ts index 24d9243d45c2a5..52468d72033d86 100644 --- a/packages/nx/src/daemon/server/handle-context-file-data.ts +++ b/packages/nx/src/daemon/server/handle-context-file-data.ts @@ -3,7 +3,7 @@ import { workspaceRoot } from '../../utils/workspace-root'; import { HandlerResult } from './server'; export async function handleContextFileData(): Promise { - const files = getAllFileDataInContext(workspaceRoot); + const files = await getAllFileDataInContext(workspaceRoot); return { response: JSON.stringify(files), description: 'handleContextFileData', diff --git a/packages/nx/src/daemon/server/handle-get-files-in-directory.ts b/packages/nx/src/daemon/server/handle-get-files-in-directory.ts index aa781ab77980b8..adcfe93f82c4e5 100644 --- a/packages/nx/src/daemon/server/handle-get-files-in-directory.ts +++ b/packages/nx/src/daemon/server/handle-get-files-in-directory.ts @@ -5,7 +5,7 @@ import { HandlerResult } from './server'; export async function handleGetFilesInDirectory( dir: string ): Promise { - const files = getFilesInDirectoryUsingContext(workspaceRoot, dir); + const files = await getFilesInDirectoryUsingContext(workspaceRoot, dir); return { response: JSON.stringify(files), description: 'handleNxWorkspaceFiles', diff --git a/packages/nx/src/daemon/server/handle-request-file-data.ts b/packages/nx/src/daemon/server/handle-request-file-data.ts deleted file mode 100644 index 2bfd24e7e16967..00000000000000 --- a/packages/nx/src/daemon/server/handle-request-file-data.ts +++ /dev/null @@ -1,10 +0,0 @@ -import { getAllFileDataInContext } from '../../utils/workspace-context'; -import { workspaceRoot } from '../../utils/workspace-root'; - -export async function handleRequestFileData() { - const response = JSON.stringify(getAllFileDataInContext(workspaceRoot)); - return { - response, - description: 'handleRequestFileData', - }; -} diff --git a/packages/nx/src/daemon/server/server.ts b/packages/nx/src/daemon/server/server.ts index 6310c497c372d1..dcd51bc464aa24 100644 --- a/packages/nx/src/daemon/server/server.ts +++ b/packages/nx/src/daemon/server/server.ts @@ -26,7 +26,6 @@ import { handleRecordOutputsHash, } from './handle-outputs-tracking'; import { handleProcessInBackground } from './handle-process-in-background'; -import { handleRequestFileData } from './handle-request-file-data'; import { handleRequestProjectGraph } from './handle-request-project-graph'; import { handleRequestShutdown } from './handle-request-shutdown'; import { serverLogger } from './logger'; @@ -132,11 +131,12 @@ async function handleMessage(socket, data: string) { ); } - if (daemonIsOutdated()) { + const outdated = daemonIsOutdated(); + if (outdated) { await respondWithErrorAndExit( socket, - `Lock files changed`, - new Error('LOCK-FILES-CHANGED') + `Daemon outdated`, + new Error(outdated) ); } @@ -164,10 +164,6 @@ async function handleMessage(socket, data: string) { ); } else if (payload.type === 'HASH_TASKS') { await handleResult(socket, 'HASH_TASKS', () => handleHashTasks(payload)); - } else if (payload.type === 'REQUEST_FILE_DATA') { - await handleResult(socket, 'REQUEST_FILE_DATA', () => - handleRequestFileData() - ); } else if (payload.type === 'PROCESS_IN_BACKGROUND') { await handleResult(socket, 'PROCESS_IN_BACKGROUND', () => handleProcessInBackground(payload) @@ -274,8 +270,13 @@ function registerProcessTerminationListeners() { let existingLockHash: string | undefined; -function daemonIsOutdated(): boolean { - return nxVersionChanged() || lockFileHashChanged(); +function daemonIsOutdated(): string | null { + if (nxVersionChanged()) { + return 'NX_VERSION_CHANGED'; + } else if (lockFileHashChanged()) { + return 'LOCK_FILES_CHANGED'; + } + return null; } function nxVersionChanged(): boolean { @@ -332,10 +333,11 @@ const handleWorkspaceChanges: FileWatcherCallback = async ( try { resetInactivityTimeout(handleInactivityTimeout); - if (daemonIsOutdated()) { + const outdatedReason = daemonIsOutdated(); + if (outdatedReason) { await handleServerProcessTermination({ server, - reason: 'Lock file changed', + reason: outdatedReason, }); return; } diff --git a/packages/nx/src/plugins/js/project-graph/build-dependencies/explicit-package-json-dependencies.spec.ts b/packages/nx/src/plugins/js/project-graph/build-dependencies/explicit-package-json-dependencies.spec.ts index 3849124542e8e9..529caa2b39ce51 100644 --- a/packages/nx/src/plugins/js/project-graph/build-dependencies/explicit-package-json-dependencies.spec.ts +++ b/packages/nx/src/plugins/js/project-graph/build-dependencies/explicit-package-json-dependencies.spec.ts @@ -89,7 +89,7 @@ describe('explicit package json dependencies', () => { const fileMap = createFileMap( projectsConfigurations as any, - getAllFileDataInContext(tempFs.tempDir) + await getAllFileDataInContext(tempFs.tempDir) ).fileMap; const builder = new ProjectGraphBuilder(undefined, fileMap.projectFileMap); diff --git a/packages/nx/src/project-graph/plugins/isolation/plugin-pool.ts b/packages/nx/src/project-graph/plugins/isolation/plugin-pool.ts index 64a86e394bbdae..5f076c8ab920c1 100644 --- a/packages/nx/src/project-graph/plugins/isolation/plugin-pool.ts +++ b/packages/nx/src/project-graph/plugins/isolation/plugin-pool.ts @@ -34,13 +34,14 @@ export function loadRemoteNxPlugin( ...(isWorkerTypescript ? { // Ensures that the worker uses the same tsconfig as the main process - TS_NODE_PROJECT: path.join(__dirname, '../../../tsconfig.lib.json'), + TS_NODE_PROJECT: path.join( + __dirname, + '../../../../tsconfig.lib.json' + ), } : {}), }; - delete env.NX_ON_DAEMON_PROCESS; - const worker = fork(workerPath, [], { stdio: ['ignore', 'inherit', 'inherit', 'ipc'], env, diff --git a/packages/nx/src/project-graph/utils/retrieve-workspace-files.spec.ts b/packages/nx/src/project-graph/utils/retrieve-workspace-files.spec.ts index 3d23230b11e692..4499637562e47b 100644 --- a/packages/nx/src/project-graph/utils/retrieve-workspace-files.spec.ts +++ b/packages/nx/src/project-graph/utils/retrieve-workspace-files.spec.ts @@ -25,7 +25,7 @@ describe('retrieveProjectConfigurationPaths', () => { }) ); - const configPaths = retrieveProjectConfigurationPaths(fs.tempDir, [ + const configPaths = await retrieveProjectConfigurationPaths(fs.tempDir, [ { createNodes: [ '{project.json,**/project.json}', diff --git a/packages/nx/src/utils/all-file-data.ts b/packages/nx/src/utils/all-file-data.ts index 4a6a36271b7c7a..7ce3da07886b33 100644 --- a/packages/nx/src/utils/all-file-data.ts +++ b/packages/nx/src/utils/all-file-data.ts @@ -1,12 +1,7 @@ import { FileData } from '../config/project-graph'; -import { daemonClient } from '../daemon/client/client'; import { getAllFileDataInContext } from './workspace-context'; import { workspaceRoot } from './workspace-root'; export function allFileData(): Promise { - if (daemonClient.enabled()) { - return daemonClient.getAllFileData(); - } else { - return Promise.resolve(getAllFileDataInContext(workspaceRoot)); - } + return getAllFileDataInContext(workspaceRoot); } diff --git a/packages/nx/src/utils/workspace-context.ts b/packages/nx/src/utils/workspace-context.ts index 9640965e758f31..ca5f0914e1e765 100644 --- a/packages/nx/src/utils/workspace-context.ts +++ b/packages/nx/src/utils/workspace-context.ts @@ -81,15 +81,15 @@ export function updateFilesInContext( return workspaceContext?.incrementalUpdate(updatedFiles, deletedFiles); } -export function getAllFileDataInContext(workspaceRoot: string) { +export async function getAllFileDataInContext(workspaceRoot: string) { if (isOnDaemon() || !daemonClient.enabled()) { ensureContextAvailable(workspaceRoot); return workspaceContext.allFileData(); } - return daemonClient.getAllFileData(); + return daemonClient.getWorkspaceContextFileData(); } -export function getFilesInDirectoryUsingContext( +export async function getFilesInDirectoryUsingContext( workspaceRoot: string, dir: string ) { diff --git a/packages/playwright/src/plugins/plugin.ts b/packages/playwright/src/plugins/plugin.ts index 69d143a3560e56..7cc373d1747840 100644 --- a/packages/playwright/src/plugins/plugin.ts +++ b/packages/playwright/src/plugins/plugin.ts @@ -163,7 +163,7 @@ async function buildPlaywrightTargets( playwrightConfig.testMatch ??= '**/*.@(spec|test).?(c|m)[jt]s?(x)'; const dependsOn: TargetConfiguration['dependsOn'] = []; - forEachTestFile( + await forEachTestFile( (testFile) => { const relativeSpecFilePath = normalizePath( relative(projectRoot, testFile) @@ -210,7 +210,7 @@ async function buildPlaywrightTargets( return { targets, metadata }; } -function forEachTestFile( +async function forEachTestFile( cb: (path: string) => void, opts: { context: CreateNodesContext; @@ -218,7 +218,7 @@ function forEachTestFile( config: PlaywrightTestConfig; } ) { - const files = getFilesInDirectoryUsingContext( + const files = await getFilesInDirectoryUsingContext( opts.context.workspaceRoot, opts.path ); diff --git a/packages/remix/src/utils/create-watch-paths.spec.ts b/packages/remix/src/utils/create-watch-paths.spec.ts index fe657316ce0984..6139227dfc27f9 100644 --- a/packages/remix/src/utils/create-watch-paths.spec.ts +++ b/packages/remix/src/utils/create-watch-paths.spec.ts @@ -6,10 +6,39 @@ import { describe('createWatchPaths', () => { it('should list root paths of dependencies relative to project root', async () => { - const testDir = joinPathFragments(workspaceRoot, 'e2e/remix'); + // This test is written based on the Nx repo's project graph. + jest + .spyOn(require('@nx/devkit'), 'createProjectGraphAsync') + .mockResolvedValue({ + nodes: { + parent: { + type: 'app', + name: 'parent', + data: { root: 'apps/parent' }, + }, + lib: { + type: 'lib', + name: 'lib', + data: { root: 'packages/lib' }, + }, + example: { + type: 'app', + name: 'example', + data: { root: 'examples/example' }, + }, + }, + dependencies: { + parent: [ + { type: 'static', source: 'parent', target: 'lib' }, + { type: 'static', source: 'parent', target: 'example' }, + ], + example: [{ type: 'static', source: 'example', target: 'lib' }], + }, + }); + const testDir = joinPathFragments(workspaceRoot, 'apps/parent'); const paths = await createWatchPaths(testDir); - expect(paths).toEqual(['../../packages', '../../graph', '../../e2e/utils']); + expect(paths).toEqual(['../../packages', '../../examples']); }); }); diff --git a/packages/vite/src/plugins/plugin.spec.ts b/packages/vite/src/plugins/plugin.spec.ts index 2c5561817322f3..a3ef77ec8cd99b 100644 --- a/packages/vite/src/plugins/plugin.spec.ts +++ b/packages/vite/src/plugins/plugin.spec.ts @@ -1,6 +1,7 @@ import { CreateNodesContext } from '@nx/devkit'; import { createNodes } from './plugin'; import { TempFs } from 'nx/src/internal-testing-utils/temp-fs'; +import { setWorkspaceRoot } from 'nx/src/utils/workspace-root'; jest.mock('vite', () => ({ resolveConfig: jest.fn().mockImplementation(() => { @@ -44,9 +45,11 @@ describe('@nx/vite/plugin', () => { production: ['!{projectRoot}/**/*.spec.ts'], }, }, - workspaceRoot: '', + workspaceRoot: tempFs.tempDir, }; + setWorkspaceRoot(tempFs.tempDir); tempFs.createFileSync('index.html', ''); + tempFs.createFileSync('package.json', ''); }); afterEach(() => { diff --git a/packages/vite/src/plugins/plugin.ts b/packages/vite/src/plugins/plugin.ts index b823ce405fd131..ca05415845cf94 100644 --- a/packages/vite/src/plugins/plugin.ts +++ b/packages/vite/src/plugins/plugin.ts @@ -6,7 +6,6 @@ import { joinPathFragments, readJsonFile, TargetConfiguration, - workspaceRoot, writeJsonFile, } from '@nx/devkit'; import { dirname, isAbsolute, join, relative } from 'path'; @@ -118,7 +117,8 @@ async function buildViteTargets( const { buildOutputs, testOutputs, hasTest, isBuildable } = getOutputs( viteConfig, - projectRoot + projectRoot, + context.workspaceRoot ); const namedInputs = getNamedInputs(projectRoot, context); @@ -244,7 +244,8 @@ function serveStaticTarget(options: VitePluginOptions) { function getOutputs( viteConfig: Record | undefined, - projectRoot: string + projectRoot: string, + workspaceRoot: string ): { buildOutputs: string[]; testOutputs: string[]; @@ -256,6 +257,7 @@ function getOutputs( const buildOutputPath = normalizeOutputPath( build?.outDir, projectRoot, + workspaceRoot, 'dist' ); @@ -267,6 +269,7 @@ function getOutputs( const reportsDirectoryPath = normalizeOutputPath( test?.coverage?.reportsDirectory, projectRoot, + workspaceRoot, 'coverage' ); @@ -281,6 +284,7 @@ function getOutputs( function normalizeOutputPath( outputPath: string | undefined, projectRoot: string, + workspaceRoot: string, path: 'coverage' | 'dist' ): string | undefined { if (!outputPath) { diff --git a/scripts/unit-test-setup.js b/scripts/unit-test-setup.js new file mode 100644 index 00000000000000..c1d106ae767160 --- /dev/null +++ b/scripts/unit-test-setup.js @@ -0,0 +1,14 @@ +module.exports = () => { + process.env.NX_DAEMON = 'false'; + + // Prevents tests from relying on the Nx repo project graph + jest.doMock('@nx/devkit', () => ({ + ...jest.requireActual('@nx/devkit'), + createProjectGraphAsync: jest.fn().mockImplementation(async () => { + return { + nodes: {}, + dependencies: {}, + }; + }), + })); +}; diff --git a/typedoc-theme/jest.config.ts b/typedoc-theme/jest.config.ts index 69cf4a39427b9c..79217ac523e8ee 100644 --- a/typedoc-theme/jest.config.ts +++ b/typedoc-theme/jest.config.ts @@ -1,5 +1,13 @@ +const nxPreset = require('@nx/jest/preset').default; + /* eslint-disable */ export default { + ...nxPreset, + testTimeout: 35000, + testMatch: ['**/+(*.)+(spec|test).+(ts|js)?(x)'], + coverageReporters: ['html'], + maxWorkers: 1, + testEnvironment: 'node', displayName: 'typedoc-theme', globals: {}, @@ -14,5 +22,5 @@ export default { resolver: '../scripts/patched-jest-resolver.js', moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx'], coverageDirectory: '../coverage/typedoc-theme', - preset: '../jest.preset.js', + setupFiles: [], };