How to use mutantPlans method in stryker-parent

Best JavaScript code snippet using stryker-parent

mutant-test-planner.ts

Source:mutant-test-planner.ts Github

copy

Full Screen

1import path from 'path';2import { TestResult } from '@stryker-mutator/api/test-runner';3import { MutantRunPlan, MutantTestPlan, PlanKind, Mutant, StrykerOptions, MutantStatus, MutantEarlyResultPlan } from '@stryker-mutator/api/core';4import { commonTokens, tokens } from '@stryker-mutator/api/plugin';5import { Logger } from '@stryker-mutator/api/logging';6import { I, notEmpty } from '@stryker-mutator/util';7import { coreTokens } from '../di/index.js';8import { StrictReporter } from '../reporters/strict-reporter.js';9import { Sandbox } from '../sandbox/index.js';10import { objectUtils } from '../utils/object-utils.js';11import { optionsPath } from '../utils/index.js';12import { Project } from '../fs/project.js';13import { IncrementalDiffer, toRelativeNormalizedFileName } from './incremental-differ.js';14import { TestCoverage } from './test-coverage.js';15/**16 * The factor by which hit count from dry run is multiplied to calculate the hit limit for a mutant.17 * This is intentionally a high value to prevent false positives.18 *19 * For example, a property testing library might execute a failing scenario multiple times to determine the smallest possible counterexample.20 * @see https://jsverify.github.io/#minimal-counterexample21 */22const HIT_LIMIT_FACTOR = 100;23/**24 * Responsible for determining the tests to execute for each mutant, as well as other run option specific details25 *26 */27export class MutantTestPlanner {28 public static readonly inject = tokens(29 coreTokens.testCoverage,30 coreTokens.incrementalDiffer,31 coreTokens.reporter,32 coreTokens.sandbox,33 coreTokens.project,34 coreTokens.timeOverheadMS,35 commonTokens.options,36 commonTokens.logger37 );38 private readonly timeSpentAllTests: number;39 constructor(40 private readonly testCoverage: I<TestCoverage>,41 private readonly incrementalDiffer: IncrementalDiffer,42 private readonly reporter: StrictReporter,43 private readonly sandbox: I<Sandbox>,44 private readonly project: I<Project>,45 private readonly timeOverheadMS: number,46 private readonly options: StrykerOptions,47 private readonly logger: Logger48 ) {49 this.timeSpentAllTests = calculateTotalTime(this.testCoverage.testsById.values());50 }51 public async makePlan(mutants: readonly Mutant[]): Promise<readonly MutantTestPlan[]> {52 const mutantsDiff = await this.incrementalDiff(mutants);53 const mutantPlans = mutantsDiff.map((mutant) => this.planMutant(mutant));54 this.reporter.onMutationTestingPlanReady({ mutantPlans });55 this.warnAboutSlow(mutantPlans);56 return mutantPlans;57 }58 private planMutant(mutant: Mutant): MutantTestPlan {59 const isStatic = this.testCoverage.hasStaticCoverage(mutant.id);60 if (mutant.status) {61 // If this mutant was already ignored, early result62 return this.createMutantEarlyResultPlan(mutant, {63 isStatic,64 coveredBy: mutant.coveredBy,65 killedBy: mutant.killedBy,66 status: mutant.status,67 statusReason: mutant.statusReason,68 });69 } else if (this.testCoverage.hasCoverage) {70 // If there was coverage information (coverageAnalysis not "off")71 const tests = this.testCoverage.testsByMutantId.get(mutant.id) ?? [];72 const coveredBy = toTestIds(tests);73 if (!isStatic || (this.options.ignoreStatic && coveredBy.length)) {74 // If not static, or it was "hybrid" (both static and perTest coverage) and ignoreStatic is on.75 // Only run covered tests with mutant active during runtime76 const netTime = calculateTotalTime(tests);77 return this.createMutantRunPlan(mutant, { netTime, coveredBy, isStatic, testFilter: coveredBy });78 } else if (this.options.ignoreStatic) {79 // Static (w/o perTest coverage) and ignoreStatic is on -> Ignore.80 return this.createMutantEarlyResultPlan(mutant, {81 status: MutantStatus.Ignored,82 statusReason: 'Static mutant (and "ignoreStatic" was enabled)',83 isStatic,84 coveredBy,85 });86 } else {87 // Static (or hybrid) and `ignoreStatic` is off -> run all tests88 return this.createMutantRunPlan(mutant, { netTime: this.timeSpentAllTests, isStatic, coveredBy });89 }90 } else {91 // No coverage information exists, all tests need to run92 return this.createMutantRunPlan(mutant, { netTime: this.timeSpentAllTests });93 }94 }95 private createMutantEarlyResultPlan(96 mutant: Mutant,97 {98 isStatic,99 status,100 statusReason,101 coveredBy,102 killedBy,103 }: { isStatic: boolean | undefined; status: MutantStatus; statusReason?: string; coveredBy?: string[]; killedBy?: string[] }104 ): MutantEarlyResultPlan {105 return {106 plan: PlanKind.EarlyResult,107 mutant: {108 ...mutant,109 status,110 static: isStatic,111 statusReason,112 coveredBy,113 killedBy,114 },115 };116 }117 private createMutantRunPlan(118 mutant: Mutant,119 {120 netTime,121 testFilter,122 isStatic,123 coveredBy,124 }: { netTime: number; testFilter?: string[] | undefined; isStatic?: boolean | undefined; coveredBy?: string[] | undefined }125 ): MutantRunPlan {126 const { disableBail, timeoutMS, timeoutFactor } = this.options;127 const timeout = timeoutFactor * netTime + timeoutMS + this.timeOverheadMS;128 const hitCount = this.testCoverage.hitsByMutantId.get(mutant.id);129 const hitLimit = hitCount === undefined ? undefined : hitCount * HIT_LIMIT_FACTOR;130 return {131 plan: PlanKind.Run,132 netTime,133 mutant: {134 ...mutant,135 coveredBy,136 static: isStatic,137 },138 runOptions: {139 // Copy over relevant mutant fields, we don't want to copy over "static" and "coveredBy", test runners should only care about the testFilter140 activeMutant: {141 id: mutant.id,142 fileName: mutant.fileName,143 location: mutant.location,144 mutatorName: mutant.mutatorName,145 replacement: mutant.replacement,146 },147 mutantActivation: testFilter ? 'runtime' : 'static',148 timeout,149 testFilter,150 sandboxFileName: this.sandbox.sandboxFileFor(mutant.fileName),151 hitLimit,152 disableBail,153 reloadEnvironment: !testFilter,154 },155 };156 }157 private warnAboutSlow(mutantPlans: readonly MutantTestPlan[]) {158 if (!this.options.ignoreStatic && objectUtils.isWarningEnabled('slow', this.options.warnings)) {159 // Only warn when the estimated time to run all static mutants exceeds 40%160 // ... and when the average performance impact of a static mutant is estimated to be twice that (or more) of a non-static mutant161 const ABSOLUTE_CUT_OFF_PERUNAGE = 0.4;162 const RELATIVE_CUT_OFF_FACTOR = 2;163 const zeroIfNaN = (n: number) => (isNaN(n) ? 0 : n);164 const runPlans = mutantPlans.filter(isRunPlan);165 const staticRunPlans = runPlans.filter(({ mutant }) => mutant.static);166 const runTimeRunPlans = runPlans.filter(({ mutant }) => !mutant.static);167 const estimatedTimeForStaticMutants = staticRunPlans.reduce((acc, { netTime }) => acc + netTime, 0);168 const estimatedTimeForRunTimeMutants = runTimeRunPlans.reduce((acc, { netTime }) => acc + netTime, 0);169 const estimatedTotalTime = estimatedTimeForRunTimeMutants + estimatedTimeForStaticMutants;170 const avgTimeForAStaticMutant = zeroIfNaN(estimatedTimeForStaticMutants / staticRunPlans.length);171 const avgTimeForARunTimeMutant = zeroIfNaN(estimatedTimeForRunTimeMutants / runTimeRunPlans.length);172 const relativeTimeForStaticMutants = estimatedTimeForStaticMutants / estimatedTotalTime;173 const absoluteCondition = relativeTimeForStaticMutants >= ABSOLUTE_CUT_OFF_PERUNAGE;174 const relativeCondition = avgTimeForAStaticMutant >= RELATIVE_CUT_OFF_FACTOR * avgTimeForARunTimeMutant;175 if (relativeCondition && absoluteCondition) {176 const percentage = (perunage: number) => Math.round(perunage * 100);177 this.logger.warn(178 `Detected ${staticRunPlans.length} static mutants (${percentage(179 staticRunPlans.length / runPlans.length180 )}% of total) that are estimated to take ${percentage(181 relativeTimeForStaticMutants182 )}% of the time running the tests!\n You might want to enable "ignoreStatic" to ignore these static mutants for your next run. \n For more information about static mutants visit: https://stryker-mutator.io/docs/mutation-testing-elements/static-mutants.\n (disable "${optionsPath(183 'warnings',184 'slow'185 )}" to ignore this warning)`186 );187 }188 }189 }190 private async incrementalDiff(currentMutants: readonly Mutant[]): Promise<readonly Mutant[]> {191 const { incrementalReport } = this.project;192 if (incrementalReport) {193 const currentFiles = await this.readAllOriginalFiles(194 currentMutants,195 this.testCoverage.testsById.values(),196 Object.keys(incrementalReport.files),197 Object.keys(incrementalReport.testFiles ?? {})198 );199 const diffedMutants = this.incrementalDiffer.diff(currentMutants, this.testCoverage, incrementalReport, currentFiles);200 return diffedMutants;201 }202 return currentMutants;203 }204 private async readAllOriginalFiles(205 ...thingsWithFileNamesOrFileNames: Array<Iterable<string | { fileName?: string }>>206 ): Promise<Map<string, string>> {207 const uniqueFileNames = [208 ...new Set(209 thingsWithFileNamesOrFileNames210 .flatMap((container) => [...container].map((thing) => (typeof thing === 'string' ? thing : thing.fileName)))211 .filter(notEmpty)212 .map((fileName) => path.resolve(fileName))213 ),214 ];215 const result = await Promise.all(216 uniqueFileNames.map(async (fileName) => {217 const originalContent = await this.project.files.get(fileName)?.readOriginal();218 if (originalContent) {219 return [toRelativeNormalizedFileName(fileName), originalContent] as const;220 } else {221 return undefined;222 }223 })224 );225 return new Map(result.filter(notEmpty));226 }227}228function calculateTotalTime(testResults: Iterable<TestResult>): number {229 let total = 0;230 for (const test of testResults) {231 total += test.timeSpentMs;232 }233 return total;234}235function toTestIds(testResults: Iterable<TestResult>): string[] {236 const result = [];237 for (const test of testResults) {238 result.push(test.id);239 }240 return result;241}242export function isEarlyResult(mutantPlan: MutantTestPlan): mutantPlan is MutantEarlyResultPlan {243 return mutantPlan.plan === PlanKind.EarlyResult;244}245export function isRunPlan(mutantPlan: MutantTestPlan): mutantPlan is MutantRunPlan {246 return mutantPlan.plan === PlanKind.Run;...

Full Screen

Full Screen

progress-reporter.spec.ts

Source:progress-reporter.spec.ts Github

copy

Full Screen

1import { expect } from 'chai';2import sinon from 'sinon';3import ProgressBar from 'progress';4import { factory } from '@stryker-mutator/test-helpers';5import { progressBarWrapper } from '../../../src/reporters/progress-bar.js';6import { ProgressBarReporter } from '../../../src/reporters/progress-reporter.js';7import { Mock, mock } from '../../helpers/producers.js';8const SECOND = 1000;9const TEN_SECONDS = SECOND * 10;10const HUNDRED_SECONDS = SECOND * 100;11const TEN_THOUSAND_SECONDS = SECOND * 10000;12const ONE_HOUR = SECOND * 3600;13describe(ProgressBarReporter.name, () => {14 let sut: ProgressBarReporter;15 let progressBar: Mock<ProgressBar>;16 let progressBarConstructorStub: sinon.SinonStub;17 const progressBarContent =18 'Mutation testing [:bar] :percent (elapsed: :et, remaining: :etc) :tested/:mutants Mutants tested (:survived survived, :timedOut timed out)';19 beforeEach(() => {20 sinon.useFakeTimers();21 sut = new ProgressBarReporter();22 progressBar = mock(ProgressBar);23 progressBarConstructorStub = sinon.stub(progressBarWrapper, 'ProgressBar');24 progressBarConstructorStub.returns(progressBar);25 });26 describe('onMutationTestingPlanReady()', () => {27 it('should calculate the correct total', () => {28 sut.onDryRunCompleted(29 factory.dryRunCompletedEvent({30 result: factory.completeDryRunResult({31 tests: [factory.testResult({ id: '1', timeSpentMs: 10 }), factory.testResult({ id: '2', timeSpentMs: 5 })],32 }),33 timing: factory.runTiming({ net: 15, overhead: 100 }),34 })35 );36 sut.onMutationTestingPlanReady(37 factory.mutationTestingPlanReadyEvent({38 mutantPlans: [39 // Ignored mutant40 factory.mutantEarlyResultPlan({ mutant: factory.ignoredMutantTestCoverage({ id: '1' }) }),41 // Run test 1, takes 10ms42 factory.mutantRunPlan({43 mutant: factory.mutantTestCoverage({ id: '2' }),44 runOptions: factory.mutantRunOptions({ testFilter: ['1'] }),45 netTime: 10,46 }),47 // Run test 2, takes 5ms48 factory.mutantRunPlan({49 mutant: factory.mutantTestCoverage({ id: '3' }),50 runOptions: factory.mutantRunOptions({ testFilter: ['2'] }),51 netTime: 5,52 }),53 // Run all tests, takes 115ms54 factory.mutantRunPlan({55 mutant: factory.mutantTestCoverage({ id: '4' }),56 runOptions: factory.mutantRunOptions({ testFilter: undefined, reloadEnvironment: true }),57 netTime: 15,58 }),59 ],60 })61 );62 expect(progressBarConstructorStub).calledWithMatch(progressBarContent, { total: 115 + 5 + 10 });63 });64 });65 describe('onMutantTested()', () => {66 let progressBarTickTokens: any;67 beforeEach(() => {68 sut.onDryRunCompleted(69 factory.dryRunCompletedEvent({70 result: factory.completeDryRunResult({71 tests: [factory.testResult({ id: '1', timeSpentMs: 10 }), factory.testResult({ id: '2', timeSpentMs: 5 })],72 }),73 timing: factory.runTiming({ net: 15, overhead: 100 }),74 })75 );76 sut.onMutationTestingPlanReady(77 factory.mutationTestingPlanReadyEvent({78 mutantPlans: [79 // Ignored mutant80 factory.mutantEarlyResultPlan({ mutant: factory.ignoredMutantTestCoverage({ id: '1' }) }),81 // Run test 1, takes 10ms82 factory.mutantRunPlan({83 mutant: factory.mutantTestCoverage({ id: '2' }),84 runOptions: factory.mutantRunOptions({ testFilter: ['1'] }),85 netTime: 10,86 }),87 // Run test 2, takes 5ms88 factory.mutantRunPlan({89 mutant: factory.mutantTestCoverage({ id: '3' }),90 runOptions: factory.mutantRunOptions({ testFilter: ['2'] }),91 netTime: 5,92 }),93 // Run all tests, takes 115ms94 factory.mutantRunPlan({95 mutant: factory.mutantTestCoverage({ id: '4' }),96 runOptions: factory.mutantRunOptions({ testFilter: undefined, reloadEnvironment: true }),97 netTime: 15,98 }),99 ],100 })101 );102 });103 it('should tick the ProgressBar with 1 tested mutant, 0 survived when status is not "Survived"', () => {104 sut.onMutantTested(factory.killedMutantResult({ id: '2' }));105 progressBarTickTokens = { total: 130, tested: 1, survived: 0 };106 sinon.assert.calledWithMatch(progressBar.tick, 10, progressBarTickTokens);107 });108 it('should not tick the ProgressBar the result did not yield any ticks', () => {109 progressBar.total = 130;110 sut.onMutantTested(factory.ignoredMutantResult({ id: '1' })); // ignored mutant111 progressBarTickTokens = { total: 130, tested: 0, survived: 0 };112 sinon.assert.notCalled(progressBar.tick);113 sinon.assert.calledWithMatch(progressBar.render, progressBarTickTokens);114 });115 it('should tick the ProgressBar with 1 survived mutant when status is "Survived"', () => {116 sut.onMutantTested(factory.survivedMutantResult({ id: '4', static: true }));117 progressBarTickTokens = { total: 130, tested: 1, survived: 1 };118 sinon.assert.calledWithMatch(progressBar.tick, 115, progressBarTickTokens);119 });120 });121 describe('ProgressBar estimated time for 3 mutants', () => {122 beforeEach(() => {123 sut.onDryRunCompleted(124 factory.dryRunCompletedEvent({125 result: factory.completeDryRunResult({126 tests: [factory.testResult({ id: '1', timeSpentMs: 10 }), factory.testResult({ id: '2', timeSpentMs: 5 })],127 }),128 timing: factory.runTiming({ net: 15, overhead: 100 }),129 })130 );131 sut.onMutationTestingPlanReady(132 factory.mutationTestingPlanReadyEvent({133 mutantPlans: [134 // Ignored mutant135 factory.mutantEarlyResultPlan({ mutant: factory.ignoredMutantTestCoverage({ id: '1' }) }),136 // Run test 1, takes 10ms137 factory.mutantRunPlan({138 mutant: factory.mutantTestCoverage({ id: '2' }),139 runOptions: factory.mutantRunOptions({ testFilter: ['1'] }),140 netTime: 10,141 }),142 // Run test 2, takes 5ms143 factory.mutantRunPlan({144 mutant: factory.mutantTestCoverage({ id: '3' }),145 runOptions: factory.mutantRunOptions({ testFilter: ['2'] }),146 netTime: 5,147 }),148 // Run all tests, takes 115ms149 factory.mutantRunPlan({150 mutant: factory.mutantTestCoverage({ id: '4' }),151 runOptions: factory.mutantRunOptions({ testFilter: undefined, reloadEnvironment: true }),152 netTime: 15,153 }),154 ],155 })156 );157 });158 it('should show correct time info after ten seconds and 3.8% tested', () => {159 sinon.clock.tick(TEN_SECONDS);160 sut.onMutantTested(factory.mutantResult({ id: '3' }));161 sinon.assert.calledWithMatch(progressBar.tick, 5, { et: '<1m', etc: '~4m' });162 });163 it('should show correct time info after a hundred seconds and 7% tested', () => {164 sinon.clock.tick(HUNDRED_SECONDS);165 sut.onMutantTested(factory.mutantResult({ id: '2' }));166 sinon.assert.calledWithMatch(progressBar.tick, 10, { et: '~1m', etc: '~20m' });167 });168 it('should show correct time info after ten thousand seconds and 88%', () => {169 sinon.clock.tick(TEN_THOUSAND_SECONDS);170 sut.onMutantTested(factory.mutantResult({ id: '4' }));171 sinon.assert.calledWithMatch(progressBar.tick, 115, { et: '~2h 46m', etc: '~21m' });172 });173 it('should show correct time info after an hour and 11% tested', () => {174 sinon.clock.tick(ONE_HOUR);175 sut.onMutantTested(factory.mutantResult({ id: '2' }));176 sut.onMutantTested(factory.mutantResult({ id: '3' }));177 sinon.assert.calledWithMatch(progressBar.tick, 10, { et: '~1h 0m', etc: '~12h 0m' });178 sinon.assert.calledWithMatch(progressBar.tick, 5, { et: '~1h 0m', etc: '~7h 40m' });179 });180 });...

Full Screen

Full Screen

progress-keeper.ts

Source:progress-keeper.ts Github

copy

Full Screen

1import { MutantResult, MutantStatus, MutantRunPlan, MutantTestPlan, PlanKind } from '@stryker-mutator/api/core';2import { DryRunCompletedEvent, MutationTestingPlanReadyEvent, Reporter, RunTiming } from '@stryker-mutator/api/report';3import { Timer } from '../utils/timer.js';4export abstract class ProgressKeeper implements Reporter {5 private timer!: Timer;6 private timing!: RunTiming;7 private ticksByMutantId!: Map<string, number>;8 protected progress = {9 survived: 0,10 timedOut: 0,11 tested: 0,12 mutants: 0,13 total: 0,14 ticks: 0,15 };16 public onDryRunCompleted({ timing }: DryRunCompletedEvent): void {17 this.timing = timing;18 }19 /**20 * An event emitted when the mutant test plan is calculated.21 * @param event The mutant test plan ready event22 */23 public onMutationTestingPlanReady({ mutantPlans }: MutationTestingPlanReadyEvent): void {24 this.timer = new Timer();25 this.ticksByMutantId = new Map(26 mutantPlans.filter(isRunPlan).map(({ netTime, mutant, runOptions }) => {27 let ticks = netTime;28 if (runOptions.reloadEnvironment) {29 ticks += this.timing.overhead;30 }31 return [mutant.id, ticks];32 })33 );34 this.progress.mutants = this.ticksByMutantId.size;35 this.progress.total = [...this.ticksByMutantId.values()].reduce((acc, n) => acc + n, 0);36 }37 public onMutantTested(result: MutantResult): number {38 const ticks = this.ticksByMutantId.get(result.id);39 if (ticks !== undefined) {40 this.progress.tested++;41 this.progress.ticks += this.ticksByMutantId.get(result.id) ?? 0;42 if (result.status === MutantStatus.Survived) {43 this.progress.survived++;44 }45 if (result.status === MutantStatus.Timeout) {46 this.progress.timedOut++;47 }48 }49 return ticks ?? 0;50 }51 protected getElapsedTime(): string {52 return this.formatTime(this.timer.elapsedSeconds());53 }54 protected getEtc(): string {55 const totalSecondsLeft = Math.floor((this.timer.elapsedSeconds() / this.progress.ticks) * (this.progress.total - this.progress.ticks));56 if (isFinite(totalSecondsLeft) && totalSecondsLeft > 0) {57 return this.formatTime(totalSecondsLeft);58 } else {59 return 'n/a';60 }61 }62 private formatTime(timeInSeconds: number) {63 const hours = Math.floor(timeInSeconds / 3600);64 const minutes = Math.floor((timeInSeconds % 3600) / 60);65 return hours > 0 // conditional time formatting66 ? `~${hours}h ${minutes}m`67 : minutes > 068 ? `~${minutes}m`69 : '<1m';70 }71}72function isRunPlan(mutantPlan: MutantTestPlan): mutantPlan is MutantRunPlan {73 return mutantPlan.plan === PlanKind.Run;...

Full Screen

Full Screen

Using AI Code Generation

copy

Full Screen

1const mutantPlans = require('stryker-parent').mutantPlans;2module.exports = function (config) {3 config.set({4 jest: {5 },6 thresholds: { high: 80, low: 60, break: 50 }7 });8};9module.exports = {10};11[2018-10-31 17:32:12.696] [INFO] Stryker - 1 Mutant(s) generated12[2018-10-31 17:32:12.697] [INFO] Stryker - 1 Mutant(s) generated13[2018-10-31 17:32:12.697] [INFO] Stryker - Starting test run 1/1 (mutant 1/1)

Full Screen

Using AI Code Generation

copy

Full Screen

1var mutantPlans = require('stryker-parent').mutantPlans;2var plans = mutantPlans.getPlans();3console.log(plans);4{5 "dependencies": {6 }7}

Full Screen

Using AI Code Generation

copy

Full Screen

1var mutantPlans = require('stryker-parent').mutantPlans;2var plans = mutantPlans({ files: ['src/**/*.js', 'test/**/*.js'] });3var mutantPlans = require('stryker-parent').mutantPlans;4var plans = mutantPlans({ files: ['src/**/*.js', 'test/**/*.js'] });5module.exports = function(config) {6 config.set({7 });8};

Full Screen

Using AI Code Generation

copy

Full Screen

1const mutantPlans = require('stryker-parent').mutantPlans;2const plans = mutantPlans({ files: ['a.js', 'b.js'] });3console.log(plans);4const a = 1;5const b = 2;6module.exports = function (config) {7 config.set({8 mutate: mutantPlans({ files: ['a.js', 'b.js'] })9 });10};11Stryker has 2 Mutant(s)12Stryker has 2 file(s) to be mutated13All Mutant(s) are tested, don't forget to review them!

Full Screen

Using AI Code Generation

copy

Full Screen

1var mutantPlans = require('stryker-parent').mutantPlans;2mutantPlans('test.js', 'test.js', function(mutant) {3});4var mutantPlans = require('stryker-parent').mutantPlans;5mutantPlans('test.js', 'test.js', function(mutant) {6});7var mutantPlans = require('stryker-parent').mutantPlans;8mutantPlans('test.js', 'test.js', function(mutant) {9});10var mutantPlans = require('stryker-parent').mutantPlans;11mutantPlans('test.js', 'test.js', function(mutant) {

Full Screen

Automation Testing Tutorials

Learn to execute automation testing from scratch with LambdaTest Learning Hub. Right from setting up the prerequisites to run your first automation test, to following best practices and diving deeper into advanced test scenarios. LambdaTest Learning Hubs compile a list of step-by-step guides to help you be proficient with different test automation frameworks i.e. Selenium, Cypress, TestNG etc.

LambdaTest Learning Hubs:

YouTube

You could also refer to video tutorials over LambdaTest YouTube channel to get step by step demonstration from industry experts.

Run stryker-parent automation tests on LambdaTest cloud grid

Perform automation testing on 3000+ real desktop and mobile devices online.

Try LambdaTest Now !!

Get 100 minutes of automation test minutes FREE!!

Next-Gen App & Browser Testing Cloud

Was this article helpful?

Helpful

NotHelpful