Set up comprehensive frontend testing infrastructure
- Install Jest for unit testing with React Testing Library - Install Playwright for end-to-end testing - Configure Jest with proper TypeScript support and module mapping - Create test setup files and utilities for both unit and e2e tests Components: * Jest configuration with coverage thresholds * Playwright configuration with browser automation * Unit tests for LoginForm, AuthContext, and useSocketIO hook * E2E tests for authentication, dashboard, and agents workflows * GitHub Actions workflow for automated testing * Mock data and API utilities for consistent testing * Test documentation with best practices Testing features: - Unit tests with 70% coverage threshold - E2E tests with API mocking and user journey testing - CI/CD integration for automated test runs - Cross-browser testing support with Playwright - Authentication system testing end-to-end 🚀 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
81
frontend/node_modules/@jest/test-sequencer/build/index.d.mts
generated
vendored
Normal file
81
frontend/node_modules/@jest/test-sequencer/build/index.d.mts
generated
vendored
Normal file
@@ -0,0 +1,81 @@
|
||||
import { AggregatedResult, Test, TestContext } from "@jest/test-result";
|
||||
import { Config } from "@jest/types";
|
||||
|
||||
//#region src/index.d.ts
|
||||
|
||||
declare const FAIL = 0;
|
||||
declare const SUCCESS = 1;
|
||||
type TestSequencerOptions = {
|
||||
contexts: ReadonlyArray<TestContext>;
|
||||
globalConfig: Config.GlobalConfig;
|
||||
};
|
||||
type Cache = {
|
||||
[key: string]: [testStatus: typeof FAIL | typeof SUCCESS, testDuration: number] | undefined;
|
||||
};
|
||||
type ShardOptions = {
|
||||
shardIndex: number;
|
||||
shardCount: number;
|
||||
};
|
||||
/**
|
||||
* The TestSequencer will ultimately decide which tests should run first.
|
||||
* It is responsible for storing and reading from a local cache
|
||||
* map that stores context information for a given test, such as how long it
|
||||
* took to run during the last run and if it has failed or not.
|
||||
* Such information is used on:
|
||||
* TestSequencer.sort(tests: Array<Test>)
|
||||
* to sort the order of the provided tests.
|
||||
*
|
||||
* After the results are collected,
|
||||
* TestSequencer.cacheResults(tests: Array<Test>, results: AggregatedResult)
|
||||
* is called to store/update this information on the cache map.
|
||||
*/
|
||||
declare class TestSequencer {
|
||||
private readonly _cache;
|
||||
constructor(_options: TestSequencerOptions);
|
||||
_getCachePath(testContext: TestContext): string;
|
||||
_getCache(test: Test): Cache;
|
||||
private _shardPosition;
|
||||
/**
|
||||
* Select tests for shard requested via --shard=shardIndex/shardCount
|
||||
* Sharding is applied before sorting
|
||||
*
|
||||
* @param tests All tests
|
||||
* @param options shardIndex and shardIndex to select
|
||||
*
|
||||
* @example
|
||||
* ```typescript
|
||||
* class CustomSequencer extends Sequencer {
|
||||
* shard(tests, { shardIndex, shardCount }) {
|
||||
* const shardSize = Math.ceil(tests.length / options.shardCount);
|
||||
* const shardStart = shardSize * (options.shardIndex - 1);
|
||||
* const shardEnd = shardSize * options.shardIndex;
|
||||
* return [...tests]
|
||||
* .sort((a, b) => (a.path > b.path ? 1 : -1))
|
||||
* .slice(shardStart, shardEnd);
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*/
|
||||
shard(tests: Array<Test>, options: ShardOptions): Array<Test> | Promise<Array<Test>>;
|
||||
/**
|
||||
* Sort test to determine order of execution
|
||||
* Sorting is applied after sharding
|
||||
* @param tests
|
||||
*
|
||||
* ```typescript
|
||||
* class CustomSequencer extends Sequencer {
|
||||
* sort(tests) {
|
||||
* const copyTests = Array.from(tests);
|
||||
* return [...tests].sort((a, b) => (a.path > b.path ? 1 : -1));
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*/
|
||||
sort(tests: Array<Test>): Array<Test> | Promise<Array<Test>>;
|
||||
allFailedTests(tests: Array<Test>): Array<Test> | Promise<Array<Test>>;
|
||||
cacheResults(tests: Array<Test>, results: AggregatedResult): void;
|
||||
private hasFailed;
|
||||
private time;
|
||||
}
|
||||
//#endregion
|
||||
export { ShardOptions, TestSequencerOptions, TestSequencer as default };
|
||||
97
frontend/node_modules/@jest/test-sequencer/build/index.d.ts
generated
vendored
Normal file
97
frontend/node_modules/@jest/test-sequencer/build/index.d.ts
generated
vendored
Normal file
@@ -0,0 +1,97 @@
|
||||
/**
|
||||
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
||||
*
|
||||
* This source code is licensed under the MIT license found in the
|
||||
* LICENSE file in the root directory of this source tree.
|
||||
*/
|
||||
|
||||
import {AggregatedResult, Test, TestContext} from '@jest/test-result';
|
||||
import {Config} from '@jest/types';
|
||||
|
||||
declare type Cache_2 = {
|
||||
[key: string]:
|
||||
| [testStatus: typeof FAIL | typeof SUCCESS, testDuration: number]
|
||||
| undefined;
|
||||
};
|
||||
|
||||
declare const FAIL = 0;
|
||||
|
||||
export declare type ShardOptions = {
|
||||
shardIndex: number;
|
||||
shardCount: number;
|
||||
};
|
||||
|
||||
declare const SUCCESS = 1;
|
||||
|
||||
/**
|
||||
* The TestSequencer will ultimately decide which tests should run first.
|
||||
* It is responsible for storing and reading from a local cache
|
||||
* map that stores context information for a given test, such as how long it
|
||||
* took to run during the last run and if it has failed or not.
|
||||
* Such information is used on:
|
||||
* TestSequencer.sort(tests: Array<Test>)
|
||||
* to sort the order of the provided tests.
|
||||
*
|
||||
* After the results are collected,
|
||||
* TestSequencer.cacheResults(tests: Array<Test>, results: AggregatedResult)
|
||||
* is called to store/update this information on the cache map.
|
||||
*/
|
||||
declare class TestSequencer {
|
||||
private readonly _cache;
|
||||
constructor(_options: TestSequencerOptions);
|
||||
_getCachePath(testContext: TestContext): string;
|
||||
_getCache(test: Test): Cache_2;
|
||||
private _shardPosition;
|
||||
/**
|
||||
* Select tests for shard requested via --shard=shardIndex/shardCount
|
||||
* Sharding is applied before sorting
|
||||
*
|
||||
* @param tests All tests
|
||||
* @param options shardIndex and shardIndex to select
|
||||
*
|
||||
* @example
|
||||
* ```typescript
|
||||
* class CustomSequencer extends Sequencer {
|
||||
* shard(tests, { shardIndex, shardCount }) {
|
||||
* const shardSize = Math.ceil(tests.length / options.shardCount);
|
||||
* const shardStart = shardSize * (options.shardIndex - 1);
|
||||
* const shardEnd = shardSize * options.shardIndex;
|
||||
* return [...tests]
|
||||
* .sort((a, b) => (a.path > b.path ? 1 : -1))
|
||||
* .slice(shardStart, shardEnd);
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*/
|
||||
shard(
|
||||
tests: Array<Test>,
|
||||
options: ShardOptions,
|
||||
): Array<Test> | Promise<Array<Test>>;
|
||||
/**
|
||||
* Sort test to determine order of execution
|
||||
* Sorting is applied after sharding
|
||||
* @param tests
|
||||
*
|
||||
* ```typescript
|
||||
* class CustomSequencer extends Sequencer {
|
||||
* sort(tests) {
|
||||
* const copyTests = Array.from(tests);
|
||||
* return [...tests].sort((a, b) => (a.path > b.path ? 1 : -1));
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*/
|
||||
sort(tests: Array<Test>): Array<Test> | Promise<Array<Test>>;
|
||||
allFailedTests(tests: Array<Test>): Array<Test> | Promise<Array<Test>>;
|
||||
cacheResults(tests: Array<Test>, results: AggregatedResult): void;
|
||||
private hasFailed;
|
||||
private time;
|
||||
}
|
||||
export default TestSequencer;
|
||||
|
||||
export declare type TestSequencerOptions = {
|
||||
contexts: ReadonlyArray<TestContext>;
|
||||
globalConfig: Config.GlobalConfig;
|
||||
};
|
||||
|
||||
export {};
|
||||
255
frontend/node_modules/@jest/test-sequencer/build/index.js
generated
vendored
Normal file
255
frontend/node_modules/@jest/test-sequencer/build/index.js
generated
vendored
Normal file
@@ -0,0 +1,255 @@
|
||||
/*!
|
||||
* /**
|
||||
* * Copyright (c) Meta Platforms, Inc. and affiliates.
|
||||
* *
|
||||
* * This source code is licensed under the MIT license found in the
|
||||
* * LICENSE file in the root directory of this source tree.
|
||||
* * /
|
||||
*/
|
||||
/******/ (() => { // webpackBootstrap
|
||||
/******/ "use strict";
|
||||
var __webpack_exports__ = {};
|
||||
// This entry needs to be wrapped in an IIFE because it uses a non-standard name for the exports (exports).
|
||||
(() => {
|
||||
var exports = __webpack_exports__;
|
||||
|
||||
|
||||
Object.defineProperty(exports, "__esModule", ({
|
||||
value: true
|
||||
}));
|
||||
exports["default"] = void 0;
|
||||
function crypto() {
|
||||
const data = _interopRequireWildcard(require("crypto"));
|
||||
crypto = function () {
|
||||
return data;
|
||||
};
|
||||
return data;
|
||||
}
|
||||
function path() {
|
||||
const data = _interopRequireWildcard(require("path"));
|
||||
path = function () {
|
||||
return data;
|
||||
};
|
||||
return data;
|
||||
}
|
||||
function fs() {
|
||||
const data = _interopRequireWildcard(require("graceful-fs"));
|
||||
fs = function () {
|
||||
return data;
|
||||
};
|
||||
return data;
|
||||
}
|
||||
function _slash() {
|
||||
const data = _interopRequireDefault(require("slash"));
|
||||
_slash = function () {
|
||||
return data;
|
||||
};
|
||||
return data;
|
||||
}
|
||||
function _jestHasteMap() {
|
||||
const data = _interopRequireDefault(require("jest-haste-map"));
|
||||
_jestHasteMap = function () {
|
||||
return data;
|
||||
};
|
||||
return data;
|
||||
}
|
||||
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
||||
function _interopRequireWildcard(e, t) { if ("function" == typeof WeakMap) var r = new WeakMap(), n = new WeakMap(); return (_interopRequireWildcard = function (e, t) { if (!t && e && e.__esModule) return e; var o, i, f = { __proto__: null, default: e }; if (null === e || "object" != typeof e && "function" != typeof e) return f; if (o = t ? n : r) { if (o.has(e)) return o.get(e); o.set(e, f); } for (const t in e) "default" !== t && {}.hasOwnProperty.call(e, t) && ((i = (o = Object.defineProperty) && Object.getOwnPropertyDescriptor(e, t)) && (i.get || i.set) ? o(f, t, i) : f[t] = e[t]); return f; })(e, t); }
|
||||
/**
|
||||
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
||||
*
|
||||
* This source code is licensed under the MIT license found in the
|
||||
* LICENSE file in the root directory of this source tree.
|
||||
*/
|
||||
|
||||
const FAIL = 0;
|
||||
const SUCCESS = 1;
|
||||
/**
|
||||
* The TestSequencer will ultimately decide which tests should run first.
|
||||
* It is responsible for storing and reading from a local cache
|
||||
* map that stores context information for a given test, such as how long it
|
||||
* took to run during the last run and if it has failed or not.
|
||||
* Such information is used on:
|
||||
* TestSequencer.sort(tests: Array<Test>)
|
||||
* to sort the order of the provided tests.
|
||||
*
|
||||
* After the results are collected,
|
||||
* TestSequencer.cacheResults(tests: Array<Test>, results: AggregatedResult)
|
||||
* is called to store/update this information on the cache map.
|
||||
*/
|
||||
class TestSequencer {
|
||||
_cache = new Map();
|
||||
|
||||
// eslint-disable-next-line @typescript-eslint/no-empty-function,@typescript-eslint/no-useless-constructor
|
||||
constructor(_options) {}
|
||||
_getCachePath(testContext) {
|
||||
const {
|
||||
config
|
||||
} = testContext;
|
||||
const HasteMapClass = _jestHasteMap().default.getStatic(config);
|
||||
return HasteMapClass.getCacheFilePath(config.cacheDirectory, `perf-cache-${config.id}`);
|
||||
}
|
||||
_getCache(test) {
|
||||
const {
|
||||
context
|
||||
} = test;
|
||||
if (!this._cache.has(context) && context.config.cache) {
|
||||
const cachePath = this._getCachePath(context);
|
||||
if (fs().existsSync(cachePath)) {
|
||||
try {
|
||||
this._cache.set(context, JSON.parse(fs().readFileSync(cachePath, 'utf8')));
|
||||
} catch {}
|
||||
}
|
||||
}
|
||||
let cache = this._cache.get(context);
|
||||
if (!cache) {
|
||||
cache = {};
|
||||
this._cache.set(context, cache);
|
||||
}
|
||||
return cache;
|
||||
}
|
||||
_shardPosition(options) {
|
||||
const shardRest = options.suiteLength % options.shardCount;
|
||||
const ratio = options.suiteLength / options.shardCount;
|
||||
return Array.from({
|
||||
length: options.shardIndex
|
||||
}).reduce((acc, _, shardIndex) => {
|
||||
const dangles = shardIndex < shardRest;
|
||||
const shardSize = dangles ? Math.ceil(ratio) : Math.floor(ratio);
|
||||
return acc + shardSize;
|
||||
}, 0);
|
||||
}
|
||||
|
||||
/**
|
||||
* Select tests for shard requested via --shard=shardIndex/shardCount
|
||||
* Sharding is applied before sorting
|
||||
*
|
||||
* @param tests All tests
|
||||
* @param options shardIndex and shardIndex to select
|
||||
*
|
||||
* @example
|
||||
* ```typescript
|
||||
* class CustomSequencer extends Sequencer {
|
||||
* shard(tests, { shardIndex, shardCount }) {
|
||||
* const shardSize = Math.ceil(tests.length / options.shardCount);
|
||||
* const shardStart = shardSize * (options.shardIndex - 1);
|
||||
* const shardEnd = shardSize * options.shardIndex;
|
||||
* return [...tests]
|
||||
* .sort((a, b) => (a.path > b.path ? 1 : -1))
|
||||
* .slice(shardStart, shardEnd);
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*/
|
||||
shard(tests, options) {
|
||||
const shardStart = this._shardPosition({
|
||||
shardCount: options.shardCount,
|
||||
shardIndex: options.shardIndex - 1,
|
||||
suiteLength: tests.length
|
||||
});
|
||||
const shardEnd = this._shardPosition({
|
||||
shardCount: options.shardCount,
|
||||
shardIndex: options.shardIndex,
|
||||
suiteLength: tests.length
|
||||
});
|
||||
return tests.map(test => {
|
||||
const relativeTestPath = path().posix.relative((0, _slash().default)(test.context.config.rootDir), (0, _slash().default)(test.path));
|
||||
return {
|
||||
hash: crypto().createHash('sha1').update(relativeTestPath).digest('hex'),
|
||||
test
|
||||
};
|
||||
}).sort((a, b) => a.hash < b.hash ? -1 : a.hash > b.hash ? 1 : 0).slice(shardStart, shardEnd).map(result => result.test);
|
||||
}
|
||||
|
||||
/**
|
||||
* Sort test to determine order of execution
|
||||
* Sorting is applied after sharding
|
||||
* @param tests
|
||||
*
|
||||
* ```typescript
|
||||
* class CustomSequencer extends Sequencer {
|
||||
* sort(tests) {
|
||||
* const copyTests = Array.from(tests);
|
||||
* return [...tests].sort((a, b) => (a.path > b.path ? 1 : -1));
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*/
|
||||
sort(tests) {
|
||||
/**
|
||||
* Sorting tests is very important because it has a great impact on the
|
||||
* user-perceived responsiveness and speed of the test run.
|
||||
*
|
||||
* If such information is on cache, tests are sorted based on:
|
||||
* -> Has it failed during the last run ?
|
||||
* Since it's important to provide the most expected feedback as quickly
|
||||
* as possible.
|
||||
* -> How long it took to run ?
|
||||
* Because running long tests first is an effort to minimize worker idle
|
||||
* time at the end of a long test run.
|
||||
* And if that information is not available they are sorted based on file size
|
||||
* since big test files usually take longer to complete.
|
||||
*
|
||||
* Note that a possible improvement would be to analyse other information
|
||||
* from the file other than its size.
|
||||
*
|
||||
*/
|
||||
const stats = {};
|
||||
const fileSize = ({
|
||||
path,
|
||||
context: {
|
||||
hasteFS
|
||||
}
|
||||
}) => stats[path] || (stats[path] = hasteFS.getSize(path) ?? 0);
|
||||
for (const test of tests) {
|
||||
test.duration = this.time(test);
|
||||
}
|
||||
return tests.sort((testA, testB) => {
|
||||
const failedA = this.hasFailed(testA);
|
||||
const failedB = this.hasFailed(testB);
|
||||
const hasTimeA = testA.duration != null;
|
||||
const hasTimeB = testB.duration != null;
|
||||
if (failedA !== failedB) {
|
||||
return failedA ? -1 : 1;
|
||||
} else if (hasTimeA !== hasTimeB) {
|
||||
// If only one of two tests has timing information, run it last
|
||||
return hasTimeA ? 1 : -1;
|
||||
} else if (testA.duration != null && testB.duration != null) {
|
||||
return testA.duration < testB.duration ? 1 : -1;
|
||||
} else {
|
||||
return fileSize(testA) < fileSize(testB) ? 1 : -1;
|
||||
}
|
||||
});
|
||||
}
|
||||
allFailedTests(tests) {
|
||||
return this.sort(tests.filter(test => this.hasFailed(test)));
|
||||
}
|
||||
cacheResults(tests, results) {
|
||||
const map = Object.create(null);
|
||||
for (const test of tests) map[test.path] = test;
|
||||
for (const testResult of results.testResults) {
|
||||
const test = map[testResult.testFilePath];
|
||||
if (test != null && !testResult.skipped) {
|
||||
const cache = this._getCache(test);
|
||||
const perf = testResult.perfStats;
|
||||
const testRuntime = perf.runtime ?? test.duration ?? perf.end - perf.start;
|
||||
cache[testResult.testFilePath] = [testResult.numFailingTests > 0 ? FAIL : SUCCESS, testRuntime || 0];
|
||||
}
|
||||
}
|
||||
for (const [context, cache] of this._cache.entries()) fs().writeFileSync(this._getCachePath(context), JSON.stringify(cache));
|
||||
}
|
||||
hasFailed(test) {
|
||||
const cache = this._getCache(test);
|
||||
return cache[test.path]?.[0] === FAIL;
|
||||
}
|
||||
time(test) {
|
||||
const cache = this._getCache(test);
|
||||
return cache[test.path]?.[1];
|
||||
}
|
||||
}
|
||||
exports["default"] = TestSequencer;
|
||||
})();
|
||||
|
||||
module.exports = __webpack_exports__;
|
||||
/******/ })()
|
||||
;
|
||||
3
frontend/node_modules/@jest/test-sequencer/build/index.mjs
generated
vendored
Normal file
3
frontend/node_modules/@jest/test-sequencer/build/index.mjs
generated
vendored
Normal file
@@ -0,0 +1,3 @@
|
||||
import cjsModule from './index.js';
|
||||
|
||||
export default cjsModule.default;
|
||||
Reference in New Issue
Block a user