From 48ddbf49535290aa9cd0e990f44d583c1baab96e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Linus=20Unneb=C3=A4ck?= Date: Fri, 3 Feb 2023 20:02:47 +0700 Subject: [PATCH 1/8] fix: ignore all unrelated messages from child process (#13543) --- CHANGELOG.md | 1 + .../src/workers/ChildProcessWorker.ts | 6 ++++- .../src/workers/NodeThreadsWorker.ts | 6 ++++- .../__tests__/ChildProcessWorker.test.ts | 13 +++++++--- .../__tests__/NodeThreadsWorker.test.ts | 26 +++++++++++++++---- 5 files changed, 41 insertions(+), 11 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index bbfa3ec54860..2fa90977d331 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,7 @@ - `[expect, @jest/expect]` Provide type of `actual` as a generic argument to `Matchers` to allow better-typed extensions ([#13848](https://github.com/facebook/jest/pull/13848)) - `[jest-circus]` Added explicit mention of test failing because `done()` is not being called in error message ([#13847](https://github.com/facebook/jest/pull/13847)) +- `[jest-worker]` Ignore IPC messages not intended for Jest ([#13543](https://github.com/facebook/jest/pull/13543)) ### Chore & Maintenance diff --git a/packages/jest-worker/src/workers/ChildProcessWorker.ts b/packages/jest-worker/src/workers/ChildProcessWorker.ts index 691bbb6edefd..9e0832c1e7fa 100644 --- a/packages/jest-worker/src/workers/ChildProcessWorker.ts +++ b/packages/jest-worker/src/workers/ChildProcessWorker.ts @@ -255,6 +255,9 @@ export default class ChildProcessWorker } private _onMessage(response: ParentMessage) { + // Ignore messages not intended for us + if (!Array.isArray(response)) return; + // TODO: Add appropriate type check let error: any; @@ -311,7 +314,8 @@ export default class ChildProcessWorker break; default: - throw new TypeError(`Unexpected response from worker: ${response[0]}`); + // Ignore messages not intended for us + break; } } diff --git a/packages/jest-worker/src/workers/NodeThreadsWorker.ts b/packages/jest-worker/src/workers/NodeThreadsWorker.ts index 2acf0054ae5d..3a9f86a46be0 100644 --- a/packages/jest-worker/src/workers/NodeThreadsWorker.ts +++ b/packages/jest-worker/src/workers/NodeThreadsWorker.ts @@ -170,6 +170,9 @@ export default class ExperimentalWorker } private _onMessage(response: ParentMessage) { + // Ignore messages not intended for us + if (!Array.isArray(response)) return; + let error; switch (response[0]) { @@ -227,7 +230,8 @@ export default class ExperimentalWorker break; default: - throw new TypeError(`Unexpected response from worker: ${response[0]}`); + // Ignore messages not intended for us + break; } } diff --git a/packages/jest-worker/src/workers/__tests__/ChildProcessWorker.test.ts b/packages/jest-worker/src/workers/__tests__/ChildProcessWorker.test.ts index 9e6b513cb22f..90c9e1b07bdd 100644 --- a/packages/jest-worker/src/workers/__tests__/ChildProcessWorker.test.ts +++ b/packages/jest-worker/src/workers/__tests__/ChildProcessWorker.test.ts @@ -363,7 +363,7 @@ it('creates error instances for known errors', () => { expect(callback3.mock.calls[0][0]).toBe(412); }); -it('throws when the child process returns a strange message', () => { +it('does not throw when the child process returns a strange message', () => { const worker = new Worker({ forkOptions: {}, maxRetries: 3, @@ -378,9 +378,14 @@ it('throws when the child process returns a strange message', () => { ); // Type 27 does not exist. - expect(() => { - forkInterface.emit('message', [27]); - }).toThrow(TypeError); + forkInterface.emit('message', [27]); + + forkInterface.emit('message', 'test'); + forkInterface.emit('message', {foo: 'bar'}); + forkInterface.emit('message', 0); + forkInterface.emit('message', null); + forkInterface.emit('message', Symbol('test')); + forkInterface.emit('message', true); }); it('does not restart the child if it cleanly exited', () => { diff --git a/packages/jest-worker/src/workers/__tests__/NodeThreadsWorker.test.ts b/packages/jest-worker/src/workers/__tests__/NodeThreadsWorker.test.ts index 8d633af24e97..7e0e53dc9d50 100644 --- a/packages/jest-worker/src/workers/__tests__/NodeThreadsWorker.test.ts +++ b/packages/jest-worker/src/workers/__tests__/NodeThreadsWorker.test.ts @@ -365,7 +365,7 @@ it('creates error instances for known errors', () => { expect(callback3.mock.calls[0][0]).toBe(412); }); -it('throws when the thread returns a strange message', () => { +it('does not throw when the thread returns a strange message', () => { const worker = new Worker({ forkOptions: {}, maxRetries: 3, @@ -380,10 +380,26 @@ it('throws when the thread returns a strange message', () => { ); // Type 27 does not exist. - expect(() => { - // @ts-expect-error: Testing internal method - worker._worker.emit('message', [27]); - }).toThrow(TypeError); + // @ts-expect-error: Testing internal method + worker._worker.emit('message', [27]); + + // @ts-expect-error: Testing internal method + worker._worker.emit('message', 'test'); + + // @ts-expect-error: Testing internal method + worker._worker.emit('message', {foo: 'bar'}); + + // @ts-expect-error: Testing internal method + worker._worker.emit('message', 0); + + // @ts-expect-error: Testing internal method + worker._worker.emit('message', null); + + // @ts-expect-error: Testing internal method + worker._worker.emit('message', Symbol('test')); + + // @ts-expect-error: Testing internal method + worker._worker.emit('message', true); }); it('does not restart the thread if it cleanly exited', () => { From ffe2352c781703b427fab10777043fb76d0d4267 Mon Sep 17 00:00:00 2001 From: Tom Mrazauskas Date: Sat, 4 Feb 2023 13:55:11 +0200 Subject: [PATCH 2/8] chore: move `@jest/test-utils` reference (#13468) --- e2e/tsconfig.json | 3 +- .../babel-jest/src/__tests__/tsconfig.json | 2 +- packages/babel-jest/tsconfig.json | 2 +- packages/expect/src/__tests__/tsconfig.json | 2 +- packages/expect/tsconfig.json | 3 +- .../jest-console/src/__tests__/tsconfig.json | 2 +- packages/jest-console/tsconfig.json | 3 +- .../jest-core/src/__tests__/tsconfig.json | 6 +- .../jest-core/src/lib/__tests__/tsconfig.json | 2 +- packages/jest-core/tsconfig.json | 3 +- .../jest-diff/src/__tests__/tsconfig.json | 2 +- packages/jest-diff/tsconfig.json | 3 +- .../src/__tests__/tsconfig.json | 2 +- packages/jest-environment-jsdom/tsconfig.json | 3 +- .../src/__tests__/tsconfig.json | 2 +- packages/jest-environment-node/tsconfig.json | 3 +- .../src/__tests__/tsconfig.json | 2 +- packages/jest-fake-timers/tsconfig.json | 3 +- packages/jest-haste-map/package.json | 1 - packages/jest-haste-map/tsconfig.json | 3 +- .../src/__tests__/tsconfig.json | 2 +- packages/jest-matcher-utils/tsconfig.json | 3 +- .../src/__tests__/tsconfig.json | 2 +- packages/jest-reporters/tsconfig.json | 3 +- .../src/__tests__/tsconfig.json | 2 +- .../jest-resolve-dependencies/tsconfig.json | 3 +- packages/jest-runner/package.json | 1 + .../jest-runner/src/__tests__/tsconfig.json | 2 +- .../jest-runtime/src/__tests__/tsconfig.json | 2 +- packages/jest-runtime/tsconfig.json | 3 +- .../jest-snapshot/src/__tests__/tsconfig.json | 2 +- packages/jest-snapshot/tsconfig.json | 3 +- .../src/__tests__/tsconfig.json | 2 +- packages/jest-test-sequencer/tsconfig.json | 6 +- .../src/__tests__/tsconfig.json | 2 +- packages/jest-transform/tsconfig.json | 3 +- packages/test-utils/package.json | 1 + packages/test-utils/src/ConditionalTest.ts | 7 +-- packages/test-utils/tsconfig.json | 6 +- scripts/buildTs.mjs | 60 +++++++++++++++++++ yarn.lock | 3 +- 41 files changed, 108 insertions(+), 62 deletions(-) diff --git a/e2e/tsconfig.json b/e2e/tsconfig.json index c3053b6da00f..dba558ebe941 100644 --- a/e2e/tsconfig.json +++ b/e2e/tsconfig.json @@ -3,5 +3,6 @@ "compilerOptions": { "rootDir": "./" }, - "include": ["./**/*"] + "include": ["./**/*"], + "references": [{"path": "../packages/test-utils"}] } diff --git a/packages/babel-jest/src/__tests__/tsconfig.json b/packages/babel-jest/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/babel-jest/src/__tests__/tsconfig.json +++ b/packages/babel-jest/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/babel-jest/tsconfig.json b/packages/babel-jest/tsconfig.json index a58a0880e8aa..3535b54c1d86 100644 --- a/packages/babel-jest/tsconfig.json +++ b/packages/babel-jest/tsconfig.json @@ -7,5 +7,5 @@ "include": ["./src/**/*"], "exclude": ["./**/__tests__/**/*"], // TODO: include `babel-preset-jest` if it's ever in TS even though we don't care about its types - "references": [{"path": "../jest-transform"}, {"path": "../test-utils"}] + "references": [{"path": "../jest-transform"}] } diff --git a/packages/expect/src/__tests__/tsconfig.json b/packages/expect/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/expect/src/__tests__/tsconfig.json +++ b/packages/expect/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/expect/tsconfig.json b/packages/expect/tsconfig.json index 330ff26d00c9..8d7a33c00c62 100644 --- a/packages/expect/tsconfig.json +++ b/packages/expect/tsconfig.json @@ -12,7 +12,6 @@ {"path": "../jest-get-type"}, {"path": "../jest-matcher-utils"}, {"path": "../jest-message-util"}, - {"path": "../jest-util"}, - {"path": "../test-utils"} + {"path": "../jest-util"} ] } diff --git a/packages/jest-console/src/__tests__/tsconfig.json b/packages/jest-console/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-console/src/__tests__/tsconfig.json +++ b/packages/jest-console/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-console/tsconfig.json b/packages/jest-console/tsconfig.json index 308e84ff26ee..735c691024a6 100644 --- a/packages/jest-console/tsconfig.json +++ b/packages/jest-console/tsconfig.json @@ -9,7 +9,6 @@ "references": [ {"path": "../jest-message-util"}, {"path": "../jest-types"}, - {"path": "../jest-util"}, - {"path": "../test-utils"} + {"path": "../jest-util"} ] } diff --git a/packages/jest-core/src/__tests__/tsconfig.json b/packages/jest-core/src/__tests__/tsconfig.json index 8ccc3083ffb0..d603a17f8dfe 100644 --- a/packages/jest-core/src/__tests__/tsconfig.json +++ b/packages/jest-core/src/__tests__/tsconfig.json @@ -1,7 +1,5 @@ { "extends": "../../../../tsconfig.test.json", - "compilerOptions": { - "rootDir": "../" - }, - "include": ["../**/*"] + "include": ["./**/*"], + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-core/src/lib/__tests__/tsconfig.json b/packages/jest-core/src/lib/__tests__/tsconfig.json index facea41430cf..1bc42294775a 100644 --- a/packages/jest-core/src/lib/__tests__/tsconfig.json +++ b/packages/jest-core/src/lib/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../../"}] + "references": [{"path": "../../../"}, {"path": "../../../../test-utils"}] } diff --git a/packages/jest-core/tsconfig.json b/packages/jest-core/tsconfig.json index 0a01ce3297dd..ac309da620f3 100644 --- a/packages/jest-core/tsconfig.json +++ b/packages/jest-core/tsconfig.json @@ -26,7 +26,6 @@ {"path": "../jest-util"}, {"path": "../jest-validate"}, {"path": "../jest-watcher"}, - {"path": "../pretty-format"}, - {"path": "../test-utils"} + {"path": "../pretty-format"} ] } diff --git a/packages/jest-diff/src/__tests__/tsconfig.json b/packages/jest-diff/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-diff/src/__tests__/tsconfig.json +++ b/packages/jest-diff/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-diff/tsconfig.json b/packages/jest-diff/tsconfig.json index e816b873f5ba..b286bcd0da96 100644 --- a/packages/jest-diff/tsconfig.json +++ b/packages/jest-diff/tsconfig.json @@ -9,7 +9,6 @@ "references": [ {"path": "../diff-sequences"}, {"path": "../jest-get-type"}, - {"path": "../pretty-format"}, - {"path": "../test-utils"} + {"path": "../pretty-format"} ] } diff --git a/packages/jest-environment-jsdom/src/__tests__/tsconfig.json b/packages/jest-environment-jsdom/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-environment-jsdom/src/__tests__/tsconfig.json +++ b/packages/jest-environment-jsdom/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-environment-jsdom/tsconfig.json b/packages/jest-environment-jsdom/tsconfig.json index 240880c13e8f..82e5be22db29 100644 --- a/packages/jest-environment-jsdom/tsconfig.json +++ b/packages/jest-environment-jsdom/tsconfig.json @@ -11,7 +11,6 @@ {"path": "../jest-fake-timers"}, {"path": "../jest-mock"}, {"path": "../jest-types"}, - {"path": "../jest-util"}, - {"path": "../test-utils"} + {"path": "../jest-util"} ] } diff --git a/packages/jest-environment-node/src/__tests__/tsconfig.json b/packages/jest-environment-node/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-environment-node/src/__tests__/tsconfig.json +++ b/packages/jest-environment-node/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-environment-node/tsconfig.json b/packages/jest-environment-node/tsconfig.json index 12bd3f484bcd..13971fdd663d 100644 --- a/packages/jest-environment-node/tsconfig.json +++ b/packages/jest-environment-node/tsconfig.json @@ -11,7 +11,6 @@ {"path": "../jest-fake-timers"}, {"path": "../jest-mock"}, {"path": "../jest-types"}, - {"path": "../jest-util"}, - {"path": "../test-utils"} + {"path": "../jest-util"} ] } diff --git a/packages/jest-fake-timers/src/__tests__/tsconfig.json b/packages/jest-fake-timers/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-fake-timers/src/__tests__/tsconfig.json +++ b/packages/jest-fake-timers/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-fake-timers/tsconfig.json b/packages/jest-fake-timers/tsconfig.json index 3c20004c2488..483142698671 100644 --- a/packages/jest-fake-timers/tsconfig.json +++ b/packages/jest-fake-timers/tsconfig.json @@ -11,7 +11,6 @@ {"path": "../jest-message-util"}, {"path": "../jest-mock"}, {"path": "../jest-types"}, - {"path": "../jest-util"}, - {"path": "../test-utils"} + {"path": "../jest-util"} ] } diff --git a/packages/jest-haste-map/package.json b/packages/jest-haste-map/package.json index d06359213809..fa4f5b8a7747 100644 --- a/packages/jest-haste-map/package.json +++ b/packages/jest-haste-map/package.json @@ -30,7 +30,6 @@ "walker": "^1.0.8" }, "devDependencies": { - "@jest/test-utils": "workspace:^", "@types/fb-watchman": "^2.0.0", "@types/micromatch": "^4.0.1", "slash": "^3.0.0" diff --git a/packages/jest-haste-map/tsconfig.json b/packages/jest-haste-map/tsconfig.json index 5fcbf9a63cd9..22b31f6d8eb1 100644 --- a/packages/jest-haste-map/tsconfig.json +++ b/packages/jest-haste-map/tsconfig.json @@ -10,7 +10,6 @@ {"path": "../jest-regex-util"}, {"path": "../jest-types"}, {"path": "../jest-util"}, - {"path": "../jest-worker"}, - {"path": "../test-utils"} + {"path": "../jest-worker"} ] } diff --git a/packages/jest-matcher-utils/src/__tests__/tsconfig.json b/packages/jest-matcher-utils/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-matcher-utils/src/__tests__/tsconfig.json +++ b/packages/jest-matcher-utils/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-matcher-utils/tsconfig.json b/packages/jest-matcher-utils/tsconfig.json index 3905821cea60..e59af10a490f 100644 --- a/packages/jest-matcher-utils/tsconfig.json +++ b/packages/jest-matcher-utils/tsconfig.json @@ -10,7 +10,6 @@ "references": [ {"path": "../jest-diff"}, {"path": "../jest-get-type"}, - {"path": "../pretty-format"}, - {"path": "../test-utils"} + {"path": "../pretty-format"} ] } diff --git a/packages/jest-reporters/src/__tests__/tsconfig.json b/packages/jest-reporters/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-reporters/src/__tests__/tsconfig.json +++ b/packages/jest-reporters/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-reporters/tsconfig.json b/packages/jest-reporters/tsconfig.json index 813ba257479e..86a51036fda6 100644 --- a/packages/jest-reporters/tsconfig.json +++ b/packages/jest-reporters/tsconfig.json @@ -14,7 +14,6 @@ {"path": "../jest-transform"}, {"path": "../jest-types"}, {"path": "../jest-util"}, - {"path": "../jest-worker"}, - {"path": "../test-utils"} + {"path": "../jest-worker"} ] } diff --git a/packages/jest-resolve-dependencies/src/__tests__/tsconfig.json b/packages/jest-resolve-dependencies/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-resolve-dependencies/src/__tests__/tsconfig.json +++ b/packages/jest-resolve-dependencies/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-resolve-dependencies/tsconfig.json b/packages/jest-resolve-dependencies/tsconfig.json index a1307345772a..503f05242ff2 100644 --- a/packages/jest-resolve-dependencies/tsconfig.json +++ b/packages/jest-resolve-dependencies/tsconfig.json @@ -12,7 +12,6 @@ {"path": "../jest-resolve"}, {"path": "../jest-runtime"}, {"path": "../jest-snapshot"}, - {"path": "../jest-types"}, - {"path": "../test-utils"} + {"path": "../jest-types"} ] } diff --git a/packages/jest-runner/package.json b/packages/jest-runner/package.json index c44fe957ced6..becf440e57f3 100644 --- a/packages/jest-runner/package.json +++ b/packages/jest-runner/package.json @@ -40,6 +40,7 @@ "source-map-support": "0.5.13" }, "devDependencies": { + "@jest/test-utils": "workspace:^", "@tsd/typescript": "^4.9.0", "@types/exit": "^0.1.30", "@types/graceful-fs": "^4.1.3", diff --git a/packages/jest-runner/src/__tests__/tsconfig.json b/packages/jest-runner/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-runner/src/__tests__/tsconfig.json +++ b/packages/jest-runner/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-runtime/src/__tests__/tsconfig.json b/packages/jest-runtime/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-runtime/src/__tests__/tsconfig.json +++ b/packages/jest-runtime/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-runtime/tsconfig.json b/packages/jest-runtime/tsconfig.json index 90a3eb84bddd..e62786852942 100644 --- a/packages/jest-runtime/tsconfig.json +++ b/packages/jest-runtime/tsconfig.json @@ -23,7 +23,6 @@ {"path": "../jest-test-result"}, {"path": "../jest-transform"}, {"path": "../jest-types"}, - {"path": "../jest-util"}, - {"path": "../test-utils"} + {"path": "../jest-util"} ] } diff --git a/packages/jest-snapshot/src/__tests__/tsconfig.json b/packages/jest-snapshot/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-snapshot/src/__tests__/tsconfig.json +++ b/packages/jest-snapshot/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-snapshot/tsconfig.json b/packages/jest-snapshot/tsconfig.json index 411a0f5a3a83..c9d79155bd00 100644 --- a/packages/jest-snapshot/tsconfig.json +++ b/packages/jest-snapshot/tsconfig.json @@ -17,7 +17,6 @@ {"path": "../jest-transform"}, {"path": "../jest-types"}, {"path": "../jest-util"}, - {"path": "../pretty-format"}, - {"path": "../test-utils"} + {"path": "../pretty-format"} ] } diff --git a/packages/jest-test-sequencer/src/__tests__/tsconfig.json b/packages/jest-test-sequencer/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-test-sequencer/src/__tests__/tsconfig.json +++ b/packages/jest-test-sequencer/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-test-sequencer/tsconfig.json b/packages/jest-test-sequencer/tsconfig.json index 7d3736031d3b..cdf9b3d35d19 100644 --- a/packages/jest-test-sequencer/tsconfig.json +++ b/packages/jest-test-sequencer/tsconfig.json @@ -6,9 +6,5 @@ }, "include": ["./src/**/*"], "exclude": ["./**/__tests__/**/*"], - "references": [ - {"path": "../jest-haste-map"}, - {"path": "../jest-test-result"}, - {"path": "../test-utils"} - ] + "references": [{"path": "../jest-haste-map"}, {"path": "../jest-test-result"}] } diff --git a/packages/jest-transform/src/__tests__/tsconfig.json b/packages/jest-transform/src/__tests__/tsconfig.json index dd1bca103251..d603a17f8dfe 100644 --- a/packages/jest-transform/src/__tests__/tsconfig.json +++ b/packages/jest-transform/src/__tests__/tsconfig.json @@ -1,5 +1,5 @@ { "extends": "../../../../tsconfig.test.json", "include": ["./**/*"], - "references": [{"path": "../../"}] + "references": [{"path": "../../"}, {"path": "../../../test-utils"}] } diff --git a/packages/jest-transform/tsconfig.json b/packages/jest-transform/tsconfig.json index b663b58b48db..cd4148bc6954 100644 --- a/packages/jest-transform/tsconfig.json +++ b/packages/jest-transform/tsconfig.json @@ -10,7 +10,6 @@ {"path": "../jest-haste-map"}, {"path": "../jest-regex-util"}, {"path": "../jest-types"}, - {"path": "../jest-util"}, - {"path": "../test-utils"} + {"path": "../jest-util"} ] } diff --git a/packages/test-utils/package.json b/packages/test-utils/package.json index b064dd1ef7c8..4c090092b54e 100644 --- a/packages/test-utils/package.json +++ b/packages/test-utils/package.json @@ -12,6 +12,7 @@ "./package.json": "./package.json" }, "dependencies": { + "@jest/globals": "workspace:^", "@jest/types": "workspace:^", "@types/node": "*", "ansi-regex": "^5.0.1", diff --git a/packages/test-utils/src/ConditionalTest.ts b/packages/test-utils/src/ConditionalTest.ts index badcb0c10ada..879c95bf06fd 100644 --- a/packages/test-utils/src/ConditionalTest.ts +++ b/packages/test-utils/src/ConditionalTest.ts @@ -5,11 +5,10 @@ * LICENSE file in the root directory of this source tree. */ -import semver = require('semver'); -import type {Global} from '@jest/types'; +/* eslint-disable jest/no-focused-tests */ -declare const describe: Global.TestFrameworkGlobals['describe']; -declare const test: Global.TestFrameworkGlobals['test']; +import semver = require('semver'); +import {describe, test} from '@jest/globals'; export function isJestJasmineRun(): boolean { return process.env.JEST_JASMINE === '1'; diff --git a/packages/test-utils/tsconfig.json b/packages/test-utils/tsconfig.json index 110097eb7fef..bcc1e9b530ae 100644 --- a/packages/test-utils/tsconfig.json +++ b/packages/test-utils/tsconfig.json @@ -5,5 +5,9 @@ "outDir": "build" }, "include": ["./src/**/*"], - "references": [{"path": "../jest-types"}, {"path": "../pretty-format"}] + "references": [ + {"path": "../jest-globals"}, + {"path": "../jest-types"}, + {"path": "../pretty-format"} + ] } diff --git a/scripts/buildTs.mjs b/scripts/buildTs.mjs index 72dcdab405d9..29f50731bd6b 100644 --- a/scripts/buildTs.mjs +++ b/scripts/buildTs.mjs @@ -57,6 +57,12 @@ packagesWithTs.forEach(({packageDir, pkg}) => { } } + // only test files depend on '@jest/test-utils', i.e. it is always a dev dependency + // see additional checks below + if (dep === '@jest/test-utils') { + return false; + } + return true; }) .map(dep => @@ -84,6 +90,60 @@ packagesWithTs.forEach(({packageDir, pkg}) => { )}\nExpected:\n\n${jestDependenciesOfPackage.join('\n')}`, ); } + + let hasJestTestUtils = Object.keys(pkg.dependencies || {}).includes( + '@jest/test-utils', + ); + + if (hasJestTestUtils) { + throw new Error( + chalk.red( + `Package '${pkg.name}' declares '@jest/test-utils' as dependency, but it must be declared as dev dependency`, + ), + ); + } + + hasJestTestUtils = Object.keys(pkg.devDependencies || {}).includes( + '@jest/test-utils', + ); + + const tsConfigPaths = glob.sync('**/__tests__/tsconfig.json', { + absolute: true, + cwd: packageDir, + }); + + const testUtilsReferences = tsConfigPaths.filter(tsConfigPath => { + const tsConfig = JSON.parse( + stripJsonComments(fs.readFileSync(tsConfigPath, 'utf8')), + ); + const references = tsConfig.references.map(({path}) => path); + + return references.some(reference => /test-utils$/.test(reference)); + }); + + if (hasJestTestUtils && testUtilsReferences.length === 0) { + throw new Error( + chalk.red( + `Package '${ + pkg.name + }' declares '@jest/test-utils' as dev dependency, but it is not referenced in:\n\n${tsConfigPaths.join( + '\n', + )}`, + ), + ); + } + + if (!hasJestTestUtils && testUtilsReferences.length > 0) { + throw new Error( + chalk.red( + `Package '${ + pkg.name + }' does not declare '@jest/test-utils' as dev dependency, but it is referenced in:\n\n${testUtilsReferences.join( + '\n', + )}`, + ), + ); + } }); const args = [ diff --git a/yarn.lock b/yarn.lock index 6621ce616bd1..b25ad34476ff 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2970,6 +2970,7 @@ __metadata: version: 0.0.0-use.local resolution: "@jest/test-utils@workspace:packages/test-utils" dependencies: + "@jest/globals": "workspace:^" "@jest/types": "workspace:^" "@types/node": "*" "@types/semver": ^7.1.0 @@ -12768,7 +12769,6 @@ __metadata: version: 0.0.0-use.local resolution: "jest-haste-map@workspace:packages/jest-haste-map" dependencies: - "@jest/test-utils": "workspace:^" "@jest/types": "workspace:^" "@types/fb-watchman": ^2.0.0 "@types/graceful-fs": ^4.1.3 @@ -12996,6 +12996,7 @@ __metadata: "@jest/console": "workspace:^" "@jest/environment": "workspace:^" "@jest/test-result": "workspace:^" + "@jest/test-utils": "workspace:^" "@jest/transform": "workspace:^" "@jest/types": "workspace:^" "@tsd/typescript": ^4.9.0 From f93ddb69547afa9b06d488429516fa8ca43eedfa Mon Sep 17 00:00:00 2001 From: Tom Mrazauskas Date: Sun, 5 Feb 2023 13:24:46 +0200 Subject: [PATCH 3/8] docs: remove extraneous tab symbol (#13861) --- docs/Webpack.md | 2 +- website/versioned_docs/version-25.x/Webpack.md | 2 +- website/versioned_docs/version-26.x/Webpack.md | 2 +- website/versioned_docs/version-27.x/Webpack.md | 2 +- website/versioned_docs/version-28.x/Webpack.md | 2 +- website/versioned_docs/version-29.0/Webpack.md | 2 +- website/versioned_docs/version-29.1/Webpack.md | 2 +- website/versioned_docs/version-29.2/Webpack.md | 2 +- website/versioned_docs/version-29.3/Webpack.md | 2 +- website/versioned_docs/version-29.4/Webpack.md | 2 +- 10 files changed, 10 insertions(+), 10 deletions(-) diff --git a/docs/Webpack.md b/docs/Webpack.md index 84b8944b7c4f..699ef4c9f805 100644 --- a/docs/Webpack.md +++ b/docs/Webpack.md @@ -202,7 +202,7 @@ For more complex webpack configurations, you may also want to investigate projec In addition to installing `babel-jest` as described earlier, you'll need to add `@babel/preset-env` like so: ```bash npm2yarn - npm install --save-dev @babel/preset-env +npm install --save-dev @babel/preset-env ``` Then, you'll want to configure Babel as follows: diff --git a/website/versioned_docs/version-25.x/Webpack.md b/website/versioned_docs/version-25.x/Webpack.md index 95a7e0676a32..df81d1a0289f 100644 --- a/website/versioned_docs/version-25.x/Webpack.md +++ b/website/versioned_docs/version-25.x/Webpack.md @@ -201,7 +201,7 @@ For more complex webpack configurations, you may also want to investigate projec In addition to installing `babel-jest` as described earlier, you'll need to add `@babel/preset-env` like so: ```bash npm2yarn - npm install --save-dev @babel/preset-env +npm install --save-dev @babel/preset-env ``` Then, you'll want to configure Babel as follows: diff --git a/website/versioned_docs/version-26.x/Webpack.md b/website/versioned_docs/version-26.x/Webpack.md index 95a7e0676a32..df81d1a0289f 100644 --- a/website/versioned_docs/version-26.x/Webpack.md +++ b/website/versioned_docs/version-26.x/Webpack.md @@ -201,7 +201,7 @@ For more complex webpack configurations, you may also want to investigate projec In addition to installing `babel-jest` as described earlier, you'll need to add `@babel/preset-env` like so: ```bash npm2yarn - npm install --save-dev @babel/preset-env +npm install --save-dev @babel/preset-env ``` Then, you'll want to configure Babel as follows: diff --git a/website/versioned_docs/version-27.x/Webpack.md b/website/versioned_docs/version-27.x/Webpack.md index 95a7e0676a32..df81d1a0289f 100644 --- a/website/versioned_docs/version-27.x/Webpack.md +++ b/website/versioned_docs/version-27.x/Webpack.md @@ -201,7 +201,7 @@ For more complex webpack configurations, you may also want to investigate projec In addition to installing `babel-jest` as described earlier, you'll need to add `@babel/preset-env` like so: ```bash npm2yarn - npm install --save-dev @babel/preset-env +npm install --save-dev @babel/preset-env ``` Then, you'll want to configure Babel as follows: diff --git a/website/versioned_docs/version-28.x/Webpack.md b/website/versioned_docs/version-28.x/Webpack.md index 95a7e0676a32..df81d1a0289f 100644 --- a/website/versioned_docs/version-28.x/Webpack.md +++ b/website/versioned_docs/version-28.x/Webpack.md @@ -201,7 +201,7 @@ For more complex webpack configurations, you may also want to investigate projec In addition to installing `babel-jest` as described earlier, you'll need to add `@babel/preset-env` like so: ```bash npm2yarn - npm install --save-dev @babel/preset-env +npm install --save-dev @babel/preset-env ``` Then, you'll want to configure Babel as follows: diff --git a/website/versioned_docs/version-29.0/Webpack.md b/website/versioned_docs/version-29.0/Webpack.md index 95a7e0676a32..df81d1a0289f 100644 --- a/website/versioned_docs/version-29.0/Webpack.md +++ b/website/versioned_docs/version-29.0/Webpack.md @@ -201,7 +201,7 @@ For more complex webpack configurations, you may also want to investigate projec In addition to installing `babel-jest` as described earlier, you'll need to add `@babel/preset-env` like so: ```bash npm2yarn - npm install --save-dev @babel/preset-env +npm install --save-dev @babel/preset-env ``` Then, you'll want to configure Babel as follows: diff --git a/website/versioned_docs/version-29.1/Webpack.md b/website/versioned_docs/version-29.1/Webpack.md index 95a7e0676a32..df81d1a0289f 100644 --- a/website/versioned_docs/version-29.1/Webpack.md +++ b/website/versioned_docs/version-29.1/Webpack.md @@ -201,7 +201,7 @@ For more complex webpack configurations, you may also want to investigate projec In addition to installing `babel-jest` as described earlier, you'll need to add `@babel/preset-env` like so: ```bash npm2yarn - npm install --save-dev @babel/preset-env +npm install --save-dev @babel/preset-env ``` Then, you'll want to configure Babel as follows: diff --git a/website/versioned_docs/version-29.2/Webpack.md b/website/versioned_docs/version-29.2/Webpack.md index 95a7e0676a32..df81d1a0289f 100644 --- a/website/versioned_docs/version-29.2/Webpack.md +++ b/website/versioned_docs/version-29.2/Webpack.md @@ -201,7 +201,7 @@ For more complex webpack configurations, you may also want to investigate projec In addition to installing `babel-jest` as described earlier, you'll need to add `@babel/preset-env` like so: ```bash npm2yarn - npm install --save-dev @babel/preset-env +npm install --save-dev @babel/preset-env ``` Then, you'll want to configure Babel as follows: diff --git a/website/versioned_docs/version-29.3/Webpack.md b/website/versioned_docs/version-29.3/Webpack.md index 84b8944b7c4f..699ef4c9f805 100644 --- a/website/versioned_docs/version-29.3/Webpack.md +++ b/website/versioned_docs/version-29.3/Webpack.md @@ -202,7 +202,7 @@ For more complex webpack configurations, you may also want to investigate projec In addition to installing `babel-jest` as described earlier, you'll need to add `@babel/preset-env` like so: ```bash npm2yarn - npm install --save-dev @babel/preset-env +npm install --save-dev @babel/preset-env ``` Then, you'll want to configure Babel as follows: diff --git a/website/versioned_docs/version-29.4/Webpack.md b/website/versioned_docs/version-29.4/Webpack.md index 84b8944b7c4f..699ef4c9f805 100644 --- a/website/versioned_docs/version-29.4/Webpack.md +++ b/website/versioned_docs/version-29.4/Webpack.md @@ -202,7 +202,7 @@ For more complex webpack configurations, you may also want to investigate projec In addition to installing `babel-jest` as described earlier, you'll need to add `@babel/preset-env` like so: ```bash npm2yarn - npm install --save-dev @babel/preset-env +npm install --save-dev @babel/preset-env ``` Then, you'll want to configure Babel as follows: From cfa2d39815e8b63555bfcac843e82aba5229aa50 Mon Sep 17 00:00:00 2001 From: Tom Mrazauskas Date: Sun, 5 Feb 2023 13:25:13 +0200 Subject: [PATCH 4/8] docs: fix link formatting in Expect API page (#13860) --- docs/ExpectAPI.md | 2 +- website/versioned_docs/version-29.4/ExpectAPI.md | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/docs/ExpectAPI.md b/docs/ExpectAPI.md index 87aa7c37b0c3..4f206f5cf8bb 100644 --- a/docs/ExpectAPI.md +++ b/docs/ExpectAPI.md @@ -1262,7 +1262,7 @@ These helper functions and properties can be found on `this` inside a custom tes #### `this.equals(a, b, customTesters?)` -This is a deep-equality function that will return `true` if two objects have the same values (recursively). It optionally takes a list of custom equality testers to apply to the deep equality checks. If you use this function, pass through the custom testers your tester is given so further equality checks `equals` applies can also use custom testers the test author may have configured. See the example in the [Recursive custom equality testers][#recursivecustomequalitytesters] section for more details. +This is a deep-equality function that will return `true` if two objects have the same values (recursively). It optionally takes a list of custom equality testers to apply to the deep equality checks. If you use this function, pass through the custom testers your tester is given so further equality checks `equals` applies can also use custom testers the test author may have configured. See the example in the [Recursive custom equality testers](#recursive-custom-equality-testers) section for more details. #### Matchers vs Testers diff --git a/website/versioned_docs/version-29.4/ExpectAPI.md b/website/versioned_docs/version-29.4/ExpectAPI.md index 87aa7c37b0c3..4f206f5cf8bb 100644 --- a/website/versioned_docs/version-29.4/ExpectAPI.md +++ b/website/versioned_docs/version-29.4/ExpectAPI.md @@ -1262,7 +1262,7 @@ These helper functions and properties can be found on `this` inside a custom tes #### `this.equals(a, b, customTesters?)` -This is a deep-equality function that will return `true` if two objects have the same values (recursively). It optionally takes a list of custom equality testers to apply to the deep equality checks. If you use this function, pass through the custom testers your tester is given so further equality checks `equals` applies can also use custom testers the test author may have configured. See the example in the [Recursive custom equality testers][#recursivecustomequalitytesters] section for more details. +This is a deep-equality function that will return `true` if two objects have the same values (recursively). It optionally takes a list of custom equality testers to apply to the deep equality checks. If you use this function, pass through the custom testers your tester is given so further equality checks `equals` applies can also use custom testers the test author may have configured. See the example in the [Recursive custom equality testers](#recursive-custom-equality-testers) section for more details. #### Matchers vs Testers From 423142cfafde90b230616f786dd652060908bc9f Mon Sep 17 00:00:00 2001 From: Tianlan Zhou Date: Sun, 5 Feb 2023 21:11:49 +0800 Subject: [PATCH 5/8] chore: Cleanup incorrect links in CHANGELOG.md (#13857) --- .github/workflows/nodejs.yml | 2 ++ CHANGELOG.md | 27 ++++++++++++++------------- package.json | 1 + scripts/checkChangelog.mjs | 31 +++++++++++++++++++++++++++++++ 4 files changed, 48 insertions(+), 13 deletions(-) create mode 100644 scripts/checkChangelog.mjs diff --git a/.github/workflows/nodejs.yml b/.github/workflows/nodejs.yml index b9111207f03a..39abe409fe23 100644 --- a/.github/workflows/nodejs.yml +++ b/.github/workflows/nodejs.yml @@ -82,6 +82,8 @@ jobs: run: yarn lint - name: run prettier run: yarn lint:prettier:ci + - name: check changelog + run: yarn check-changelog - name: check copyright headers run: yarn check-copyright-headers diff --git a/CHANGELOG.md b/CHANGELOG.md index 2fa90977d331..ff0bd2ebfba1 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,6 +11,7 @@ ### Chore & Maintenance - `[*]` make sure to exclude `.eslintcache` from published module ([#13832](https://github.com/facebook/jest/pull/13832)) +- `[docs]` Cleanup incorrect links in CHANGELOG.md ([#13857](https://github.com/facebook/jest/pull/13857)) ### Performance @@ -294,7 +295,7 @@ ### Fixes -- `[@jest/expect-utils]` Fix deep equality of ImmutableJS OrderedMaps ([#12763](https://github.com/facebook/jest/pull/12899)) +- `[@jest/expect-utils]` Fix deep equality of ImmutableJS OrderedMaps ([#12899](https://github.com/facebook/jest/pull/12899)) - `[jest-docblock]` Handle multiline comments in parseWithComments ([#12845](https://github.com/facebook/jest/pull/12845)) - `[jest-mock]` Improve `spyOn` error messages ([#12901](https://github.com/facebook/jest/pull/12901)) - `[jest-runtime]` Correctly report V8 coverage with `resetModules: true` ([#12912](https://github.com/facebook/jest/pull/12912)) @@ -772,7 +773,7 @@ ### Features -- `[@jest/fake-timers]` Flush callbacks scheduled with `requestAnimationFrame` every 16ms when using legacy timers. ([#11523](https://github.com/facebook/jest/pull/11567)) +- `[@jest/fake-timers]` Flush callbacks scheduled with `requestAnimationFrame` every 16ms when using legacy timers. ([#11567](https://github.com/facebook/jest/pull/11567)) - `[pretty-format]` Use `globalThis` (with polyfill if required) to bring support for esbuild's browser bundling mode ([#11569](https://github.com/facebook/jest/pull/11569)) ### Fixes @@ -859,7 +860,7 @@ - `[jest-runtime]` Detect reexports from CJS as named exports in ESM ([#10988](https://github.com/facebook/jest/pull/10988)) - `[jest-runtime]` Support for async code transformations ([#11191](https://github.com/facebook/jest/pull/11191) & [#11220](https://github.com/facebook/jest/pull/11220)) - `[jest-snapshot]` [**BREAKING**] Make prettier optional for inline snapshots - fall back to string replacement ([#7792](https://github.com/facebook/jest/pull/7792) & [#11192](https://github.com/facebook/jest/pull/11192)) -- `[jest-snapshot]` [**BREAKING**] Run transforms over `snapshotResolver` ([#8751](https://github.com/facebook/jest/pull/8829)) +- `[jest-snapshot]` [**BREAKING**] Run transforms over `snapshotResolver` ([#8829](https://github.com/facebook/jest/pull/8829)) - `[jest-transform]` Pass config options defined in Jest's config to transformer's `process` and `getCacheKey` functions ([#10926](https://github.com/facebook/jest/pull/10926)) - `[jest-transform]` Add support for transformers written in ESM ([#11163](https://github.com/facebook/jest/pull/11163)) - `[jest-transform]` [**BREAKING**] Do not export `ScriptTransformer` class, instead export the async function `createScriptTransformer` ([#11163](https://github.com/facebook/jest/pull/11163)) @@ -1107,7 +1108,7 @@ - `[jest-core]` Don't report ELDHistogram as open handle ([#10417](https://github.com/facebook/jest/pull/10417)) - `[jest-matcher-utils]` Fix diffing object contain readonly symbol key object ([#10414](https://github.com/facebook/jest/pull/10414)) -- `[jest-reporters]` Fixes notify reporter on Linux (using notify-send) ([#10393](https://github.com/facebook/jest/pull/10400)) +- `[jest-reporters]` Fixes notify reporter on Linux (using notify-send) ([#10400](https://github.com/facebook/jest/pull/10400)) - `[jest-snapshot]` Correctly handles arrays and property matchers in snapshots ([#10404](https://github.com/facebook/jest/pull/10404)) ## 26.4.0 @@ -1156,7 +1157,7 @@ - `[jest-haste-map]` Watchman crawler now includes dotfiles ([#10075](https://github.com/facebook/jest/pull/10075)) - `[jest-worker]` Added support for workers to send custom messages to parent in jest-worker ([#10293](https://github.com/facebook/jest/pull/10293)) - `[jest-worker]` Support passing `resourceLimits` ([#10335](https://github.com/facebook/jest/pull/10335)) -- `[pretty-format]` Added support for serializing custom elements (web components) ([#10217](https://github.com/facebook/jest/pull/10237)) +- `[pretty-format]` Added support for serializing custom elements (web components) ([#10237](https://github.com/facebook/jest/pull/10237)) ### Fixes @@ -1573,7 +1574,7 @@ - `[expect]` Display equal values for ReturnedWith similar to CalledWith ([#8791](https://github.com/facebook/jest/pull/8791)) - `[expect, jest-snapshot]` Change color from green for some args in matcher hints ([#8812](https://github.com/facebook/jest/pull/8812)) - `[jest-snapshot]` Highlight substring differences when matcher fails, part 3 ([#8569](https://github.com/facebook/jest/pull/8569)) -- `[jest-core]` Improve report when snapshots are obsolete ([#8448](https://github.com/facebook/jest/pull/8665)) +- `[jest-core]` Improve report when snapshots are obsolete ([#8665](https://github.com/facebook/jest/pull/8665)) - `[jest-cli]` Improve chai support (with detailed output, to match jest exceptions) ([#8454](https://github.com/facebook/jest/pull/8454)) - `[*]` Manage the global timeout with `--testTimeout` command line argument. ([#8456](https://github.com/facebook/jest/pull/8456)) - `[pretty-format]` Render custom displayName of memoized components ([#8546](https://github.com/facebook/jest/pull/8546)) @@ -1862,7 +1863,7 @@ We skipped 24.2.0 because a draft was accidentally published. Please use `24.3.0 - `[jest-docblock]`: Migrate to TypeScript ([#7836](https://github.com/facebook/jest/pull/7836)) - `[jest-each]`: Migrate to Typescript ([#8007](https://github.com/facebook/jest/pull/8007)) - `[jest-each]`: Refactor into multiple files with better types ([#8018](https://github.com/facebook/jest/pull/8018)) -- `[jest-environment-jsdom]`: Migrate to TypeScript ([#7985](https://github.com/facebook/jest/pull/8003)) +- `[jest-environment-jsdom]`: Migrate to TypeScript ([#8003](https://github.com/facebook/jest/pull/8003)) - `[jest-environment-node]`: Migrate to TypeScript ([#7985](https://github.com/facebook/jest/pull/7985)) - `[jest-get-type]`: Migrate to TypeScript ([#7818](https://github.com/facebook/jest/pull/7818)) - `[jest-haste-map]`: Migrate to TypeScript ([#7854](https://github.com/facebook/jest/pull/7854), [#7951](https://github.com/facebook/jest/pull/7951)) @@ -1885,7 +1886,7 @@ We skipped 24.2.0 because a draft was accidentally published. Please use `24.3.0 - `[jest-watcher]`: Migrate to TypeScript ([#7843](https://github.com/facebook/jest/pull/7843)) - `[jest-worker]`: Migrate to TypeScript ([#7853](https://github.com/facebook/jest/pull/7853)) - `[jest]`: Migrate to TypeScript ([#8024](https://github.com/facebook/jest/pull/8024)) -- `[pretty-format]`: Migrate to TypeScript ([#7809](https://github.com/facebook/jest/pull/7809), [#7809](https://github.com/facebook/jest/pull/7972)) +- `[pretty-format]`: Migrate to TypeScript ([#7809](https://github.com/facebook/jest/pull/7809), [#7972](https://github.com/facebook/jest/pull/7972)) ### Performance @@ -2179,7 +2180,7 @@ We skipped 24.2.0 because a draft was accidentally published. Please use `24.3.0 - `[jest-jasmine2]` Use prettier through `require` instead of `localRequire`. Fixes `matchInlineSnapshot` where prettier dependencies like `path` and `fs` are mocked with `jest.mock`. ([#6776](https://github.com/facebook/jest/pull/6776)) - `[docs]` Fix contributors link ([#6711](https://github.com/facebook/jest/pull/6711)) - `[website]` Fix website versions page to link to correct language ([#6734](https://github.com/facebook/jest/pull/6734)) -- `[expect]` Update `toContain` suggestion to contain equal message ([#6792](https://github.com/facebook/jest/pull/6810)) +- `[expect]` Update `toContain` suggestion to contain equal message ([#6810](https://github.com/facebook/jest/pull/6810)) ## 23.4.1 @@ -2320,7 +2321,7 @@ We skipped 24.2.0 because a draft was accidentally published. Please use `24.3.0 - `[jest-runtime]` Prevent modules from marking themselves as their own parent ([#5235](https://github.com/facebook/jest/issues/5235)) - `[jest-mock]` Add support for auto-mocking generator functions ([#5983](https://github.com/facebook/jest/pull/5983)) - `[expect]` Add support for async matchers ([#5919](https://github.com/facebook/jest/pull/5919)) -- `[expect]` Suggest toContainEqual ([#5948](https://github.com/facebook/jest/pull/5953)) +- `[expect]` Suggest toContainEqual ([#5953](https://github.com/facebook/jest/pull/5953)) - `[jest-config]` Export Jest's default options ([#5948](https://github.com/facebook/jest/pull/5948)) - `[jest-editor-support]` Move `coverage` to `ProjectWorkspace.collectCoverage` ([#5929](https://github.com/facebook/jest/pull/5929)) - `[jest-editor-support]` Add `coverage` option to runner ([#5836](https://github.com/facebook/jest/pull/5836)) @@ -2398,7 +2399,7 @@ We skipped 24.2.0 because a draft was accidentally published. Please use `24.3.0 - `[docs]` Add explanation on how to mock methods not implemented in JSDOM - `[jest-jasmine2]` Simplify `Env.execute` and TreeProcessor to setup and clean resources for the top suite the same way as for all of the children suites ([#5885](https://github.com/facebook/jest/pull/5885)) - `[babel-jest]` [**BREAKING**] Always return object from transformer ([#5991](https://github.com/facebook/jest/pull/5991)) -- `[*]` Run Prettier on compiled output ([#5858](https://github.com/facebook/jest/pull/3497)) +- `[*]` Run Prettier on compiled output ([#5858](https://github.com/facebook/jest/pull/5858)) - `[jest-cli]` Add fileChange hook for plugins ([#5708](https://github.com/facebook/jest/pull/5708)) - `[docs]` Add docs on using `jest.mock(...)` ([#5648](https://github.com/facebook/jest/pull/5648)) - `[docs]` Mention Jest Puppeteer Preset ([#5722](https://github.com/facebook/jest/pull/5722)) @@ -2484,7 +2485,7 @@ We skipped 24.2.0 because a draft was accidentally published. Please use `24.3.0 ### Fixes - `[babel-jest]` Revert "Remove retainLines from babel-jest" ([#5496](https://github.com/facebook/jest/pull/5496)) -- `[jest-docblock]` Support multiple of the same `@pragma`. ([#5154](https://github.com/facebook/jest/pull/5502)) +- `[jest-docblock]` Support multiple of the same `@pragma`. ([#5502](https://github.com/facebook/jest/pull/5502)) ### Features @@ -2668,7 +2669,7 @@ We skipped 24.2.0 because a draft was accidentally published. Please use `24.3.0 - `[pretty-format]` Fix errors when identity-obj-proxy mocks CSS Modules ([#4935](https://github.com/facebook/jest/pull/4935)) - `[babel-jest]` Fix support for namespaced babel version 7 ([#4918](https://github.com/facebook/jest/pull/4918)) - `[expect]` fix .toThrow for promises ([#4884](https://github.com/facebook/jest/pull/4884)) -- `[jest-docblock]` pragmas should preserve urls ([#4837](https://github.com/facebook/jest/pull/4629)) +- `[jest-docblock]` pragmas should preserve urls ([#4837](https://github.com/facebook/jest/pull/4837)) - `[jest-cli]` Check if `npm_lifecycle_script` calls Jest directly ([#4629](https://github.com/facebook/jest/pull/4629)) - `[jest-cli]` Fix --showConfig to show all configs ([#4494](https://github.com/facebook/jest/pull/4494)) - `[jest-cli]` Throw if `maxWorkers` doesn't have a value ([#4591](https://github.com/facebook/jest/pull/4591)) diff --git a/package.json b/package.json index 8caee36a19c3..bf5568503c6e 100644 --- a/package.json +++ b/package.json @@ -89,6 +89,7 @@ "build:js": "node ./scripts/build.mjs", "build:ts": "node ./scripts/buildTs.mjs", "bundle:ts": "node ./scripts/bundleTs.mjs", + "check-changelog": "node ./scripts/checkChangelog.mjs", "check-copyright-headers": "node ./scripts/checkCopyrightHeaders.mjs", "clean-all": "yarn clean-e2e && yarn build-clean && rimraf './packages/*/node_modules' && rimraf './node_modules'", "clean-e2e": "node ./scripts/cleanE2e.mjs", diff --git a/scripts/checkChangelog.mjs b/scripts/checkChangelog.mjs new file mode 100644 index 000000000000..85cc6d0119b7 --- /dev/null +++ b/scripts/checkChangelog.mjs @@ -0,0 +1,31 @@ +/** + * Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */ + +import fs from 'graceful-fs'; + +const linkRegex = + /\[#(\d+)\]\(https:\/\/github.com\/facebook\/jest\/(issues|pull)\/(\d+)\)/g; + +const changelogPath = 'CHANGELOG.md'; +const data = fs.readFileSync(changelogPath, 'utf-8'); + +let error = false; +let lineNumber = 1; +for (const line of data.split('\n')) { + for (const match of line.matchAll(linkRegex)) + if (match[1] !== match[3]) { + const column = match.index + 1; + console.error( + `${changelogPath}:${lineNumber}:${column}: error: ` + + `Link is incorrect: ${match[0]}`, + ); + error = true; + } + ++lineNumber; +} + +process.exit(error ? 1 : 0); From c8e7577df435a292906e88b6d170dc610451326b Mon Sep 17 00:00:00 2001 From: Rob Hogan <2590098+robhogan@users.noreply.github.com> Date: Sun, 5 Feb 2023 20:15:34 +0000 Subject: [PATCH 6/8] feat(jest-core): Add performance markers around significant lifecycle events (#13859) --- CHANGELOG.md | 2 ++ packages/jest-core/src/cli/index.ts | 11 +++++++++++ packages/jest-core/src/runJest.ts | 16 ++++++++++++++++ 3 files changed, 29 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index ff0bd2ebfba1..6aa5f81fd386 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,8 @@ ### Features +- `[@jest/core]` Instrument significant lifecycle events with [`performance.mark()`](https://nodejs.org/docs/latest-v16.x/api/perf_hooks.html#performancemarkname-options) ([#13859](https://github.com/facebook/jest/pull/13859)) + ### Fixes - `[expect, @jest/expect]` Provide type of `actual` as a generic argument to `Matchers` to allow better-typed extensions ([#13848](https://github.com/facebook/jest/pull/13848)) diff --git a/packages/jest-core/src/cli/index.ts b/packages/jest-core/src/cli/index.ts index b3e7df8c5e12..b9ac488c3278 100644 --- a/packages/jest-core/src/cli/index.ts +++ b/packages/jest-core/src/cli/index.ts @@ -5,6 +5,7 @@ * LICENSE file in the root directory of this source tree. */ +import {performance} from 'perf_hooks'; import chalk = require('chalk'); import exit = require('exit'); import * as fs from 'graceful-fs'; @@ -41,6 +42,7 @@ export async function runCLI( results: AggregatedResult; globalConfig: Config.GlobalConfig; }> { + performance.mark('jest/runCLI:start'); let results: AggregatedResult | undefined; // If we output a JSON object, we can't write anything to stdout, since @@ -133,6 +135,7 @@ export async function runCLI( console.error(message); } + performance.mark('jest/runCLI:end'); return {globalConfig, results}; } @@ -173,6 +176,12 @@ const _run10000 = async ( // Queries to hg/git can take a while, so we need to start the process // as soon as possible, so by the time we need the result it's already there. const changedFilesPromise = getChangedFilesPromise(globalConfig, configs); + if (changedFilesPromise) { + performance.mark('jest/getChangedFiles:start'); + changedFilesPromise.finally(() => { + performance.mark('jest/getChangedFiles:end'); + }); + } // Filter may need to do an HTTP call or something similar to setup. // We will wait on an async response from this before using the filter. @@ -204,11 +213,13 @@ const _run10000 = async ( }; } + performance.mark('jest/buildContextsAndHasteMaps:start'); const {contexts, hasteMapInstances} = await buildContextsAndHasteMaps( configs, globalConfig, outputStream, ); + performance.mark('jest/buildContextsAndHasteMaps:end'); globalConfig.watch || globalConfig.watchAll ? await runWatch( diff --git a/packages/jest-core/src/runJest.ts b/packages/jest-core/src/runJest.ts index 9aa77a4346ec..640407a0f6b1 100644 --- a/packages/jest-core/src/runJest.ts +++ b/packages/jest-core/src/runJest.ts @@ -6,6 +6,7 @@ */ import * as path from 'path'; +import {performance} from 'perf_hooks'; import chalk = require('chalk'); import exit = require('exit'); import * as fs from 'graceful-fs'; @@ -179,6 +180,7 @@ export default async function runJest({ const searchSources = contexts.map(context => new SearchSource(context)); + performance.mark('jest/getTestPaths:start'); const testRunData: TestRunData = await Promise.all( contexts.map(async (context, index) => { const searchSource = searchSources[index]; @@ -195,6 +197,7 @@ export default async function runJest({ return {context, matches}; }), ); + performance.mark('jest/getTestPaths:end'); if (globalConfig.shard) { if (typeof sequencer.shard !== 'function') { @@ -260,7 +263,9 @@ export default async function runJest({ } if (hasTests) { + performance.mark('jest/globalSetup:start'); await runGlobalHook({allTests, globalConfig, moduleName: 'globalSetup'}); + performance.mark('jest/globalSetup:end'); } if (changedFilesPromise) { @@ -289,14 +294,24 @@ export default async function runJest({ ...testSchedulerContext, }); + // @ts-expect-error - second arg is unsupported (but harmless) in Node 14 + performance.mark('jest/scheduleAndRun:start', { + detail: {numTests: allTests.length}, + }); const results = await scheduler.scheduleTests(allTests, testWatcher); + performance.mark('jest/scheduleAndRun:start'); + performance.mark('jest/cacheResults:start'); sequencer.cacheResults(allTests, results); + performance.mark('jest/cacheResults:end'); if (hasTests) { + performance.mark('jest/globalTeardown:start'); await runGlobalHook({allTests, globalConfig, moduleName: 'globalTeardown'}); + performance.mark('jest/globalTeardown:end'); } + performance.mark('jest/processResults:start'); await processResults(results, { collectHandles, json: globalConfig.json, @@ -305,4 +320,5 @@ export default async function runJest({ outputStream, testResultsProcessor: globalConfig.testResultsProcessor, }); + performance.mark('jest/processResults:end'); } From 1a2ffe451e1ff0a8fa1d568510fe410833abe21f Mon Sep 17 00:00:00 2001 From: Tom Mrazauskas Date: Mon, 6 Feb 2023 01:40:26 +0200 Subject: [PATCH 7/8] docs: bring back missing images (#13863) --- website/static/img/logos/ibm.png | Bin 0 -> 8322 bytes website/static/img/logos/paypal.png | Bin 0 -> 10972 bytes website/static/img/logos/pinterest.png | Bin 0 -> 17121 bytes 3 files changed, 0 insertions(+), 0 deletions(-) create mode 100644 website/static/img/logos/ibm.png create mode 100644 website/static/img/logos/paypal.png create mode 100644 website/static/img/logos/pinterest.png diff --git a/website/static/img/logos/ibm.png b/website/static/img/logos/ibm.png new file mode 100644 index 0000000000000000000000000000000000000000..f8ddd46d8789a294c9a803d47034c23ed65550ef GIT binary patch literal 8322 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 Q`rl`DWo<~EqGkC10G+ANsQ>@~ literal 0 HcmV?d00001 diff --git a/website/static/img/logos/paypal.png b/website/static/img/logos/paypal.png new file mode 100644 index 0000000000000000000000000000000000000000..532b2be40f1aa52f858a354be40081ecedc37ca3 GIT binary patch literal 10972 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 HQON%Xjp@@P literal 0 HcmV?d00001 diff --git a/website/static/img/logos/pinterest.png b/website/static/img/logos/pinterest.png new file mode 100644 index 0000000000000000000000000000000000000000..a45c216811a3ddaed80b376bdcda9e178c5185f5 GIT binary patch literal 17121 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 Date: Mon, 6 Feb 2023 15:56:17 +0200 Subject: [PATCH 8/8] docs: clean up formatting of blog posts (#13862) * doc: clean up formatting of blog posts * admonitions * fix typo * prettier --- .eslintrc.cjs | 9 ++++++++ website/blog/2016-04-12-jest-11.md | 2 +- website/blog/2016-07-27-jest-14.md | 6 ++++- website/blog/2016-09-01-jest-15.md | 6 ++--- website/blog/2016-12-15-2016-in-jest.md | 22 +++++++++---------- ...e-watch-mode-test-platform-improvements.md | 4 ++-- ...delightful-testing-multi-project-runner.md | 2 +- .../2018-06-27-supporting-jest-open-source.md | 16 +++++++------- ...refreshing-polished-typescript-friendly.md | 6 ++++- website/blog/2022-08-25-jest-29.md | 2 +- 10 files changed, 46 insertions(+), 29 deletions(-) diff --git a/.eslintrc.cjs b/.eslintrc.cjs index c22fcecb4c0d..5a573f07e1cb 100644 --- a/.eslintrc.cjs +++ b/.eslintrc.cjs @@ -235,6 +235,15 @@ module.exports = { 'jest/prefer-to-have-length': 'off', }, }, + // snapshot in an example needs to keep escapes + { + files: [ + '**/2017-02-21-jest-19-immersive-watch-mode-test-platform-improvements.md/**', + ], + rules: { + 'no-useless-escape': 'off', + }, + }, // snapshots in examples plus inline snapshots need to keep backtick { diff --git a/website/blog/2016-04-12-jest-11.md b/website/blog/2016-04-12-jest-11.md index af68f07cbc87..230e5f32c7ac 100644 --- a/website/blog/2016-04-12-jest-11.md +++ b/website/blog/2016-04-12-jest-11.md @@ -9,7 +9,7 @@ Today we're announcing a switch to major revisions for Jest with Jest 11.0 being If you are using Jest 0.9 or Jest 0.10 the upgrade should be seamless. All changes from the last few months were rolled into Jest 11.0. -### New in Jest 11.0. +### New in Jest 11.0 #### Babel Integration and Simplified Setup diff --git a/website/blog/2016-07-27-jest-14.md b/website/blog/2016-07-27-jest-14.md index 79babd8fd57c..77ddd42b622b 100644 --- a/website/blog/2016-07-27-jest-14.md +++ b/website/blog/2016-07-27-jest-14.md @@ -63,7 +63,11 @@ You can start using Jest with react-native by running `yarn add --dev jest-react - [Example project](https://github.com/facebook/jest/tree/main/examples/react-native) - [Example pull request for _snowflake_](https://github.com/bartonhammond/snowflake/pull/110), a popular react-native open source library. -_Note: the preset currently only implements the minimal set of configuration necessary to get started with React Native testing. We are hoping for community contributions to improve this project. Please try it and file [issues](https://github.com/facebook/jest/issues) or send pull requests!_ +:::info + +The preset currently only implements the minimal set of configuration necessary to get started with React Native testing. We are hoping for community contributions to improve this project. Please try it and file [issues](https://github.com/facebook/jest/issues) or send pull requests! + +::: ## Why snapshot testing? diff --git a/website/blog/2016-09-01-jest-15.md b/website/blog/2016-09-01-jest-15.md index 0fc020d7aae6..e7afc6369dd7 100644 --- a/website/blog/2016-09-01-jest-15.md +++ b/website/blog/2016-09-01-jest-15.md @@ -27,7 +27,7 @@ We completely rewrote `jest --watch` to be more interactive. It can now switch b Mocks for `ListView`, `TextInput`, `ActivityIndicator`, `ScrollView` and more were added. The existing mocks were updated to use the real implementations and existing snapshots likely have to be updated when upgrading to Jest 15. A `mockComponent` function was added to `jest-react-native` that can be used to mock native components: -``` +```js jest.mock('MyNativeComponent', () => { const jestReactNative = require('jest-react-native'); return jestReactNative.mockComponent('MyNativeComponent'); @@ -60,12 +60,12 @@ This has lead to numerous incompatibilities. We also noticed that at Facebook we Here is an example: -``` +```js const React1 = require('react'); jest.resetModules(); const React2 = require('react'); -React1 !== React2 // These two are separate copies of React. +React1 !== React2; // These two are separate copies of React. ``` The call to `resetModules` wipes away the require cache. A second call to require the same module will result in a new instantiation of the same module and all of its dependencies. This feature is especially useful when dealing with modules that have side effects, like [jest-haste-map](https://github.com/facebook/jest/blob/3bbf32a239fc4aad8cc6928a787f235bd86fecac/packages/jest-haste-map/src/__tests__/index-test.js#L64). diff --git a/website/blog/2016-12-15-2016-in-jest.md b/website/blog/2016-12-15-2016-in-jest.md index 19942581ab08..307c7319930a 100644 --- a/website/blog/2016-12-15-2016-in-jest.md +++ b/website/blog/2016-12-15-2016-in-jest.md @@ -61,20 +61,20 @@ Jest was initially created more than five years ago and as such an old framework - **Breaking:** Replaced `scriptPreprocessor` with the new `transform` option. - **Breaking:** The `testResultsProcessor` function is now required to return the modified results. - **Potentially Breaking:** Properly resolve `snapshotSerializers`, `setupFiles`, `transform`, `testRunner` and `testResultsProcessor` with a resolution algorithm instead of using `path.resolve`. This mainly means that `` is no longer needed for these options. -- **Added: **`pretty-format` and `jest-editor-support` were merged into Jest. +- **Added:** `pretty-format` and `jest-editor-support` were merged into Jest. - **Added:** `expect.any`, `expect.anything`, `expect.objectContaining`, `expect.arrayContaining`, `expect.stringMatching`. -- **Added: **`--testResultsProcessor` is now exposed through the cli. +- **Added:** `--testResultsProcessor` is now exposed through the cli. - **Added:** Implemented file watching in `jest-haste-map`. - **Added:** Usage of Jest in watch mode can be hidden through `JEST_HIDE_USAGE`. - **Added:** `expect.assertions(number)` which will ensure that a specified amount of assertions is made in one test. -- **Added: **`.toMatchSnapshot(?string)` feature to give snapshots a name. -- **Added: **`toMatchObject`, `toHaveProperty` , `toHaveLength` matchers. +- **Added:** `.toMatchSnapshot(?string)` feature to give snapshots a name. +- **Added:** `toMatchObject`, `toHaveProperty` , `toHaveLength` matchers. - **Added:** `expect.extend`. - **Added:** Added support for custom snapshots serializers. - **Added:** Big diffs are now collapsed by default in snapshots and assertions. Added `--expand` (or `-e`) to show the full diff. - **Added:** `jest.resetAllMocks` which replaces `jest.clearAllMocks`. -- **Added: **`--json` now includes information about individual tests inside a file. -- **Fixed: **`test.concurrent` unhandled promise rejections. +- **Added:** `--json` now includes information about individual tests inside a file. +- **Fixed:** `test.concurrent` unhandled promise rejections. - **Fixed:** `babel-plugin-jest-hoist` when using `jest.mock` with three arguments. - **Fixed:** The `JSON` global in `jest-environment-node` now comes from the vm context instead of the parent context. - **Fixed:** Jest does not print stack traces from babel any longer. @@ -84,16 +84,16 @@ Jest was initially created more than five years ago and as such an old framework - **Fixed:** `NaN% Failed` in the OS notification when using `--notify`. - **Fixed:** The first test run without cached timings will now use separate processes instead of running in band. - **Fixed:** `Map`/`Set` comparisons. -- **Fixed: **`test.concurrent` now works with `--testNamePattern`. +- **Fixed:** `test.concurrent` now works with `--testNamePattern`. - **Fixed:** Improved `.toContain` matcher. - **Fixed:** Properly resolve modules with platform extensions on react-native. - **Fixed:** global built in objects in `jest-environment-node` now work properly. - **Fixed:** Create mock objects in the vm context instead of the parent context. -- **Fixed: **`.babelrc` is now part of the transform cache key in `babel-jest`. +- **Fixed:** `.babelrc` is now part of the transform cache key in `babel-jest`. - **Fixed:** docblock parsing with haste modules. - **Fixed:** Exit with the proper code when the coverage threshold is not reached. -- **Fixed: **Jest now clears the entire scrollback in watch mode. -- **Deprecated: **`jest-react-native` was deprecated and now forwards `react-native`. +- **Fixed:** Jest now clears the entire scrollback in watch mode. +- **Deprecated:** `jest-react-native` was deprecated and now forwards `react-native`. ## Plans for Jest in H1 2017 @@ -102,7 +102,7 @@ Six months ago [we shared our plans for Jest](/blog/2016/07/27/jest-14#what-s-ne - **Instant feedback:** [Nuclide](https://nuclide.io/) integration and an improved and [faster watch mode](https://github.com/facebook/jest/pull/2324#issuecomment-267149669). - **Improved developer experience:** new mocking APIs and improved assertions. - **Better performance and memory usage:** analyze Jest and be more conscious about efficiency. -- **Snapshot Improvements: **snapshot approval mode, syntax highlighting and improved `react-test-renderer` APIs. +- **Snapshot Improvements:** snapshot approval mode, syntax highlighting and improved `react-test-renderer` APIs. - **Website:** We'll overhaul the website and documentation and add a Jest cheat sheet. We won't be providing timelines or estimates for these features and we may not actually get to all of these things. If you'd like to help make these things a reality, send us issues and pull requests with your ideas and let's work on improving Jest together in 2017. diff --git a/website/blog/2017-02-21-jest-19-immersive-watch-mode-test-platform-improvements.md b/website/blog/2017-02-21-jest-19-immersive-watch-mode-test-platform-improvements.md index b35b32016f50..372b43bfe026 100644 --- a/website/blog/2017-02-21-jest-19-immersive-watch-mode-test-platform-improvements.md +++ b/website/blog/2017-02-21-jest-19-immersive-watch-mode-test-platform-improvements.md @@ -23,7 +23,7 @@ We made a couple of changes to the snapshot format. We don't make changes like t Before: -``` +```js exports[`test snap 1`] = `

@@ -37,7 +37,7 @@ exports[`test snap 1`] = ` After (no “test” prefix, better JSX rendering, version header): -``` +```js // Jest Snapshot v1, https://goo.gl/fbAQLP exports[`snap 1`] = ` diff --git a/website/blog/2017-05-06-jest-20-delightful-testing-multi-project-runner.md b/website/blog/2017-05-06-jest-20-delightful-testing-multi-project-runner.md index a17f26439b59..aa7ed8bbe447 100644 --- a/website/blog/2017-05-06-jest-20-delightful-testing-multi-project-runner.md +++ b/website/blog/2017-05-06-jest-20-delightful-testing-multi-project-runner.md @@ -19,7 +19,7 @@ Jest is now collapsing the usage guide after the first test run to save vertical Further, we completely overhauled how the configuration system works inside of Jest. You can now pass any configuration option through the CLI to overwrite the ones specified in your configuration file. Along with that, we changed Jest to look for a `jest.config.js` file by default which means you are now able to define a Jest configuration using JavaScript as well as being able to configure it through `package.json` like before. Through the addition of all these new features, you are now able to combine Jest in more powerful ways than ever before. For example, if you would like to find out which tests Jest would run given a set of changed files from a commit across multiple projects in a monorepo, you can combine cli arguments like this now: -``` +```bash $ jest --projects projectA projectB --listTests --findRelatedTests projectA/banana.js projectB/kiwi.js [ "projectA/banana.test.js", diff --git a/website/blog/2018-06-27-supporting-jest-open-source.md b/website/blog/2018-06-27-supporting-jest-open-source.md index c3ec31ad6adb..b15d24a171ea 100644 --- a/website/blog/2018-06-27-supporting-jest-open-source.md +++ b/website/blog/2018-06-27-supporting-jest-open-source.md @@ -17,13 +17,13 @@ In this post we'll outline what the Jest Open Collective is, the structure, and -# The Jest Open Collective +## The Jest Open Collective ![Banner image for Jest Open Collective](/img/blog/collective.png) Open Collective is a platform to manage groups of people transparently. Jest joins projects like [webpack](https://opencollective.com/webpack), [Babel](https://opencollective.com/babel), [Mocha](https://opencollective.com/mochajs), [Preact](https://opencollective.com/preact), [Vue](https://opencollective.com/vuejs), and many more in supporting the open source community through the Open Collective platform. -## What is the Jest Open Collective +### What is the Jest Open Collective The Jest Open Collective is a group of open source contributors who operate in full transparency to: @@ -31,7 +31,7 @@ The Jest Open Collective is a group of open source contributors who operate in f - Approve expenses submitted by the community - Give everyone visibility on the budget -## What is the structure of the collective +### What is the structure of the collective The Jest Open Collective is currently managed by three non-Facebook core contributors from the open source community: @@ -43,15 +43,15 @@ Michal has been an active Jest contributor since September 2016, Simen joined th There are two levels of support for the collective: Backer and Sponsor. -### Backers +#### Backers Backers of the collective are individuals contributing at least \$2/month. We'll include a list of backers on the Jest homepage, README on github/yarn/npm, and Contributors page. -### Sponsors +#### Sponsors -Sponsors of the collective are individuals and organizations contributing at least \$100/month. We'll place sponsor logos with a link to their site on the Jest homepage, README on github/yarn/npm, and Contributors page. +Sponsors of the collective are individuals and organizations contributing at least $100/month. We'll place sponsor logos with a link to their site on the Jest homepage, README on github/yarn/npm, and Contributors page. -## What is the goal of the collective +### What is the goal of the collective The goal of the collective is to support the work of open source contributors to Jest in order to make testing delightful. @@ -64,7 +64,7 @@ To achieve that goal, we will use the funds to: This is just the beginning and we're committed to getting this right. If you have ideas on how else we can support the community, or feedback on the structure of the collective, please reach out to us on [twitter](https://twitter.com/fbjest)! -# Thank You +## Thank You Finally, thank you to everyone who contributes to the Jest community and open source in general. We are incredibly grateful that we get the opportunity to work on improving JavaScript testing together. diff --git a/website/blog/2019-01-25-jest-24-refreshing-polished-typescript-friendly.md b/website/blog/2019-01-25-jest-24-refreshing-polished-typescript-friendly.md index 21ff728b565e..2083a34edebe 100644 --- a/website/blog/2019-01-25-jest-24-refreshing-polished-typescript-friendly.md +++ b/website/blog/2019-01-25-jest-24-refreshing-polished-typescript-friendly.md @@ -27,7 +27,11 @@ If you want to run typechecks while you test, you should use [`ts-jest`](https:/ See [the docs](/docs/getting-started#using-typescript) for more details. -_Note that if you for whatever reason cannot upgrade to Babel 7, you can still use Jest 24 with `babel@6` as long as you keep `babel-jest` at version 23._ +:::tip + +If you for whatever reason cannot upgrade to Babel 7, you can still use Jest 24 with `babel@6` as long as you keep `babel-jest` at version 23. + +::: ## `test.todo` diff --git a/website/blog/2022-08-25-jest-29.md b/website/blog/2022-08-25-jest-29.md index e8014f818388..9ab4001bc644 100644 --- a/website/blog/2022-08-25-jest-29.md +++ b/website/blog/2022-08-25-jest-29.md @@ -9,7 +9,7 @@ Jest 29 is here, just a few short months after Jest 28. As mentioned in the [Jes -The only breaking changes that should be noticable are: +The only breaking changes that should be noticeable are: - Node versions 12 and 17 are dropped, both of which are EOL - The `snapshotFormat` property is changed to: