diff --git a/docs/configuration.md b/docs/configuration.md
index e57bef029f..bffa0e68be 100644
--- a/docs/configuration.md
+++ b/docs/configuration.md
@@ -438,12 +438,13 @@ _Note:_ It is **not** possible to combine mutation range with a [globbing expres
Default: `{}`
Command line: _none_
-Config file: `"mutator": { "plugins": ["classProperties"], "excludedMutations": ["StringLiteral"] }`
+Config file: `"mutator": { "plugins": ["classProperties"], "includedMutations": ["MutationSpecification"], "excludedMutations": ["MutationSpecification"] }`
- `plugins`: allows you to override the default [babel plugins](https://babeljs.io/docs/en/plugins) to use for JavaScript files.
By default, Stryker uses [a default list of babel plugins to parse your JS file](https://github.com/stryker-mutator/stryker-js/blob/master/packages/instrumenter/src/parsers/js-parser.ts#L8-L32). It also loads any plugins or presets you might have configured yourself with `.babelrc` or `babel.config.js` files.
In the rare situation where the plugins Stryker loads conflict with your own local plugins (for example, when using the decorators and decorators-legacy plugins together), you can override the `plugins` here to `[]`.
-- `excludedMutations`: allow you to specify a [list of mutator names](https://stryker-mutator.io/docs/mutation-testing-elements/supported-mutators/#supported-mutators) to be excluded (`ignored`) from the test run. See [Disable mutants](./disable-mutants.md) for more options of how to disable specific mutants.
+- `includedMutations`: allow you to specify a [list of mutator names](https://stryker-mutator.io/docs/mutation-testing-elements/supported-mutators/#supported-mutators), mutation operators, or mutation level to be included in the test run. This will exclude anything not specified in this list.
+- `excludedMutations`: allow you to specify a [list of mutator names](https://stryker-mutator.io/docs/mutation-testing-elements/supported-mutators/#supported-mutators) to be excluded (`ignored`) from the test run. See [Disable mutants](./disable-mutants.md) for more options of how to disable specific mutants. In case `includedMutations` is also specified, this will exclude mutation operators from that list.
_Note: prior to Stryker version 4, the mutator also needed a `name` (or be defined as `string`). This is removed in version 4. Stryker now supports mutating of JavaScript and friend files out of the box, without the need for a mutator plugin._
diff --git a/docs/mutation-levels.md b/docs/mutation-levels.md
new file mode 100644
index 0000000000..25b139aa2e
--- /dev/null
+++ b/docs/mutation-levels.md
@@ -0,0 +1,75 @@
+---
+title: Mutation Levels
+custom_edit_url: https://github.com/stryker-mutator/stryker-js/edit/master/docs/mutation-levels.md
+---
+
+This page describes the concept of mutation levels and how to use them in your configuration.
+
+## Terminology
+The smallest unit in mutation testing is the **mutation operator**. This is a single type of mutation, like `AdditionOperatorNegation`, which changes a `+`-operator into a `-`-operator.
+
+Every mutation operator belongs to a **mutator**, also referred to as a **mutator group**. This is a set of mutation operators that can be applied to the same node type.
+For example, the `AdditionOperatorNegation` mutation operator belongs to the `ArithmeticOperator` mutator group.
+
+Finally, a **mutation level** is an artificial grouping of mutation operators with the purpose of striking a balance between performance and efficacy of a mutation run.
+Such a level is not necessarily in line with the previously mentioned mutator groups, but designed to work right away.
+Currently, mutation levels are named from `Level1` to `Level3`, where `Level1` has the best performance and `Level3` has the best efficacy.
+
+## Specifying included/excluded mutators
+By default, all of Stryker's mutators will be run on your project, which gives the maximum efficacy but also takes the most resources to run.
+If you want to enable mutation levels, you can choose a level from 1 to 3, like this: ``includedMutators: ['@Level1']``.
+For most users, this should suffice without further tweaks as these mutation levels are designed based upon a representable sample of JS and TS projects.
+
+In case you want a more advanced and customized configuration, you can tweak your selected mutation level by adding or removing mutation operators and/or mutator groups.
+A **mutation operator** can be specified with its literal name. **Mutator groups** and **mutation levels** are specified with the `@` prefix, for example `@ArithmeticOperator` or `@Level1`.
+For example, if you want to tweak level 2 by removing all `ArithmeticOperator`'s mutation operators except for the `AdditionOperatorNegation`, you would write this:
+```
+ includedMutators: ['@Level1', 'AdditionOperatorNegation'],
+ excludedMutators: ['@ArithmeticOperator']
+```
+When making these customization tweaks, it is recommended to test the efficacy and performance of these tweaks against the base level to see whether they make a significant enough effect.
+
+## How to reason about modifying a mutation run
+
+A mutation run can be modified such that either the execution time is faster, or the number of covered tests is higher. Unfortunately, these two properties cannot occur at the same time, since they are inversely proportional: as the performance of a run increases, fewer test cases will be executed, which will result in lower coverage.
+
+### Using predefined levels
+
+Stryker pre-defines a few mutation levels such that they provide an attractive range of efficiency-performance tradeoffs. A multitude of Javascript/TypeScript projects was used for designing these levels, and chances are that they will be suitable for your project as well. For this reason, restricting a mutation run by using a predefined level should be the first option that you should consider when you desire to gain additional performance.
+
+### Customized configuration
+
+However, it could be that the predefined levels do not provide suitable results for your project, and you need to further customize the configuration. Although the previous sections specify the syntax for including/excluding mutators, they do not provide the intuition on how to exactly pick the most suitable choices.
+
+To find these best choices, we will use an external tool called [Callisto](https://github.com/stryker-mutator/callisto), which is used to quantify the resolution and the performance impact of mutation operators. This is a CLI tool that takes as input the JSON mutation report generated by Stryker and outputs a CSV file with several statistics (quality, performance impact, mutant count, etc) for each mutation operator. Callisto can be used in the following manner:
+
+
+1. Generate JSON Report with Stryker
+
+ Currently, only StrykerJS generates a JSON report which is directly compatible with Callisto. To ensure that Stryker has the correct configuration for generating the JSON file, you need to make sure that the following options are selected in the configuration:
+
+ ```json
+ {
+ ...
+ "disableBail": true,
+ "coverageAnalysis": "perTest",
+ "reporters": [..., "json", ...]
+ ...
+ }
+ ```
+
+2. (Optionally) Correct the JSON report
+
+ Some testing frameworks that StrykerJS uses might result in occasionally small mistakes, which prevents Callisto from deducing a mutation operator name. If there are any such mistakes, Callisto will report any mutants for which it cannot determine a mutation operator name through the terminal. The JSON report needs to be manually corrected with the reported errors before moving on to the next step.
+
+3. Run Callisto on the JSON file to determine statistics for each mutation operator. For details on how to do this, please refer to the documentation of the Callisto tool.
+
+4. Inspect the results
+
+Using the resolution and performance impact metrics, you can make decisions about which mutator/groups to include or exclude from your project.
+
+For example, if you would like to shorten the time a mutation run takes, take a look at the performance metric and add to the `excludedMutations` list the mutators/groups with the highest performance impact.
+
+Similarly, you might want to execute more test cases. Then, you should look at the quality metric and add the mutators with the highest values to the `includedMutations` list. However, this metric needs to be inspected together with its mutation count; consider the scenario where an operator has a high-quality score but a small number of generated mutants. Then, the measurement is not very reliable.
+
+Note that if you are using one of the predefined levels, some mutators might be already included or excluded.
\ No newline at end of file
diff --git a/e2e/test/ignore-project/stryker.conf.json b/e2e/test/ignore-project/stryker.conf.json
index 98d24896e6..4317c54823 100644
--- a/e2e/test/ignore-project/stryker.conf.json
+++ b/e2e/test/ignore-project/stryker.conf.json
@@ -4,7 +4,8 @@
"concurrency": 2,
"coverageAnalysis": "perTest",
"mutator": {
- "excludedMutations": ["ArithmeticOperator", "BlockStatement"]
+ "includedMutations": ["@StringLiteral", "@ConditionalExpression", "@EqualityOperator", "@LogicalOperator", "@BooleanLiteral"],
+ "excludedMutations": ["@ArithmeticOperator", "BlockStatementRemoval"]
},
"reporters": [
"clear-text",
diff --git a/e2e/test/ignore-project/verify/verify.js b/e2e/test/ignore-project/verify/verify.js
index e2e3b1d997..50cfba88d3 100644
--- a/e2e/test/ignore-project/verify/verify.js
+++ b/e2e/test/ignore-project/verify/verify.js
@@ -30,14 +30,37 @@ describe('After running stryker on jest-react project', () => {
});
});
- it('should report mutants that result from excluded mutators with the correct ignore reason', async () => {
+ it('should report mutants that are excluded from the excludedMutation list with the correct ignore reason', async () => {
const report = await readMutationTestingJsonResult();
const circleResult = report.files['src/Circle.js'];
const mutantsAtLine3 = circleResult.mutants.filter(({ location }) => location.start.line === 3);
expect(mutantsAtLine3).lengthOf(2);
mutantsAtLine3.forEach((mutant) => {
expect(mutant.status).eq('Ignored');
- expect(mutant.statusReason).eq('Ignored because of excluded mutation "ArithmeticOperator"');
+ expect(mutant.statusReason).eq('Ignored because the operator "MultiplicationOperatorNegation" is excluded from the mutation run');
+ });
+ });
+
+ it('should report mutants that are excluded because they were not in the includedMutations list', async () => {
+ const report = await readMutationTestingJsonResult();
+ const addResult = report.files['src/Add.js'];
+ const mutantsAtLine7 = addResult.mutants.filter(({ location }) => location.start.line === 7);
+ const updateOperatorMutants = mutantsAtLine7.filter(({ mutatorName }) => mutatorName === 'UpdateOperator');
+
+ const mutantsAtLine14 = addResult.mutants.filter(({ location }) => location.start.line === 14);
+ const unaryOperatorMutants = mutantsAtLine14.filter(({ mutatorName }) => mutatorName === 'UnaryOperator');
+
+ expect(updateOperatorMutants).lengthOf(1);
+ expect(unaryOperatorMutants).lengthOf(1);
+
+ updateOperatorMutants.forEach((updateMutant) => {
+ expect(updateMutant.status).eq('Ignored');
+ expect(updateMutant.statusReason).eq('Ignored because the operator "PostfixIncrementOperatorNegation" is excluded from the mutation run');
+ });
+
+ unaryOperatorMutants.forEach((updateMutant) => {
+ expect(updateMutant.status).eq('Ignored');
+ expect(updateMutant.statusReason).eq('Ignored because the operator "UnaryMinOperatorNegation" is excluded from the mutation run');
});
});
diff --git a/e2e/test/ignore-project/verify/verify.js.snap b/e2e/test/ignore-project/verify/verify.js.snap
index 8e0ae76987..b6248d3de6 100644
--- a/e2e/test/ignore-project/verify/verify.js.snap
+++ b/e2e/test/ignore-project/verify/verify.js.snap
@@ -3,20 +3,20 @@
exports[`After running stryker on jest-react project should report expected scores 1`] = `
Object {
"compileErrors": 0,
- "ignored": 32,
- "killed": 8,
- "mutationScore": 50,
+ "ignored": 34,
+ "killed": 6,
+ "mutationScore": 42.857142857142854,
"mutationScoreBasedOnCoveredCode": 100,
"noCoverage": 8,
"pending": 0,
"runtimeErrors": 0,
"survived": 0,
"timeout": 0,
- "totalCovered": 8,
- "totalDetected": 8,
+ "totalCovered": 6,
+ "totalDetected": 6,
"totalInvalid": 0,
"totalMutants": 48,
"totalUndetected": 8,
- "totalValid": 16,
+ "totalValid": 14,
}
`;
diff --git a/package-lock.json b/package-lock.json
index 9071e29b92..08c4a81f0b 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -11512,6 +11512,18 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
+ "node_modules/find-up-simple": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/find-up-simple/-/find-up-simple-1.0.0.tgz",
+ "integrity": "sha512-q7Us7kcjj2VMePAa02hDAF6d+MzsdsAWEwYyOpwUtlerRBkOEPBCRZrAV4XfcSN8fHAgaD0hP7miwoay6DCprw==",
+ "dev": true,
+ "engines": {
+ "node": ">=18"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
"node_modules/findup-sync": {
"version": "5.0.0",
"resolved": "https://registry.npmjs.org/findup-sync/-/findup-sync-5.0.0.tgz",
@@ -21349,11 +21361,8 @@
"resolved": "https://registry.npmjs.org/ignore-walk/-/ignore-walk-6.0.4.tgz",
"integrity": "sha512-t7sv42WkwFkyKbivUCglsQW5YWMskWtbEf4MNKX5u/CCWHKSPzN4FtBQGsQZgCLbxOzpVlcbWVK5KB3auIOjSw==",
"dev": true,
- "dependencies": {
- "minimatch": "^9.0.0"
- },
"engines": {
- "node": "^14.17.0 || ^16.13.0 || >=18.0.0"
+ "node": "14 || >=16.14"
}
},
"node_modules/pacote/node_modules/json-parse-even-better-errors": {
@@ -21370,6 +21379,12 @@
"resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.2.0.tgz",
"integrity": "sha512-2bIM8x+VAf6JT4bKAljS1qUWgMsqZRPGJS6FSahIMPVvctcNhyVp7AJu7quxOW9jwkryBReKZY5tY5JYv2n/7Q==",
"dev": true,
+ "dependencies": {
+ "hosted-git-info": "^7.0.0",
+ "is-core-module": "^2.8.1",
+ "semver": "^7.3.5",
+ "validate-npm-package-license": "^3.0.4"
+ },
"engines": {
"node": "14 || >=16.14"
}
diff --git a/packages/api/schema/stryker-core.json b/packages/api/schema/stryker-core.json
index 91069d4695..9c68cb39fd 100644
--- a/packages/api/schema/stryker-core.json
+++ b/packages/api/schema/stryker-core.json
@@ -218,12 +218,20 @@
],
"default": null
},
- "excludedMutations": {
+ "includedMutations": {
"type": "array",
+ "uniqueItems": true,
"items": {
- "type": "string"
+ "$ref": "#/definitions/MutationSpecification"
},
- "default": []
+ "minItems": 1
+ },
+ "excludedMutations": {
+ "type": "array",
+ "uniqueItems": true,
+ "items": {
+ "$ref": "#/definitions/MutationSpecification"
+ }
}
}
},
@@ -253,6 +261,452 @@
"default": true
}
}
+ },
+ "MutationSpecification": {
+ "anyOf": [
+ { "$ref": "#/definitions/MutationLevelName"},
+ { "$ref": "#/definitions/MutatorDefinition"},
+ { "$ref": "#/definitions/MutatorGroupName"}
+ ]
+ },
+ "MutationLevelName": {
+ "title": "MutationLevelName",
+ "type": "string"
+ },
+ "MutatorDefinition" : {
+ "anyOf": [
+ { "$ref": "#/definitions/ArithmeticOperator" },
+ { "$ref": "#/definitions/ArrayDeclaration" },
+ { "$ref": "#/definitions/ArrowFunction"},
+ { "$ref": "#/definitions/AssignmentOperator" },
+ { "$ref": "#/definitions/BlockStatement" },
+ { "$ref": "#/definitions/BooleanLiteral" },
+ { "$ref": "#/definitions/ConditionalExpression" },
+ { "$ref": "#/definitions/EqualityOperator" },
+ { "$ref": "#/definitions/LogicalOperator"},
+ { "$ref": "#/definitions/MethodExpression" },
+ { "$ref": "#/definitions/ObjectLiteral" },
+ { "$ref": "#/definitions/OptionalChaining" },
+ { "$ref": "#/definitions/Regex" },
+ { "$ref": "#/definitions/StringLiteral" },
+ { "$ref": "#/definitions/UnaryOperator" },
+ { "$ref": "#/definitions/UpdateOperator" }
+ ]
+ },
+ "MutatorGroupName": {
+ "enum": [
+ "@ArithmeticOperator",
+ "@ArrayDeclaration",
+ "@AssignmentOperator",
+ "@BooleanLiteral",
+ "@ConditionalExpression",
+ "@EqualityOperator",
+ "@MethodExpression",
+ "@ObjectLiteral",
+ "@OptionalChaining",
+ "@StringLiteral",
+ "@UnaryOperator",
+ "@UpdateOperator"
+ ]
+ },
+ "ArithmeticOperator": {
+ "title": "ArithmeticOperator",
+ "anyOf": [
+ {
+ "const" : "AdditionOperatorNegation",
+ "description": "Replace ```a + b``` with ```a - b```."
+ },
+ {
+ "const" : "DivisionOperatorNegation",
+ "description": "Replace ```a / b``` with ```a * b```."
+ },
+ {
+ "const" : "MultiplicationOperatorNegation",
+ "description": "Replace ```a * b``` with ```a / b```."
+ },
+ {
+ "const" : "RemainderOperatorToMultiplicationReplacement",
+ "description": "Replace ```a % b``` with ```a * b```."
+ },
+ {
+ "const" : "SubtractionOperatorNegation",
+ "description": "Replace ```a - b``` with ```a + b```."
+ }
+ ]
+ },
+ "ArrayDeclaration": {
+ "title": "ArrayDeclaration",
+ "anyOf": [
+ {
+ "const": "ArrayConstructorItemsFill",
+ "description": "Replace ```new Array()``` with ```new Array(Stryker was here)```."
+ },
+ {
+ "const": "ArrayConstructorItemsRemoval",
+ "description": "Replace ```new Array([1, 2, 3, 4])``` with ```new Array()```."
+ },
+ {
+ "const": "ArrayLiteralItemsFill",
+ "description": "Replace ```[ ]``` with ```[Stryker was here]```."
+ },
+ {
+ "const": "ArrayLiteralItemsRemoval",
+ "description": "Replace ```[1, 2, 3, 4]``` with ```[ ]```."
+ }
+ ]
+ },
+ "AssignmentOperator": {
+ "title": "AssignmentOperator",
+ "anyOf": [
+ {
+ "const" : "AdditionAssignmentNegation",
+ "description": "Replace ```a += b``` with ```a -= b```."
+ },
+ {
+ "const" : "BitwiseAndAssignmentToBitwiseOrReplacement",
+ "description": "Replace ```a &= b``` with ```a |= b```."
+ },
+ {
+ "const" : "BitwiseOrAssignmentToBitwiseAndReplacement",
+ "description": "Replace ```a |= b``` with ```a &= b```."
+ },
+ {
+ "const" : "DivisionAssignmentNegation",
+ "description": "Replace ```a /= b``` with ```a *= b```."
+ },
+ {
+ "const" : "LeftShiftAssignmentNegation",
+ "description": "Replace ```a <<= b``` with ```a >>= b```."
+ },
+ {
+ "const" : "LogicalAndAssignmentToLogicalOrReplacement",
+ "description": "Replace ```a &&= b``` with ```a ||= b```."
+ },
+ {
+ "const" : "LogicalOrAssignmentToLogicalAndReplacement",
+ "description": "Replace ```a ||= b``` with ```a &&= b```."
+ },
+ {
+ "const" : "MultiplicationAssignmentNegation",
+ "description": "Replace ```a *= b``` with ```a /= b```."
+ },
+ {
+ "const" : "NullishCoalescingAssignmentToLogicalAndReplacement",
+ "description": "Replace ```a ??= b``` with ```a &&= b```."
+ },
+ {
+ "const" : "RemainderAssignmentToMultiplicationReplacement",
+ "description": "Replace ```a %= b``` with ```a *= b```."
+ },
+ {
+ "const" : "RightShiftAssignmentNegation",
+ "description": "Replace ```a >>= b``` with ```a <<= b```."
+ },
+ {
+ "const" : "SubtractionAssignmentNegation",
+ "description": "Replace ```a -= b``` with ```a += b```."
+ }
+ ]
+ },
+ "ArrowFunction": {
+ "const": "ArrowFunctionRemoval",
+ "description": "Mutates bodies of arrow functions to undefined"
+ },
+ "BlockStatement": {
+ "const": "BlockStatementRemoval",
+ "description": "Removes the content of every block statement."
+ },
+ "BooleanLiteral": {
+ "title": "BooleanLiteral",
+ "anyOf": [
+ {
+ "const" : "FalseLiteralNegation",
+ "description": "Replace ```false``` with ```true```."
+ },
+ {
+ "const" : "LogicalNotRemoval",
+ "description": "Replace ```!(a == b)``` with ```a == b```."
+ },
+ {
+ "const" : "TrueLiteralNegation",
+ "description": "Replace ```true``` with ```false```."
+ }
+ ]
+ },
+ "ConditionalExpression": {
+ "title": "ConditionalExpression",
+ "anyOf": [
+ {
+ "const" : "BooleanExpressionToFalseReplacement",
+ "description": "Replace ```var x = a > b ? 1 : 2;``` with ```var x = false ? 1 : 2;```."
+ },
+ {
+ "const" : "BooleanExpressionToTrueReplacement",
+ "description": "Replace ```var x = a > b ? 1 : 2;``` with ```var x = true ? 1 : 2;```."
+ },
+ {
+ "const" : "DoWhileLoopConditionToFalseReplacement",
+ "description": "Replace ```do { } while (a > b);``` with ```do { } while (false);```."
+ },
+ {
+ "const" : "ForLoopConditionToFalseReplacement",
+ "description": "Replace ```for (var i = 0; i < 10; i++) { }``` with ```for (var i = 0; false; i++) { }```."
+ },
+ {
+ "const" : "IfConditionToFalseReplacement",
+ "description": "Replace ```if (a > b) { }``` with ```if (false) { }```."
+ },
+ {
+ "const" : "IfConditionToTrueReplacement",
+ "description": "Replace ```if (a > b) { }``` with ```if (true) { }```."
+ },
+ {
+ "const" : "SwitchStatementBodyRemoval",
+ "description": "Replace ```switch(x) { case 1: doSomething(); default: default(); } with switch(x) { case 1: default: default(); }```."
+ },
+ {
+ "const" : "WhileLoopConditionToFalseReplacement",
+ "description": "Replace ```while (a > b) { }``` with ```while (false) { }```."
+ }
+ ]
+ },
+ "EqualityOperator": {
+ "title": "EqualityOperator",
+ "anyOf": [
+ {
+ "const" : "EqualityOperatorNegation",
+ "description": "Replace ```a == b``` with ```a != b```."
+ },
+ {
+ "const" : "GreaterThanEqualOperatorBoundary",
+ "description": "Replace ```a >= b``` with ```a > b```."
+ },
+ {
+ "const" : "GreaterThanEqualOperatorNegation",
+ "description": "Replace ```a >= b``` with ```a < b```."
+ },
+ {
+ "const" : "GreaterThanOperatorBoundary",
+ "description": "Replace ```a > b``` with ```a >= b```."
+ },
+ {
+ "const" : "GreaterThanOperatorNegation",
+ "description": "Replace ```a > b``` with ```a <= b```."
+ },
+ {
+ "const" : "InequalityOperatorNegation",
+ "description": "Replace ```a != b``` with ```a == b```."
+ },
+ {
+ "const" : "LessThanEqualOperatorBoundary",
+ "description": "Replace ```a <= b``` with ```a < b```."
+ },
+ {
+ "const" : "LessThanEqualOperatorNegation",
+ "description": "Replace ```a <= b``` with ```a > b```."
+ },
+ {
+ "const" : "LessThanOperatorBoundary",
+ "description": "Replace ```a < b``` with ```a <= b```."
+ },
+ {
+ "const" : "LessThanOperatorNegation",
+ "description": "Replace ```a < b``` with ```a >= b```."
+ },
+ {
+ "const" : "StrictEqualityOperatorNegation",
+ "description": "Replace ```a === b``` with ```a !== b```."
+ },
+ {
+ "const" : "StrictInequalityOperatorNegation",
+ "description": "Replace ```a !== b``` with ```a === b```."
+ }
+ ]
+ },
+ "LogicalOperator": {
+ "title": "LogicalOperator",
+ "anyOf": [
+ {
+ "const" : "LogicalAndOperatorToLogicalOrReplacement",
+ "description": "Replace ```a && b``` with ```a || b```."
+ },
+ {
+ "const" : "LogicalOrOperatorToLogicalAndReplacement",
+ "description": "Replace ```a || b``` with ```a && b```."
+ },
+ {
+ "const" : "NullishCoalescingOperatorToLogicalAndReplacement",
+ "description": "Replace ```a ?? b``` with ```a && b```."
+ }
+ ]
+ },
+ "MethodExpression": {
+ "title": "MethodExpression",
+ "anyOf": [
+ {
+ "const": "CharAtMethodCallRemoval",
+ "description": "Remove ```charAt()``` call."
+ },
+ {
+ "const": "EndsWithMethodCallNegation",
+ "description": "Replace ```endsWith()``` with ```startsWith()```."
+ },
+ {
+ "const": "EveryMethodCallToSomeReplacement",
+ "description": "Replace ```every()``` with ```some()```."
+ },
+ {
+ "const": "FilterMethodCallRemoval",
+ "description": "Remove ```filter()``` call."
+ },
+ {
+ "const": "MaxMethodCallNegation",
+ "description": "Replace ```max()``` with ```min()```."
+ },
+ {
+ "const": "MinMethodCallNegation",
+ "description": "Replace ```min()``` with ```max()```."
+ },
+ {
+ "const": "ReverseMethodCallRemoval",
+ "description": "Remove ```reverse()``` call"
+ },
+ {
+ "const": "SliceMethodCallRemoval",
+ "description": "Remove ```slice()``` call."
+ },
+ {
+ "const": "SomeMethodCallToEveryReplacement",
+ "description": "Replace ```some()``` with ```every()```."
+ },
+ {
+ "const": "SortMethodCallRemoval",
+ "description": "Remove ```sort()``` call."
+ },
+ {
+ "const": "StartsWithMethodCallNegation",
+ "description": "Replace ```startsWith()``` with ```endsWith()```."
+ },
+ {
+ "const": "SubstringMethodCallRemoval",
+ "description": "Remove ```substring()``` call."
+ },
+ {
+ "const": "SubstrMethodCallRemoval",
+ "description": "Remove ```substr()``` call."
+ },
+ {
+ "const": "ToLocaleLowerCaseMethodCallNegation",
+ "description": "Replace ```toLocaleLowerCase()``` with ```toLocaleUpperCase()```."
+ },
+ {
+ "const": "ToLocaleUpperCaseMethodCallNegation",
+ "description": "Replace ```toLocaleUpperCase()``` with ```toLocaleLowerCase()```."
+ },
+ {
+ "const": "ToLowerCaseMethodCallNegation",
+ "description": "Replace ```toLowerCase()``` with ```toUpperCase()```."
+ },
+ {
+ "const": "ToUpperCaseMethodCallNegation",
+ "description": "Replace ```toUpperCase()``` with ```toLowerCase()```."
+ },
+ {
+ "const": "TrimEndMethodCallNegation",
+ "description": "Replace ```trimEnd()``` with ```trimStart()```."
+ },
+ {
+ "const": "TrimMethodCallRemoval",
+ "description": "Remove ```trim()``` call."
+ },
+ {
+ "const": "TrimStartMethodCallNegation",
+ "description": "Replace ```trimStart()``` with ```trimEnd()```."
+ }
+ ]
+ },
+ "ObjectLiteral": {
+ "const": "ObjectLiteralPropertiesRemoval",
+ "description": "Replace ```{ foo: 'bar' }``` with ```{ }```."
+ },
+ "OptionalChaining": {
+ "title": "OptionalChaining",
+ "anyOf": [
+ {
+ "const": "OptionalCallExpressionOptionalRemoval",
+ "description": "Replace ```foo?.()``` with ```foo()```."
+ },
+ {
+ "const": "OptionalComputedMemberExpressionOptionalRemoval",
+ "description": "Replace ```foo?.[1]``` with ```foo[1]```."
+ },
+ {
+ "const": "OptionalMemberExpressionOptionalRemoval",
+ "description": "Replace ```foo?.bar``` with ```foo.bar```."
+ }
+ ]
+ },
+ "Regex": {
+ "const": "RegexRemoval"
+ },
+ "StringLiteral": {
+ "title": "StringLiteral",
+ "anyOf": [
+ {
+ "const": "EmptyInterpolatedStringToFilledReplacement",
+ "description": "Replace ```s\"\"``` with ```s\"Stryker was here!\"```."
+ },
+ {
+ "const": "EmptyStringLiteralToFilledReplacement",
+ "description": "Replace ```\"\"``` with ```\"Stryker was here!\"```."
+ },
+ {
+ "const": "FilledInterpolatedStringToEmptyReplacement",
+ "description": "Replace ```s\"foo ${bar}\"``` with ```s\"\"```."
+ },
+ {
+ "const": "FilledStringLiteralToEmptyReplacement",
+ "description": "Replace ```\"foo\"``` with ```\"\"```."
+ }
+ ]
+ },
+ "UnaryOperator": {
+ "title": "UnaryOperator",
+ "anyOf": [
+ {
+ "const": "UnaryBitwiseNotRemoval",
+ "description": "Remove ```~``` from ```~a```."
+ },
+ {
+ "const": "UnaryMinOperatorNegation",
+ "description": "Replace ```-a``` with ```+a.```"
+ },
+ {
+ "const": "UnaryPlusOperatorNegation",
+ "description": "Replace ```+a``` with ```-a```."
+ }
+ ]
+ },
+ "UpdateOperator": {
+ "title": "UpdateOperator",
+ "anyOf": [
+ {
+ "const": "PostfixDecrementOperatorNegation",
+ "description": "Replace ```a--``` with ```a++```."
+ },
+ {
+ "const": "PostfixIncrementOperatorNegation",
+ "description": "Replace ```a++``` with ```a--```."
+ },
+ {
+ "const": "PrefixDecrementOperatorNegation",
+ "description": "Replace ```--a``` with ```++a```."
+ },
+ {
+ "const": "PrefixIncrementOperatorNegation",
+ "description": "Replace ```++a``` with ```--a```."
+ }
+ ]
}
},
"properties": {
diff --git a/packages/core/src/process/2-mutant-instrumenter-executor.ts b/packages/core/src/process/2-mutant-instrumenter-executor.ts
index cc4d7413e9..fa68729842 100644
--- a/packages/core/src/process/2-mutant-instrumenter-executor.ts
+++ b/packages/core/src/process/2-mutant-instrumenter-executor.ts
@@ -49,7 +49,10 @@ export class MutantInstrumenterExecutor {
// Instrument files in-memory
const ignorers = this.options.ignorers.map((name) => this.pluginCreator.create(PluginKind.Ignore, name));
- const instrumentResult = await instrumenter.instrument(await this.readFilesToMutate(), { ignorers, ...this.options.mutator });
+ const instrumentResult = await instrumenter.instrument(await this.readFilesToMutate(), {
+ ignorers,
+ ...this.options.mutator,
+ });
// Preprocess the project
const preprocess = this.injector.injectFunction(createPreprocessor);
diff --git a/packages/core/test/unit/config/options-validator.spec.ts b/packages/core/test/unit/config/options-validator.spec.ts
index bfcbfd62f7..4d836d6171 100644
--- a/packages/core/test/unit/config/options-validator.spec.ts
+++ b/packages/core/test/unit/config/options-validator.spec.ts
@@ -79,7 +79,6 @@ describe(OptionsValidator.name, () => {
'!{src,lib}/**/__tests__/**/*.+(cjs|mjs|js|ts|jsx|tsx|html|vue|svelte)',
],
mutator: {
- excludedMutations: [],
plugins: null,
},
plugins: ['@stryker-mutator/*'],
diff --git a/packages/core/test/unit/reporters/clear-text-score-table.spec.ts b/packages/core/test/unit/reporters/clear-text-score-table.spec.ts
index 45c496c7b6..0eb22dbd31 100644
--- a/packages/core/test/unit/reporters/clear-text-score-table.spec.ts
+++ b/packages/core/test/unit/reporters/clear-text-score-table.spec.ts
@@ -28,6 +28,7 @@ describe(ClearTextScoreTable.name, () => {
runtimeErrors: 4,
survived: 3,
timeout: 2,
+ totalMutants: 1,
}),
);
const sut = new ClearTextScoreTable(metricsResult, testInjector.options);
diff --git a/packages/instrumenter/src/mutation-level/default-mutation-levels.ts b/packages/instrumenter/src/mutation-level/default-mutation-levels.ts
new file mode 100644
index 0000000000..79cda601e5
--- /dev/null
+++ b/packages/instrumenter/src/mutation-level/default-mutation-levels.ts
@@ -0,0 +1,48 @@
+import { MutationLevel } from './mutation-level.js';
+
+const Level1: MutationLevel = {
+ name: 'Level1',
+ UpdateOperator: ['PrefixDecrementOperatorNegation'],
+ EqualityOperator: [
+ 'LessThanEqualOperatorNegation',
+ 'LessThanEqualOperatorBoundary',
+ 'EqualityOperatorNegation',
+ 'InequalityOperatorNegation',
+ 'GreaterThanEqualOperatorNegation',
+ ],
+ ArrayDeclaration: ['ArrayConstructorItemsRemoval'],
+ ConditionalExpression: ['BooleanExpressionToFalseReplacement', 'BooleanExpressionToTrueReplacement'],
+ UnaryOperator: ['UnaryPlusOperatorNegation'],
+ AssignmentOperator: ['NullishCoalescingAssignmentToLogicalAndReplacement'],
+ ArithmeticOperator: ['DivisionOperatorNegation', 'RemainderOperatorToMultiplicationReplacement', 'MultiplicationOperatorNegation'],
+ OptionalChaining: ['OptionalCallExpressionOptionalRemoval', 'OptionalMemberExpressionOptionalRemoval'],
+};
+
+const Level2: MutationLevel = {
+ ...Level1,
+ name: 'Level2',
+ UpdateOperator: [...(Level1.UpdateOperator ?? []), 'PostfixIncrementOperatorNegation'],
+ EqualityOperator: [
+ ...(Level1.EqualityOperator ?? []),
+ 'LessThanOperatorNegation',
+ 'GreaterThanEqualOperatorBoundary',
+ 'StrictInequalityOperatorNegation',
+ 'GreaterThanOperatorBoundary',
+ ],
+ ConditionalExpression: [...(Level1.ConditionalExpression ?? []), 'SwitchStatementBodyRemoval'],
+ ArithmeticOperator: [...(Level1.ArithmeticOperator ?? []), 'AdditionOperatorNegation', 'SubtractionOperatorNegation'],
+ StringLiteral: ['EmptyStringLiteralToFilledReplacement', 'EmptyInterpolatedStringToFilledReplacement'],
+ Regex: ['RegexRemoval'],
+ BooleanLiteral: ['TrueLiteralNegation'],
+};
+
+const Level3: MutationLevel = {
+ ...Level2,
+ name: 'Level3',
+ EqualityOperator: [...(Level2.EqualityOperator ?? []), 'LessThanOperatorBoundary', 'GreaterThanOperatorNegation'],
+ ArrayDeclaration: [...(Level2.ArrayDeclaration ?? []), 'ArrayLiteralItemsRemoval', 'ArrayLiteralItemsFill'],
+ UnaryOperator: [...(Level2.UnaryOperator ?? []), 'UnaryMinOperatorNegation'],
+ BooleanLiteral: [...(Level2.BooleanLiteral ?? []), 'FalseLiteralNegation', 'LogicalNotRemoval'],
+};
+
+export const defaultMutationLevels: MutationLevel[] = [Level1, Level2, Level3];
diff --git a/packages/instrumenter/src/mutation-level/mutation-level.ts b/packages/instrumenter/src/mutation-level/mutation-level.ts
new file mode 100644
index 0000000000..3d034c63fd
--- /dev/null
+++ b/packages/instrumenter/src/mutation-level/mutation-level.ts
@@ -0,0 +1,66 @@
+import { Node } from '@babel/core';
+import {
+ ArithmeticOperator,
+ ArrayDeclaration,
+ ArrowFunction,
+ AssignmentOperator,
+ BlockStatement,
+ BooleanLiteral,
+ ConditionalExpression,
+ EqualityOperator,
+ LogicalOperator,
+ MethodExpression,
+ MutatorDefinition,
+ ObjectLiteral,
+ OptionalChaining,
+ Regex,
+ StringLiteral,
+ UnaryOperator,
+ UpdateOperator,
+} from '@stryker-mutator/api/core';
+
+/**
+ * A Record that maps a replaceable fragment of code to a ReplacementConfiguration.
+ */
+export type NodeMutatorConfiguration = Record>;
+
+/**
+ * Consists of a replacement, or none if it removes the fragment; and a name for the mutation.
+ */
+interface ReplacementConfiguration {
+ /**
+ * Replacement for the fragment of code. ``undefined`` signifies removal of the fragment.
+ */
+ replacement?: Node | Node[] | boolean | string | null;
+ /**
+ * Name of the mutation.
+ */
+ mutationOperator: T;
+}
+
+/**
+ * Mutation Level. Has a name, and optionally a list of allowed mutations grouped per node type.
+ */
+export interface MutationLevel {
+ /**
+ * Name of the mutation level.
+ */
+ name: string;
+ ArithmeticOperator?: ArithmeticOperator[];
+ ArrayDeclaration?: ArrayDeclaration[];
+ AssignmentOperator?: AssignmentOperator[];
+ ArrowFunction?: ArrowFunction[];
+ BlockStatement?: BlockStatement[];
+ BooleanLiteral?: BooleanLiteral[];
+ ConditionalExpression?: ConditionalExpression[];
+ EqualityOperator?: EqualityOperator[];
+ LogicalOperator?: LogicalOperator[];
+ MethodExpression?: MethodExpression[];
+ ObjectLiteral?: ObjectLiteral[];
+ OptionalChaining?: OptionalChaining[];
+ Regex?: Regex[];
+ StringLiteral?: StringLiteral[];
+ UnaryOperator?: UnaryOperator[];
+ UpdateOperator?: UpdateOperator[];
+ [k: string]: MutatorDefinition[] | string | undefined;
+}
diff --git a/packages/instrumenter/src/mutators/arithmetic-operator-mutator.ts b/packages/instrumenter/src/mutators/arithmetic-operator-mutator.ts
index 18c21e58fc..dfaee2c562 100644
--- a/packages/instrumenter/src/mutators/arithmetic-operator-mutator.ts
+++ b/packages/instrumenter/src/mutators/arithmetic-operator-mutator.ts
@@ -1,32 +1,34 @@
import type { types } from '@babel/core';
+import { ArithmeticOperator } from '@stryker-mutator/api/core';
+
import { deepCloneNode } from '../util/index.js';
import { NodeMutator } from './node-mutator.js';
-const arithmeticOperatorReplacements = Object.freeze({
- '+': '-',
- '-': '+',
- '*': '/',
- '/': '*',
- '%': '*',
-} as const);
-
-export const arithmeticOperatorMutator: NodeMutator = {
+export const arithmeticOperatorMutator: NodeMutator = {
name: 'ArithmeticOperator',
+ operators: {
+ '+': { replacement: '-', mutationOperator: 'AdditionOperatorNegation' },
+ '-': { replacement: '+', mutationOperator: 'SubtractionOperatorNegation' },
+ '*': { replacement: '/', mutationOperator: 'MultiplicationOperatorNegation' },
+ '/': { replacement: '*', mutationOperator: 'DivisionOperatorNegation' },
+ '%': { replacement: '*', mutationOperator: 'RemainderOperatorToMultiplicationReplacement' },
+ },
+
*mutate(path) {
if (path.isBinaryExpression() && isSupported(path.node.operator, path.node)) {
- const mutatedOperator = arithmeticOperatorReplacements[path.node.operator];
- const replacement = deepCloneNode(path.node);
- replacement.operator = mutatedOperator;
- yield replacement;
+ const { replacement, mutationOperator } = this.operators[path.node.operator];
+ const nodeClone = deepCloneNode(path.node);
+ nodeClone.operator = replacement as types.BinaryExpression['operator'];
+ yield [nodeClone, mutationOperator];
}
},
};
-function isSupported(operator: string, node: types.BinaryExpression): operator is keyof typeof arithmeticOperatorReplacements {
- if (!Object.keys(arithmeticOperatorReplacements).includes(operator)) {
+function isSupported(operator: string, node: types.BinaryExpression): boolean {
+ if (!Object.keys(arithmeticOperatorMutator.operators).includes(operator)) {
return false;
}
diff --git a/packages/instrumenter/src/mutators/array-declaration-mutator.ts b/packages/instrumenter/src/mutators/array-declaration-mutator.ts
index e6a2e9ec59..ef7f89d84b 100644
--- a/packages/instrumenter/src/mutators/array-declaration-mutator.ts
+++ b/packages/instrumenter/src/mutators/array-declaration-mutator.ts
@@ -1,4 +1,6 @@
-import babel, { type NodePath } from '@babel/core';
+import babel, { Node } from '@babel/core';
+
+import { ArrayDeclaration } from '@stryker-mutator/api/core';
import { deepCloneNode } from '../util/index.js';
@@ -6,20 +8,36 @@ import { NodeMutator } from './node-mutator.js';
const { types } = babel;
-export const arrayDeclarationMutator: NodeMutator = {
+export const arrayDeclarationMutator: NodeMutator = {
name: 'ArrayDeclaration',
- *mutate(path: NodePath): Iterable {
+ operators: {
+ ArrayLiteralItemsFill: {
+ replacement: types.arrayExpression([types.stringLiteral('Stryker was here')]),
+ mutationOperator: 'ArrayLiteralItemsFill',
+ },
+ ArrayConstructorItemsFill: { replacement: [types.stringLiteral('Stryker was here')], mutationOperator: 'ArrayConstructorItemsFill' },
+ ArrayLiteralItemsRemoval: { replacement: types.arrayExpression(), mutationOperator: 'ArrayLiteralItemsRemoval' },
+ ArrayConstructorItemsRemoval: { replacement: [], mutationOperator: 'ArrayConstructorItemsRemoval' },
+ },
+
+ *mutate(path) {
+ // The check of the [] construct in code
if (path.isArrayExpression()) {
- const replacement = path.node.elements.length ? types.arrayExpression() : types.arrayExpression([types.stringLiteral('Stryker was here')]);
- yield replacement;
+ const { replacement, mutationOperator } =
+ path.node.elements.length > 0 ? this.operators.ArrayLiteralItemsRemoval : this.operators.ArrayLiteralItemsFill;
+ yield [replacement as Node, mutationOperator];
}
+ // Check for the new Array() construct in code
if ((path.isCallExpression() || path.isNewExpression()) && types.isIdentifier(path.node.callee) && path.node.callee.name === 'Array') {
- const mutatedCallArgs = path.node.arguments.length ? [] : [types.arrayExpression()];
- const replacement = types.isNewExpression(path.node)
- ? types.newExpression(deepCloneNode(path.node.callee), mutatedCallArgs)
- : types.callExpression(deepCloneNode(path.node.callee), mutatedCallArgs);
- yield replacement;
+ const { replacement, mutationOperator } =
+ path.node.arguments.length > 0 ? this.operators.ArrayConstructorItemsRemoval : this.operators.ArrayConstructorItemsFill;
+
+ const nodeClone = types.isNewExpression(path.node)
+ ? types.newExpression(deepCloneNode(path.node.callee), replacement as babel.types.Expression[])
+ : types.callExpression(deepCloneNode(path.node.callee), replacement as babel.types.Expression[]);
+
+ yield [nodeClone, mutationOperator];
}
},
};
diff --git a/packages/instrumenter/src/mutators/arrow-function-mutator.ts b/packages/instrumenter/src/mutators/arrow-function-mutator.ts
index 7ed6a37ddb..999a63622d 100644
--- a/packages/instrumenter/src/mutators/arrow-function-mutator.ts
+++ b/packages/instrumenter/src/mutators/arrow-function-mutator.ts
@@ -2,18 +2,24 @@ import babel from '@babel/core';
const { types } = babel;
+import { ArrowFunction } from '@stryker-mutator/api/core';
+
import { NodeMutator } from './index.js';
-export const arrowFunctionMutator: NodeMutator = {
+export const arrowFunctionMutator: NodeMutator = {
name: 'ArrowFunction',
+ operators: {
+ ArrowFunctionRemoval: { mutationOperator: 'ArrowFunctionRemoval' },
+ },
+
*mutate(path) {
if (
path.isArrowFunctionExpression() &&
!types.isBlockStatement(path.node.body) &&
!(types.isIdentifier(path.node.body) && path.node.body.name === 'undefined')
) {
- yield types.arrowFunctionExpression([], types.identifier('undefined'));
+ yield [types.arrowFunctionExpression([], types.identifier('undefined')), this.operators.ArrowFunctionRemoval.mutationOperator];
}
},
};
diff --git a/packages/instrumenter/src/mutators/assignment-operator-mutator.ts b/packages/instrumenter/src/mutators/assignment-operator-mutator.ts
index 3f17374ca0..a6e8f77502 100644
--- a/packages/instrumenter/src/mutators/assignment-operator-mutator.ts
+++ b/packages/instrumenter/src/mutators/assignment-operator-mutator.ts
@@ -1,45 +1,47 @@
-import type { types as t } from '@babel/core';
+import type { types } from '@babel/core';
+
+import { AssignmentOperator } from '@stryker-mutator/api/core';
import { deepCloneNode } from '../util/index.js';
import { NodeMutator } from './index.js';
-const assignmentOperatorReplacements = Object.freeze({
- '+=': '-=',
- '-=': '+=',
- '*=': '/=',
- '/=': '*=',
- '%=': '*=',
- '<<=': '>>=',
- '>>=': '<<=',
- '&=': '|=',
- '|=': '&=',
- '&&=': '||=',
- '||=': '&&=',
- '??=': '&&=',
-} as const);
-
const stringTypes = Object.freeze(['StringLiteral', 'TemplateLiteral']);
const stringAssignmentTypes = Object.freeze(['&&=', '||=', '??=']);
-export const assignmentOperatorMutator: NodeMutator = {
+export const assignmentOperatorMutator: NodeMutator = {
name: 'AssignmentOperator',
+ operators: {
+ '+=': { replacement: '-=', mutationOperator: 'AdditionAssignmentNegation' },
+ '-=': { replacement: '+=', mutationOperator: 'SubtractionAssignmentNegation' },
+ '*=': { replacement: '/=', mutationOperator: 'MultiplicationAssignmentNegation' },
+ '/=': { replacement: '*=', mutationOperator: 'DivisionAssignmentNegation' },
+ '%=': { replacement: '*=', mutationOperator: 'RemainderAssignmentToMultiplicationReplacement' },
+ '<<=': { replacement: '>>=', mutationOperator: 'LeftShiftAssignmentNegation' },
+ '>>=': { replacement: '<<=', mutationOperator: 'RightShiftAssignmentNegation' },
+ '&=': { replacement: '|=', mutationOperator: 'BitwiseAndAssignmentToBitwiseOrReplacement' },
+ '|=': { replacement: '&=', mutationOperator: 'BitwiseOrAssignmentToBitwiseAndReplacement' },
+ '&&=': { replacement: '||=', mutationOperator: 'LogicalAndAssignmentToLogicalOrReplacement' },
+ '||=': { replacement: '&&=', mutationOperator: 'LogicalOrAssignmentToLogicalAndReplacement' },
+ '??=': { replacement: '&&=', mutationOperator: 'NullishCoalescingAssignmentToLogicalAndReplacement' },
+ },
+
*mutate(path) {
if (path.isAssignmentExpression() && isSupportedAssignmentOperator(path.node.operator) && isSupported(path.node)) {
- const mutatedOperator = assignmentOperatorReplacements[path.node.operator];
- const replacement = deepCloneNode(path.node);
- replacement.operator = mutatedOperator;
- yield replacement;
+ const { replacement, mutationOperator } = this.operators[path.node.operator];
+ const nodeClone = deepCloneNode(path.node);
+ nodeClone.operator = replacement as string;
+ yield [nodeClone, mutationOperator];
}
},
};
-function isSupportedAssignmentOperator(operator: string): operator is keyof typeof assignmentOperatorReplacements {
- return Object.keys(assignmentOperatorReplacements).includes(operator);
+function isSupportedAssignmentOperator(operator: string): boolean {
+ return Object.keys(assignmentOperatorMutator.operators).includes(operator);
}
-function isSupported(node: t.AssignmentExpression): boolean {
+function isSupported(node: types.AssignmentExpression): boolean {
// Excludes assignment operators that apply to strings.
if (stringTypes.includes(node.right.type) && !stringAssignmentTypes.includes(node.operator)) {
return false;
diff --git a/packages/instrumenter/src/mutators/block-statement-mutator.ts b/packages/instrumenter/src/mutators/block-statement-mutator.ts
index adfeed8cfa..724c1ff306 100644
--- a/packages/instrumenter/src/mutators/block-statement-mutator.ts
+++ b/packages/instrumenter/src/mutators/block-statement-mutator.ts
@@ -1,15 +1,21 @@
import babel, { type NodePath } from '@babel/core';
+import { BlockStatement } from '@stryker-mutator/api/core';
+
import { NodeMutator } from './node-mutator.js';
const { types } = babel;
-export const blockStatementMutator: NodeMutator = {
+export const blockStatementMutator: NodeMutator = {
name: 'BlockStatement',
+ operators: {
+ BlockStatementRemoval: { mutationOperator: 'BlockStatementRemoval' },
+ },
+
*mutate(path) {
if (path.isBlockStatement() && isValid(path)) {
- yield types.blockStatement([]);
+ yield [types.blockStatement([]), this.operators.BlockStatementRemoval.mutationOperator];
}
},
};
diff --git a/packages/instrumenter/src/mutators/boolean-literal-mutator.ts b/packages/instrumenter/src/mutators/boolean-literal-mutator.ts
index c339c99b3d..6b4cc6ac29 100644
--- a/packages/instrumenter/src/mutators/boolean-literal-mutator.ts
+++ b/packages/instrumenter/src/mutators/boolean-literal-mutator.ts
@@ -1,20 +1,30 @@
import babel from '@babel/core';
+import { BooleanLiteral } from '@stryker-mutator/api/core';
+
import { deepCloneNode } from '../util/index.js';
const { types } = babel;
import { NodeMutator } from './index.js';
-export const booleanLiteralMutator: NodeMutator = {
+export const booleanLiteralMutator: NodeMutator = {
name: 'BooleanLiteral',
+ operators: {
+ true: { replacement: false, mutationOperator: 'TrueLiteralNegation' },
+ false: { replacement: true, mutationOperator: 'FalseLiteralNegation' },
+ '!': { replacement: '', mutationOperator: 'LogicalNotRemoval' },
+ },
+
*mutate(path) {
if (path.isBooleanLiteral()) {
- yield types.booleanLiteral(!path.node.value);
+ const { replacement, mutationOperator } = path.node.value ? this.operators.true : this.operators.false;
+
+ yield [types.booleanLiteral(replacement as boolean), mutationOperator];
}
if (path.isUnaryExpression() && path.node.operator === '!' && path.node.prefix) {
- yield deepCloneNode(path.node.argument);
+ yield [deepCloneNode(path.node.argument), this.operators['!'].mutationOperator];
}
},
};
diff --git a/packages/instrumenter/src/mutators/conditional-expression-mutator.ts b/packages/instrumenter/src/mutators/conditional-expression-mutator.ts
index dd0fab6833..33227235a1 100644
--- a/packages/instrumenter/src/mutators/conditional-expression-mutator.ts
+++ b/packages/instrumenter/src/mutators/conditional-expression-mutator.ts
@@ -1,4 +1,6 @@
-import babel, { type NodePath } from '@babel/core';
+import babel, { Node, type NodePath } from '@babel/core';
+
+import { ConditionalExpression } from '@stryker-mutator/api/core';
import { deepCloneNode } from '../util/index.js';
@@ -8,43 +10,98 @@ const booleanOperators = Object.freeze(['!=', '!==', '&&', '<', '<=', '==', '===
const { types } = babel;
-export const conditionalExpressionMutator: NodeMutator = {
+export const conditionalExpressionMutator: NodeMutator = {
name: 'ConditionalExpression',
+ operators: {
+ BooleanExpressionToFalseReplacement: {
+ replacement: types.booleanLiteral(false),
+ mutationOperator: 'BooleanExpressionToFalseReplacement',
+ },
+ BooleanExpressionToTrueReplacement: {
+ replacement: types.booleanLiteral(true),
+ mutationOperator: 'BooleanExpressionToTrueReplacement',
+ },
+ DoWhileLoopConditionToFalseReplacement: {
+ replacement: types.booleanLiteral(false),
+ mutationOperator: 'DoWhileLoopConditionToFalseReplacement',
+ },
+ ForLoopConditionToFalseReplacement: {
+ replacement: types.booleanLiteral(false),
+ mutationOperator: 'ForLoopConditionToFalseReplacement',
+ },
+ IfConditionToFalseReplacement: {
+ replacement: types.booleanLiteral(false),
+ mutationOperator: 'IfConditionToFalseReplacement',
+ },
+ IfConditionToTrueReplacement: {
+ replacement: types.booleanLiteral(true),
+ mutationOperator: 'IfConditionToTrueReplacement',
+ },
+ WhileLoopConditionToFalseReplacement: {
+ replacement: types.booleanLiteral(false),
+ mutationOperator: 'WhileLoopConditionToFalseReplacement',
+ },
+ SwitchStatementBodyRemoval: { replacement: [], mutationOperator: 'SwitchStatementBodyRemoval' },
+ },
+
*mutate(path) {
if (isTestOfLoop(path)) {
- yield types.booleanLiteral(false);
+ if (isTestOfWhileLoop(path)) {
+ const { replacement, mutationOperator } = this.operators.WhileLoopConditionToFalseReplacement;
+ yield [replacement as Node, mutationOperator];
+ }
+
+ if (isTestOfDoWhileLoop(path)) {
+ const { replacement, mutationOperator } = this.operators.DoWhileLoopConditionToFalseReplacement;
+ yield [replacement as Node, mutationOperator];
+ }
+ if (isTestOfForLoop(path)) {
+ const { replacement, mutationOperator } = this.operators.ForLoopConditionToFalseReplacement;
+ yield [replacement as Node, mutationOperator];
+ }
} else if (isTestOfCondition(path)) {
- yield types.booleanLiteral(true);
- yield types.booleanLiteral(false);
+ yield [this.operators.IfConditionToTrueReplacement.replacement as Node, this.operators.IfConditionToTrueReplacement.mutationOperator];
+ yield [this.operators.IfConditionToFalseReplacement.replacement as Node, this.operators.IfConditionToFalseReplacement.mutationOperator];
} else if (isBooleanExpression(path)) {
if (path.parent?.type === 'LogicalExpression') {
// For (x || y), do not generate the (true || y) mutation as it
// has the same behavior as the (true) mutator, handled in the
// isTestOfCondition branch above
if (path.parent.operator === '||') {
- yield types.booleanLiteral(false);
+ const { replacement, mutationOperator } = this.operators.BooleanExpressionToFalseReplacement;
+ yield [replacement as Node, mutationOperator];
return;
}
// For (x && y), do not generate the (false && y) mutation as it
// has the same behavior as the (false) mutator, handled in the
// isTestOfCondition branch above
if (path.parent.operator === '&&') {
- yield types.booleanLiteral(true);
+ const { replacement, mutationOperator } = this.operators.BooleanExpressionToTrueReplacement;
+ yield [replacement as Node, mutationOperator];
return;
}
}
- yield types.booleanLiteral(true);
- yield types.booleanLiteral(false);
+ yield [
+ this.operators.BooleanExpressionToTrueReplacement.replacement as Node,
+ this.operators.BooleanExpressionToTrueReplacement.mutationOperator,
+ ];
+ yield [
+ this.operators.BooleanExpressionToFalseReplacement.replacement as Node,
+ this.operators.BooleanExpressionToFalseReplacement.mutationOperator,
+ ];
} else if (path.isForStatement() && !path.node.test) {
- const replacement = deepCloneNode(path.node);
- replacement.test = types.booleanLiteral(false);
- yield replacement;
+ const nodeClone = deepCloneNode(path.node);
+ const { replacement, mutationOperator } = this.operators.ForLoopConditionToFalseReplacement;
+ nodeClone.test = replacement as babel.types.Expression;
+ yield [nodeClone, mutationOperator];
} else if (path.isSwitchCase() && path.node.consequent.length > 0) {
// if not a fallthrough case
- const replacement = deepCloneNode(path.node);
- replacement.consequent = [];
- yield replacement;
+ const nodeClone = deepCloneNode(path.node);
+ const { replacement, mutationOperator } = this.operators.SwitchStatementBodyRemoval;
+
+ nodeClone.consequent = replacement as babel.types.Statement[];
+ yield [nodeClone, mutationOperator];
}
},
};
@@ -57,6 +114,21 @@ function isTestOfLoop(path: NodePath): boolean {
return (parentPath.isForStatement() || parentPath.isWhileStatement() || parentPath.isDoWhileStatement()) && parentPath.node.test === path.node;
}
+function isTestOfWhileLoop(path: NodePath): boolean {
+ const { parentPath } = path;
+ return parentPath !== null && parentPath && parentPath.isWhileStatement() && parentPath.node.test === path.node;
+}
+
+function isTestOfForLoop(path: NodePath): boolean {
+ const { parentPath } = path;
+ return parentPath !== null && parentPath && parentPath.isForStatement() && parentPath.node.test === path.node;
+}
+
+function isTestOfDoWhileLoop(path: NodePath): boolean {
+ const { parentPath } = path;
+ return parentPath !== null && parentPath.isDoWhileStatement() && parentPath.node.test === path.node;
+}
+
function isTestOfCondition(path: NodePath): boolean {
const { parentPath } = path;
if (!parentPath) {
diff --git a/packages/instrumenter/src/mutators/equality-operator-mutator.ts b/packages/instrumenter/src/mutators/equality-operator-mutator.ts
index 0aace5eb0d..b1facfaef7 100644
--- a/packages/instrumenter/src/mutators/equality-operator-mutator.ts
+++ b/packages/instrumenter/src/mutators/equality-operator-mutator.ts
@@ -1,33 +1,48 @@
import babel from '@babel/core';
+import { EqualityOperator } from '@stryker-mutator/api/core';
+
import { NodeMutator } from './node-mutator.js';
const { types: t } = babel;
-const operators = {
- '<': ['<=', '>='],
- '<=': ['<', '>'],
- '>': ['>=', '<='],
- '>=': ['>', '<'],
- '==': ['!='],
- '!=': ['=='],
- '===': ['!=='],
- '!==': ['==='],
-} as const;
-
-function isEqualityOperator(operator: string): operator is keyof typeof operators {
- return Object.keys(operators).includes(operator);
-}
-export const equalityOperatorMutator: NodeMutator = {
+export const equalityOperatorMutator: NodeMutator = {
name: 'EqualityOperator',
+ operators: {
+ '=': { replacement: '>=', mutationOperator: 'LessThanOperatorNegation' },
+
+ '<=To<': { replacement: '<', mutationOperator: 'LessThanEqualOperatorBoundary' },
+ '<=To>': { replacement: '>', mutationOperator: 'LessThanEqualOperatorNegation' },
+
+ '>To>=': { replacement: '>=', mutationOperator: 'GreaterThanOperatorBoundary' },
+ '>To<=': { replacement: '<=', mutationOperator: 'GreaterThanOperatorNegation' },
+
+ '>=To>': { replacement: '>', mutationOperator: 'GreaterThanEqualOperatorBoundary' },
+ '>=To<': { replacement: '<', mutationOperator: 'GreaterThanEqualOperatorNegation' },
+
+ '==To!=': { replacement: '!=', mutationOperator: 'EqualityOperatorNegation' },
+ '!=To==': { replacement: '==', mutationOperator: 'InequalityOperatorNegation' },
+ '===To!==': { replacement: '!==', mutationOperator: 'StrictEqualityOperatorNegation' },
+ '!==To===': { replacement: '===', mutationOperator: 'StrictInequalityOperatorNegation' },
+ },
+
*mutate(path) {
if (path.isBinaryExpression() && isEqualityOperator(path.node.operator)) {
- for (const mutableOperator of operators[path.node.operator]) {
- const replacement = t.cloneNode(path.node, true);
- replacement.operator = mutableOperator;
- yield replacement;
+ const allMutations = Object.keys(equalityOperatorMutator.operators)
+ .filter((k) => k.startsWith(path.node.operator + 'To'))
+ .map((k) => equalityOperatorMutator.operators[k]);
+
+ for (const mutableOperator of allMutations) {
+ const nodeClone = t.cloneNode(path.node, true);
+ nodeClone.operator = mutableOperator.replacement as babel.types.BinaryExpression['operator'];
+ yield [nodeClone, mutableOperator.mutationOperator];
}
}
},
};
+
+function isEqualityOperator(operator: string): operator is keyof typeof equalityOperatorMutator.operators {
+ return Object.keys(equalityOperatorMutator.operators).some((k) => k.startsWith(operator + 'To'));
+}
diff --git a/packages/instrumenter/src/mutators/logical-operator-mutator.ts b/packages/instrumenter/src/mutators/logical-operator-mutator.ts
index 57b239e98e..a2d7f85498 100644
--- a/packages/instrumenter/src/mutators/logical-operator-mutator.ts
+++ b/packages/instrumenter/src/mutators/logical-operator-mutator.ts
@@ -1,27 +1,29 @@
+import { LogicalOperator } from '@stryker-mutator/api/core';
+
import { deepCloneNode } from '../util/index.js';
import { NodeMutator } from './index.js';
-const logicalOperatorReplacements = Object.freeze({
- '&&': '||',
- '||': '&&',
- '??': '&&',
-} as const);
-
-export const logicalOperatorMutator: NodeMutator = {
+export const logicalOperatorMutator: NodeMutator = {
name: 'LogicalOperator',
+ operators: {
+ '&&': { replacement: '||', mutationOperator: 'LogicalAndOperatorToLogicalOrReplacement' },
+ '||': { replacement: '&&', mutationOperator: 'LogicalOrOperatorToLogicalAndReplacement' },
+ '??': { replacement: '&&', mutationOperator: 'NullishCoalescingOperatorToLogicalAndReplacement' },
+ },
+
*mutate(path) {
if (path.isLogicalExpression() && isSupported(path.node.operator)) {
- const mutatedOperator = logicalOperatorReplacements[path.node.operator];
+ const { replacement, mutationOperator } = this.operators[path.node.operator];
- const replacement = deepCloneNode(path.node);
- replacement.operator = mutatedOperator;
- yield replacement;
+ const nodeClone = deepCloneNode(path.node);
+ nodeClone.operator = replacement as babel.types.LogicalExpression['operator'];
+ yield [nodeClone, mutationOperator];
}
},
};
-function isSupported(operator: string): operator is keyof typeof logicalOperatorReplacements {
- return Object.keys(logicalOperatorReplacements).includes(operator);
+function isSupported(operator: string): operator is keyof typeof logicalOperatorMutator.operators {
+ return Object.keys(logicalOperatorMutator.operators).includes(operator);
}
diff --git a/packages/instrumenter/src/mutators/method-expression-mutator.ts b/packages/instrumenter/src/mutators/method-expression-mutator.ts
index 56731bd17e..0567c066b5 100644
--- a/packages/instrumenter/src/mutators/method-expression-mutator.ts
+++ b/packages/instrumenter/src/mutators/method-expression-mutator.ts
@@ -1,38 +1,42 @@
import babel from '@babel/core';
-import { deepCloneNode } from '../util/syntax-helpers.js';
+import { MethodExpression } from '@stryker-mutator/api/core';
+
+import { deepCloneNode } from '../util/index.js';
import { NodeMutator } from './node-mutator.js';
const { types } = babel;
-const replacements = new Map([
- ['charAt', null],
- ['endsWith', 'startsWith'],
- ['every', 'some'],
- ['filter', null],
- ['reverse', null],
- ['slice', null],
- ['sort', null],
- ['substr', null],
- ['substring', null],
- ['toLocaleLowerCase', 'toLocaleUpperCase'],
- ['toLowerCase', 'toUpperCase'],
- ['trim', null],
- ['trimEnd', 'trimStart'],
- ['min', 'max'],
-]);
-
-for (const [key, value] of Array.from(replacements)) {
- if (value) {
- replacements.set(value, key);
- }
-}
-
-export const methodExpressionMutator: NodeMutator = {
+export const methodExpressionMutator: NodeMutator = {
name: 'MethodExpression',
+ operators: {
+ charAt: { replacement: null, mutationOperator: 'CharAtMethodCallRemoval' },
+ endsWith: { replacement: 'startsWith', mutationOperator: 'EndsWithMethodCallNegation' },
+ startsWith: { replacement: 'endsWith', mutationOperator: 'StartsWithMethodCallNegation' },
+ every: { replacement: 'some', mutationOperator: 'EveryMethodCallToSomeReplacement' },
+ some: { replacement: 'every', mutationOperator: 'SomeMethodCallToEveryReplacement' },
+ filter: { replacement: null, mutationOperator: 'FilterMethodCallRemoval' },
+ reverse: { replacement: null, mutationOperator: 'ReverseMethodCallRemoval' },
+ slice: { replacement: null, mutationOperator: 'SliceMethodCallRemoval' },
+ sort: { replacement: null, mutationOperator: 'SortMethodCallRemoval' },
+ substr: { replacement: null, mutationOperator: 'SubstrMethodCallRemoval' },
+ substring: { replacement: null, mutationOperator: 'SubstringMethodCallRemoval' },
+ toLocaleLowerCase: { replacement: 'toLocaleUpperCase', mutationOperator: 'ToLocaleLowerCaseMethodCallNegation' },
+ toLocaleUpperCase: { replacement: 'toLocaleLowerCase', mutationOperator: 'ToLocaleUpperCaseMethodCallNegation' },
+ toLowerCase: { replacement: 'toUpperCase', mutationOperator: 'ToLowerCaseMethodCallNegation' },
+ toUpperCase: { replacement: 'toLowerCase', mutationOperator: 'ToUpperCaseMethodCallNegation' },
+ trim: { replacement: null, mutationOperator: 'TrimMethodCallRemoval' },
+ trimEnd: { replacement: 'trimStart', mutationOperator: 'TrimEndMethodCallNegation' },
+ trimStart: { replacement: 'trimEnd', mutationOperator: 'TrimStartMethodCallNegation' },
+ min: { replacement: 'max', mutationOperator: 'MinMethodCallNegation' },
+ max: { replacement: 'min', mutationOperator: 'MaxMethodCallNegation' },
+ },
+
*mutate(path) {
+ // In case `operations` is undefined, any checks will short-circuit to true and allow the mutation
+
if (!(path.isCallExpression() || path.isOptionalCallExpression())) {
return;
}
@@ -42,26 +46,33 @@ export const methodExpressionMutator: NodeMutator = {
return;
}
- const newName = replacements.get(callee.property.name);
- if (newName === undefined) {
- return;
- }
-
- if (newName === null) {
- // Remove the method expression. I.e. `foo.trim()` => `foo`
- yield deepCloneNode(callee.object);
+ const mutation = this.operators[callee.property.name];
+ if (mutation === undefined) {
+ // Function is not known in `operators`, so no mutations
return;
}
// Replace the method expression. I.e. `foo.toLowerCase()` => `foo.toUpperCase`
const nodeArguments = path.node.arguments.map((argumentNode) => deepCloneNode(argumentNode));
- const mutatedCallee = types.isMemberExpression(callee)
- ? types.memberExpression(deepCloneNode(callee.object), types.identifier(newName), false, callee.optional)
- : types.optionalMemberExpression(deepCloneNode(callee.object), types.identifier(newName), false, callee.optional);
+ let mutatedCallee = undefined;
+
+ if (mutation.replacement != null) {
+ mutatedCallee = types.isMemberExpression(callee)
+ ? types.memberExpression(deepCloneNode(callee.object), types.identifier(mutation.replacement as string), false, callee.optional)
+ : types.optionalMemberExpression(deepCloneNode(callee.object), types.identifier(mutation.replacement as string), false, callee.optional);
+ } else if (typeof mutation.replacement == 'object' && mutation.replacement == null) {
+ yield [deepCloneNode(callee.object), mutation.mutationOperator];
+ return;
+ }
- yield types.isCallExpression(path.node)
- ? types.callExpression(mutatedCallee, nodeArguments)
- : types.optionalCallExpression(mutatedCallee, nodeArguments, path.node.optional);
+ if (mutatedCallee != undefined) {
+ yield [
+ types.isCallExpression(path.node)
+ ? types.callExpression(mutatedCallee, nodeArguments)
+ : types.optionalCallExpression(mutatedCallee, nodeArguments, path.node.optional),
+ mutation.mutationOperator,
+ ];
+ }
},
};
diff --git a/packages/instrumenter/src/mutators/mutate.ts b/packages/instrumenter/src/mutators/mutate.ts
index 7350fead57..03c98645f2 100644
--- a/packages/instrumenter/src/mutators/mutate.ts
+++ b/packages/instrumenter/src/mutators/mutate.ts
@@ -1,3 +1,5 @@
+import { MutatorDefinition } from '@stryker-mutator/api/core';
+
import { arithmeticOperatorMutator } from './arithmetic-operator-mutator.js';
import { NodeMutator } from './node-mutator.js';
import { blockStatementMutator } from './block-statement-mutator.js';
@@ -16,7 +18,7 @@ import { regexMutator } from './regex-mutator.js';
import { optionalChainingMutator } from './optional-chaining-mutator.js';
import { assignmentOperatorMutator } from './assignment-operator-mutator.js';
-export const allMutators: NodeMutator[] = [
+export const allMutators: Array> = [
arithmeticOperatorMutator,
arrayDeclarationMutator,
arrowFunctionMutator,
diff --git a/packages/instrumenter/src/mutators/mutator-options.ts b/packages/instrumenter/src/mutators/mutator-options.ts
index 5c41bc6e3f..08c1a65689 100644
--- a/packages/instrumenter/src/mutators/mutator-options.ts
+++ b/packages/instrumenter/src/mutators/mutator-options.ts
@@ -1,4 +1,7 @@
+import { MutationSpecification } from '@stryker-mutator/api/core';
+
export interface MutatorOptions {
- excludedMutations: string[];
+ includedMutations?: MutationSpecification[];
+ excludedMutations?: MutationSpecification[];
noHeader?: boolean;
}
diff --git a/packages/instrumenter/src/mutators/node-mutator.ts b/packages/instrumenter/src/mutators/node-mutator.ts
index ab3318f9c4..b31d3b9f67 100644
--- a/packages/instrumenter/src/mutators/node-mutator.ts
+++ b/packages/instrumenter/src/mutators/node-mutator.ts
@@ -1,6 +1,20 @@
import type { types, NodePath } from '@babel/core';
-export interface NodeMutator {
- mutate(path: NodePath): Iterable;
+import { NodeMutatorConfiguration, MutationLevel } from '../mutation-level/mutation-level.js';
+
+export interface NodeMutator {
+ /**
+ * Generates the mutations that fit a given Node, restricted by the Mutation Level.
+ * @param path the NodePath to mutate.
+ * @param levelMutations the relevant group of allowed mutations in the Mutation Level. Allows all if undefined.
+ */
+ // It would be stricter for the type to be `MutatorDefinition` rather than `keyof MutationLevel` but that
+ // prevents the definition of custom mutators from {@link babel.transformer.spec.ts}
+ mutate(path: NodePath): Iterable<[types.Node, keyof MutationLevel]>;
readonly name: string;
+
+ /**
+ * A record of all possible mutations in a Node.
+ */
+ operators: NodeMutatorConfiguration;
}
diff --git a/packages/instrumenter/src/mutators/object-literal-mutator.ts b/packages/instrumenter/src/mutators/object-literal-mutator.ts
index 6bcbd0747b..fd6a657cb3 100644
--- a/packages/instrumenter/src/mutators/object-literal-mutator.ts
+++ b/packages/instrumenter/src/mutators/object-literal-mutator.ts
@@ -1,15 +1,21 @@
import babel from '@babel/core';
+import { ObjectLiteral } from '@stryker-mutator/api/core';
+
import { NodeMutator } from './index.js';
const { types } = babel;
-export const objectLiteralMutator: NodeMutator = {
+export const objectLiteralMutator: NodeMutator = {
name: 'ObjectLiteral',
+ operators: {
+ ObjectLiteralPropertiesRemoval: { mutationOperator: 'ObjectLiteralPropertiesRemoval' },
+ },
+
*mutate(path) {
if (path.isObjectExpression() && path.node.properties.length > 0) {
- yield types.objectExpression([]);
+ yield [types.objectExpression([]), this.operators.ObjectLiteralPropertiesRemoval.mutationOperator];
}
},
};
diff --git a/packages/instrumenter/src/mutators/optional-chaining-mutator.ts b/packages/instrumenter/src/mutators/optional-chaining-mutator.ts
index 23ea455616..a517611f7d 100644
--- a/packages/instrumenter/src/mutators/optional-chaining-mutator.ts
+++ b/packages/instrumenter/src/mutators/optional-chaining-mutator.ts
@@ -1,5 +1,7 @@
import babel from '@babel/core';
+import { OptionalChaining } from '@stryker-mutator/api/core';
+
import { NodeMutator } from './index.js';
const { types: t } = babel;
@@ -16,24 +18,40 @@ const { types: t } = babel;
* foo?.[1] -> foo[1]
* foo?.() -> foo()
*/
-export const optionalChainingMutator: NodeMutator = {
+
+export const optionalChainingMutator: NodeMutator = {
name: 'OptionalChaining',
+ operators: {
+ OptionalCallExpressionOptionalRemoval: { mutationOperator: 'OptionalCallExpressionOptionalRemoval' },
+ OptionalMemberExpressionOptionalRemoval: { mutationOperator: 'OptionalMemberExpressionOptionalRemoval' },
+ OptionalComputedMemberExpressionOptionalRemoval: { mutationOperator: 'OptionalComputedMemberExpressionOptionalRemoval' },
+ },
+
*mutate(path) {
if (path.isOptionalMemberExpression() && path.node.optional) {
- yield t.optionalMemberExpression(
- t.cloneNode(path.node.object, true),
- t.cloneNode(path.node.property, true),
- path.node.computed,
- /*optional*/ false,
- );
+ const mutationOperator = path.node.computed
+ ? this.operators.OptionalComputedMemberExpressionOptionalRemoval.mutationOperator
+ : this.operators.OptionalMemberExpressionOptionalRemoval.mutationOperator;
+ yield [
+ t.optionalMemberExpression(
+ t.cloneNode(path.node.object, true),
+ t.cloneNode(path.node.property, true),
+ path.node.computed,
+ /*optional*/ false,
+ ),
+ mutationOperator,
+ ];
}
if (path.isOptionalCallExpression() && path.node.optional) {
- yield t.optionalCallExpression(
- t.cloneNode(path.node.callee, true),
- path.node.arguments.map((arg) => t.cloneNode(arg, true)),
- /*optional*/ false,
- );
+ yield [
+ t.optionalCallExpression(
+ t.cloneNode(path.node.callee, true),
+ path.node.arguments.map((arg) => t.cloneNode(arg, true)),
+ /*optional*/ false,
+ ),
+ this.operators.OptionalCallExpressionOptionalRemoval.mutationOperator,
+ ];
}
},
};
diff --git a/packages/instrumenter/src/mutators/regex-mutator.ts b/packages/instrumenter/src/mutators/regex-mutator.ts
index 0c927f4278..6cdabac1dc 100644
--- a/packages/instrumenter/src/mutators/regex-mutator.ts
+++ b/packages/instrumenter/src/mutators/regex-mutator.ts
@@ -1,6 +1,8 @@
import babel, { NodePath, type types as t } from '@babel/core';
import * as weaponRegex from 'weapon-regex';
+import { Regex } from '@stryker-mutator/api/core';
+
import { NodeMutator } from './index.js';
const { types } = babel;
@@ -29,19 +31,23 @@ function getFlags(path: NodePath): string | undefined {
const weaponRegexOptions: weaponRegex.MutationOptions = { mutationLevels: [1] };
-export const regexMutator: NodeMutator = {
+export const regexMutator: NodeMutator = {
name: 'Regex',
+ operators: {
+ RegexRemoval: { mutationOperator: 'RegexRemoval' },
+ },
+
*mutate(path) {
if (path.isRegExpLiteral()) {
for (const replacementPattern of mutatePattern(path.node.pattern, path.node.flags)) {
const replacement = types.regExpLiteral(replacementPattern, path.node.flags);
- yield replacement;
+ yield [replacement, this.operators.RegexRemoval.mutationOperator];
}
} else if (path.isStringLiteral() && isObviousRegexString(path)) {
const flags = getFlags(path.parentPath as NodePath);
for (const replacementPattern of mutatePattern(path.node.value, flags)) {
- yield types.stringLiteral(replacementPattern);
+ yield [types.stringLiteral(replacementPattern), this.operators.RegexRemoval.mutationOperator];
}
}
},
diff --git a/packages/instrumenter/src/mutators/string-literal-mutator.ts b/packages/instrumenter/src/mutators/string-literal-mutator.ts
index 9b3dc4c03f..2925313c15 100644
--- a/packages/instrumenter/src/mutators/string-literal-mutator.ts
+++ b/packages/instrumenter/src/mutators/string-literal-mutator.ts
@@ -1,19 +1,51 @@
-import babel, { type NodePath } from '@babel/core';
+import babel, { Node, type NodePath } from '@babel/core';
+
+import { StringLiteral } from '@stryker-mutator/api/core';
import { NodeMutator } from './node-mutator.js';
const { types } = babel;
-export const stringLiteralMutator: NodeMutator = {
+export const stringLiteralMutator: NodeMutator = {
name: 'StringLiteral',
+ operators: {
+ EmptyStringLiteralToFilledReplacement: {
+ replacement: types.stringLiteral('Stryker was here!'),
+ mutationOperator: 'EmptyStringLiteralToFilledReplacement',
+ },
+ FilledStringLiteralToEmptyReplacement: {
+ replacement: types.stringLiteral(''),
+ mutationOperator: 'FilledStringLiteralToEmptyReplacement',
+ },
+ FilledInterpolatedStringToEmptyReplacement: {
+ replacement: types.templateLiteral([types.templateElement({ raw: '' })], []),
+ mutationOperator: 'FilledInterpolatedStringToEmptyReplacement',
+ },
+ EmptyInterpolatedStringToFilledReplacement: {
+ replacement: types.templateLiteral([types.templateElement({ raw: 'Stryker was here!' })], []),
+ mutationOperator: 'EmptyInterpolatedStringToFilledReplacement',
+ },
+ },
+
*mutate(path) {
if (path.isTemplateLiteral()) {
- const replacement = path.node.quasis.length === 1 && path.node.quasis[0].value.raw.length === 0 ? 'Stryker was here!' : '';
- yield types.templateLiteral([types.templateElement({ raw: replacement })], []);
+ const stringIsEmpty = path.node.quasis.length === 1 && path.node.quasis[0].value.raw.length === 0;
+
+ const { replacement, mutationOperator } = stringIsEmpty
+ ? this.operators.EmptyInterpolatedStringToFilledReplacement
+ : this.operators.FilledInterpolatedStringToEmptyReplacement;
+
+ yield [replacement as Node, mutationOperator];
}
if (path.isStringLiteral() && isValidParent(path)) {
- yield types.stringLiteral(path.node.value.length === 0 ? 'Stryker was here!' : '');
+ const stringIsEmpty = path.node.value.length === 0;
+
+ const { replacement, mutationOperator } = stringIsEmpty
+ ? this.operators.EmptyStringLiteralToFilledReplacement
+ : this.operators.FilledStringLiteralToEmptyReplacement;
+
+ yield [replacement as Node, mutationOperator];
}
},
};
diff --git a/packages/instrumenter/src/mutators/unary-operator-mutator.ts b/packages/instrumenter/src/mutators/unary-operator-mutator.ts
index 0c970dc8f0..06aa89b84f 100644
--- a/packages/instrumenter/src/mutators/unary-operator-mutator.ts
+++ b/packages/instrumenter/src/mutators/unary-operator-mutator.ts
@@ -1,32 +1,35 @@
import babel from '@babel/core';
+import { UnaryOperator } from '@stryker-mutator/api/core';
+
import { deepCloneNode } from '../util/index.js';
import { NodeMutator } from './index.js';
const { types } = babel;
-enum UnaryOperator {
- '+' = '-',
- '-' = '+',
- '~' = '',
-}
-
-export const unaryOperatorMutator: NodeMutator = {
+export const unaryOperatorMutator: NodeMutator = {
name: 'UnaryOperator',
+ operators: {
+ '+': { replacement: '-', mutationOperator: 'UnaryPlusOperatorNegation' },
+ '-': { replacement: '+', mutationOperator: 'UnaryMinOperatorNegation' },
+ '~': { replacement: '', mutationOperator: 'UnaryBitwiseNotRemoval' },
+ },
+
*mutate(path) {
if (path.isUnaryExpression() && isSupported(path.node.operator) && path.node.prefix) {
- const mutatedOperator = UnaryOperator[path.node.operator];
- const replacement = mutatedOperator.length
- ? types.unaryExpression(mutatedOperator as '-' | '+', deepCloneNode(path.node.argument))
+ const { replacement, mutationOperator } = this.operators[path.node.operator];
+
+ const nodeClone = (replacement as string).length
+ ? types.unaryExpression(replacement as '-' | '+', deepCloneNode(path.node.argument))
: deepCloneNode(path.node.argument);
- yield replacement;
+ yield [nodeClone, mutationOperator];
}
},
};
-function isSupported(operator: string): operator is keyof typeof UnaryOperator {
- return Object.keys(UnaryOperator).includes(operator);
+function isSupported(operator: string): operator is keyof typeof unaryOperatorMutator.operators {
+ return operator in unaryOperatorMutator.operators;
}
diff --git a/packages/instrumenter/src/mutators/update-operator-mutator.ts b/packages/instrumenter/src/mutators/update-operator-mutator.ts
index 970c83cae5..c36538a924 100644
--- a/packages/instrumenter/src/mutators/update-operator-mutator.ts
+++ b/packages/instrumenter/src/mutators/update-operator-mutator.ts
@@ -1,22 +1,46 @@
import babel from '@babel/core';
+import { UpdateOperator } from '@stryker-mutator/api/core';
+
import { deepCloneNode } from '../util/index.js';
import { NodeMutator } from './index.js';
const { types } = babel;
-enum UpdateOperators {
- '++' = '--',
- '--' = '++',
-}
-
-export const updateOperatorMutator: NodeMutator = {
+export const updateOperatorMutator: NodeMutator = {
name: 'UpdateOperator',
+ operators: {
+ PostfixIncrementOperatorNegation: {
+ replacement: '--',
+ mutationOperator: 'PostfixIncrementOperatorNegation',
+ },
+ PostfixDecrementOperatorNegation: {
+ replacement: '++',
+ mutationOperator: 'PostfixDecrementOperatorNegation',
+ },
+ PrefixIncrementOperatorNegation: {
+ replacement: '--',
+ mutationOperator: 'PrefixIncrementOperatorNegation',
+ },
+ PrefixDecrementOperatorNegation: {
+ replacement: '++',
+ mutationOperator: 'PrefixDecrementOperatorNegation',
+ },
+ },
+
*mutate(path) {
if (path.isUpdateExpression()) {
- yield types.updateExpression(UpdateOperators[path.node.operator], deepCloneNode(path.node.argument), path.node.prefix);
+ let operator;
+ if (path.node.operator === '++') {
+ operator = path.node.prefix ? this.operators.PrefixIncrementOperatorNegation : this.operators.PostfixIncrementOperatorNegation;
+ } else {
+ operator = path.node.prefix ? this.operators.PrefixDecrementOperatorNegation : this.operators.PostfixDecrementOperatorNegation;
+ }
+
+ const { replacement, mutationOperator } = operator;
+ yield [types.updateExpression(replacement as '--' | '++', deepCloneNode(path.node.argument), path.node.prefix), mutationOperator];
}
},
};
diff --git a/packages/instrumenter/src/transformers/babel-transformer.ts b/packages/instrumenter/src/transformers/babel-transformer.ts
index f2cedf90e4..f0033c2a14 100644
--- a/packages/instrumenter/src/transformers/babel-transformer.ts
+++ b/packages/instrumenter/src/transformers/babel-transformer.ts
@@ -5,12 +5,18 @@ import babel, { type NodePath, type types } from '@babel/core';
import { File } from '@babel/core';
/* eslint-enable import/no-duplicates */
+import { MutationSpecification, MutatorDefinition } from '@stryker-mutator/api/core';
+
import { isImportDeclaration, isTypeNode, locationIncluded, locationOverlaps, placeHeaderIfNeeded } from '../util/syntax-helpers.js';
import { ScriptFormat } from '../syntax/index.js';
import { allMutantPlacers, MutantPlacer, throwPlacementError } from '../mutant-placers/index.js';
import { Mutable, Mutant } from '../mutant.js';
import { allMutators } from '../mutators/index.js';
+import { MutationLevel } from '../mutation-level/mutation-level.js';
+
+import { defaultMutationLevels } from '../mutation-level/default-mutation-levels.js';
+
import { DirectiveBookkeeper } from './directive-bookkeeper.js';
import { IgnorerBookkeeper } from './ignorer-bookkeeper.js';
@@ -155,24 +161,104 @@ export const transformBabel: AstTransformer = (
* Generate mutants for the current node.
*/
function* mutate(node: NodePath): Iterable {
+ const runLevel = createRunLevel();
+
for (const mutator of mutators) {
- for (const replacement of mutator.mutate(node)) {
+ for (const [replacement, mutationOperator] of mutator.mutate(node)) {
yield {
replacement,
mutatorName: mutator.name,
ignoreReason:
directiveBookkeeper.findIgnoreReason(node.node.loc!.start.line, mutator.name) ??
- findExcludedMutatorIgnoreReason(mutator.name) ??
+ findExcludedMutatorIgnoreReason(runLevel, mutator.name, mutationOperator) ??
ignorerBookkeeper.currentIgnoreMessage,
};
}
}
+ }
- function findExcludedMutatorIgnoreReason(mutatorName: string): string | undefined {
- if (options.excludedMutations.includes(mutatorName)) {
- return `Ignored because of excluded mutation "${mutatorName}"`;
- } else {
+ function findExcludedMutatorIgnoreReason(
+ runLevel: MutationLevel | undefined,
+ mutatorName: string,
+ mutationOperator: keyof MutationLevel,
+ ): string | undefined {
+ if (runLevel === undefined) {
+ return;
+ }
+
+ if (!(mutatorName in runLevel)) {
+ return `Ignored because "${mutatorName}" is not recognised as a mutator`;
+ }
+
+ if (!runLevel[mutatorName]?.includes(mutationOperator as MutatorDefinition)) {
+ return `Ignored because the operator "${mutationOperator}" is excluded from the mutation run`;
+ }
+
+ return;
+ }
+
+ /**
+ * @returns `undefined` for the default stryker behaviour or a MutationLevel according to the specification
+ */
+ function createRunLevel(): MutationLevel | undefined {
+ const runLevel: MutationLevel = { name: 'RunningLevel' };
+ mutators.forEach((mut) => (runLevel[mut.name] = []));
+
+ if (options.includedMutations === undefined || options.includedMutations.length === 0) {
+ if (options.excludedMutations === undefined) {
+ // include everything
return undefined;
+ } else {
+ // remove `excludedMutations` from a complete level
+ mutators.forEach((mut) =>
+ Object.values(mut.operators).forEach((op) => (runLevel[mut.name] as MutatorDefinition[]).push(op.mutationOperator as MutatorDefinition)),
+ );
+ }
+ }
+
+ updateRunLevel(runLevel, options.includedMutations, true);
+ updateRunLevel(runLevel, options.excludedMutations, false);
+
+ return runLevel;
+ }
+
+ function updateRunLevel(runLevel: MutationLevel, mutations: MutationSpecification[] | undefined, includeMutations: boolean) {
+ if (mutations) {
+ const updateFunc: (mutatorList: MutatorDefinition[], ...toUpdate: MutatorDefinition[]) => void = includeMutations
+ ? (mutatorList, toAdd) => mutatorList.push(toAdd)
+ : (mutatorList, toRemove) => mutatorList.splice(0, mutatorList.length, ...mutatorList.filter((m) => !toRemove.includes(m))); // in-place filter
+
+ for (const spec of mutations) {
+ // Check if it's a mutation level
+ const defaultLevel = defaultMutationLevels.find((dl) => '@' + dl.name === spec);
+ if (defaultLevel) {
+ Object.keys(defaultLevel)
+ .filter((k) => k !== 'name')
+ .forEach((levelKey) => updateFunc(runLevel[levelKey] as MutatorDefinition[], ...(defaultLevel[levelKey] as MutatorDefinition[])));
+ continue;
+ }
+
+ // Check if it's a operator group
+ const opGroupName = Object.keys(runLevel).find((levelKey) => levelKey !== 'name' && '@' + levelKey === spec);
+ if (opGroupName) {
+ const nodeMutatorToAdd = mutators.find((mut) => mut.name === opGroupName);
+ if (nodeMutatorToAdd) {
+ Object.values(nodeMutatorToAdd.operators).forEach((mutator) => {
+ updateFunc(runLevel[opGroupName] as MutatorDefinition[], mutator.mutationOperator as MutatorDefinition);
+ });
+ continue;
+ }
+ }
+
+ // Else, must be a suboperator
+ const nodeMutator = mutators.find((mut) => Object.values(mut.operators).some((mutator) => mutator.mutationOperator === spec));
+
+ if (nodeMutator) {
+ updateFunc(runLevel[nodeMutator.name] as MutatorDefinition[], spec as MutatorDefinition);
+ continue;
+ }
+
+ logger.warn(`Mutation operator "${spec}" not recognised. Did you make a typo?`);
}
}
}
diff --git a/packages/instrumenter/src/transformers/directive-bookkeeper.ts b/packages/instrumenter/src/transformers/directive-bookkeeper.ts
index 6d724058da..15857acd47 100644
--- a/packages/instrumenter/src/transformers/directive-bookkeeper.ts
+++ b/packages/instrumenter/src/transformers/directive-bookkeeper.ts
@@ -4,6 +4,7 @@ import { notEmpty } from '@stryker-mutator/util';
import { Logger } from '@stryker-mutator/api/logging';
import { NodeMutator } from '../mutators/node-mutator.js';
+import { MutationLevel } from '../mutation-level/mutation-level.js';
const WILDCARD = 'all';
const DEFAULT_REASON = 'Ignored using a comment';
@@ -60,7 +61,7 @@ export class DirectiveBookkeeper {
constructor(
private readonly logger: Logger,
- private readonly allMutators: NodeMutator[],
+ private readonly allMutators: Array>,
private readonly originFileName: string,
) {
this.allMutatorNames = this.allMutators.map((x) => x.name.toLowerCase());
diff --git a/packages/instrumenter/test/helpers/expect-mutation.ts b/packages/instrumenter/test/helpers/expect-mutation.ts
index eb3554e6b1..d46ddd580e 100644
--- a/packages/instrumenter/test/helpers/expect-mutation.ts
+++ b/packages/instrumenter/test/helpers/expect-mutation.ts
@@ -4,6 +4,7 @@ import generator from '@babel/generator';
import { expect } from 'chai';
import { NodeMutator } from '../../src/mutators/node-mutator.js';
+import { MutationLevel } from '../../src/mutation-level/mutation-level.js';
const generate = generator.default;
@@ -35,7 +36,16 @@ const plugins = [
'typescript',
] as ParserPlugin[];
-export function expectJSMutation(sut: NodeMutator, originalCode: string, ...expectedReplacements: string[]): void {
+export function expectJSMutation(sut: NodeMutator, originalCode: string, ...expectedReplacements: string[]): void {
+ expectJSMutationWithLevel(sut, undefined, originalCode, ...expectedReplacements);
+}
+
+export function expectJSMutationWithLevel(
+ sut: NodeMutator,
+ level: string[] | undefined,
+ originalCode: string,
+ ...expectedReplacements: string[]
+): void {
const sourceFileName = 'source.js';
const ast = parse(originalCode, {
sourceFilename: sourceFileName,
@@ -47,20 +57,22 @@ export function expectJSMutation(sut: NodeMutator, originalCode: string, ...expe
babel.traverse(ast, {
enter(path) {
- for (const replacement of sut.mutate(path)) {
- const mutatedCode = generate(replacement).code;
- const beforeMutatedCode = originalCode.substring(0, path.node.start ?? 0);
- const afterMutatedCode = originalCode.substring(path.node.end ?? 0);
- const mutant = `${beforeMutatedCode}${mutatedCode}${afterMutatedCode}`;
- mutants.push(mutant);
+ for (const [replacement, mutationOperator] of sut.mutate(path)) {
+ if (level === undefined || level.includes(mutationOperator as string)) {
+ const mutatedCode = generate(replacement).code;
+ const beforeMutatedCode = originalCode.substring(0, path.node.start ?? 0);
+ const afterMutatedCode = originalCode.substring(path.node.end ?? 0);
+ const mutant = `${beforeMutatedCode}${mutatedCode}${afterMutatedCode}`;
+ mutants.push(mutant);
- for (const replacementNode of nodeSet(replacement, path)) {
- if (originalNodeSet.has(replacementNode)) {
- expect.fail(
- `Mutated ${replacementNode.type} node \`${
- generate(replacementNode).code
- }\` was found in the original AST. Please be sure to deep clone it (using \`cloneNode(ast, true)\`)`,
- );
+ for (const replacementNode of nodeSet(replacement, path)) {
+ if (originalNodeSet.has(replacementNode)) {
+ expect.fail(
+ `Mutated ${replacementNode.type} node \`${
+ generate(replacementNode).code
+ }\` was found in the original AST. Please be sure to deep clone it (using \`cloneNode(ast, true)\`)`,
+ );
+ }
}
}
}
diff --git a/packages/instrumenter/test/helpers/factories.ts b/packages/instrumenter/test/helpers/factories.ts
index 52fcc5fe34..8745db5dd2 100644
--- a/packages/instrumenter/test/helpers/factories.ts
+++ b/packages/instrumenter/test/helpers/factories.ts
@@ -17,7 +17,6 @@ export function createParserOptions(overrides?: Partial): ParserO
export function createTransformerOptions(overrides?: Partial): TransformerOptions {
return {
- excludedMutations: [],
ignorers: [],
...overrides,
};
diff --git a/packages/instrumenter/test/integration/instrumenter.it.spec.ts b/packages/instrumenter/test/integration/instrumenter.it.spec.ts
index d33f3c2a2c..3644b470ae 100644
--- a/packages/instrumenter/test/integration/instrumenter.it.spec.ts
+++ b/packages/instrumenter/test/integration/instrumenter.it.spec.ts
@@ -50,7 +50,10 @@ describe('instrumenter integration', () => {
await arrangeAndActAssert('shebang.js');
});
it('should not place excluded mutations', async () => {
- await arrangeAndActAssert('excluded-mutations.js', createInstrumenterOptions({ excludedMutations: ['ArithmeticOperator'] }));
+ await arrangeAndActAssert(
+ 'excluded-mutations.js',
+ createInstrumenterOptions({ excludedMutations: ['@Level1', '@ArithmeticOperator', 'AdditionOperatorNegation'] }),
+ );
});
it('should not place disabled mutants', async () => {
await arrangeAndActAssert('disabled.js');
diff --git a/packages/instrumenter/test/unit/instrumenter.spec.ts b/packages/instrumenter/test/unit/instrumenter.spec.ts
index 56265e7e0f..4a994a9d3c 100644
--- a/packages/instrumenter/test/unit/instrumenter.spec.ts
+++ b/packages/instrumenter/test/unit/instrumenter.spec.ts
@@ -64,9 +64,7 @@ describe(Instrumenter.name, () => {
// Assert
// eslint-disable-next-line @typescript-eslint/prefer-destructuring
const actual = helper.transformerStub.getCall(0).args[2];
- const expected: transformers.TransformerOptions = createInstrumenterOptions({
- excludedMutations: [],
- });
+ const expected: transformers.TransformerOptions = createInstrumenterOptions();
expect(actual).deep.eq({
options: expected,
mutateDescription: [{ start: { line: 1, column: 0 }, end: { line: 7, column: 42 } }],
diff --git a/packages/instrumenter/test/unit/mutators/arithmatic-operator-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/arithmatic-operator-mutator.spec.ts
index ecda7f8d17..829e185169 100644
--- a/packages/instrumenter/test/unit/mutators/arithmatic-operator-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/arithmatic-operator-mutator.spec.ts
@@ -1,7 +1,15 @@
import { expect } from 'chai';
import { arithmeticOperatorMutator as sut } from '../../../src/mutators/arithmetic-operator-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const arithmeticLevel: MutationLevel = {
+ name: 'ArithemticLevel',
+ ArithmeticOperator: ['AdditionOperatorNegation', 'SubtractionOperatorNegation', 'MultiplicationOperatorNegation'],
+};
+const arithmeticOperatorUndefinedLevel: MutationLevel = { name: 'ArithmeticOperatorLevel', ArithmeticOperator: [] };
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "ArithmeticOperator"', () => {
@@ -30,4 +38,35 @@ describe(sut.name, () => {
expectJSMutation(sut, '"a" + b + "c" + d + "e"');
});
+
+ describe('mutation level', () => {
+ it('should only mutate +, - and *', () => {
+ expectJSMutationWithLevel(
+ sut,
+ arithmeticLevel.ArithmeticOperator,
+ 'a + b; a - b; a * b; a % b; a / b; a % b',
+ 'a - b; a - b; a * b; a % b; a / b; a % b', // mutates +
+ 'a + b; a + b; a * b; a % b; a / b; a % b', // mutates -
+ 'a + b; a - b; a / b; a % b; a / b; a % b', // mutates *
+ );
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, arithmeticOperatorUndefinedLevel.ArithmeticOperator, 'a + b; a - b; a * b; a % b; a / b; a % b');
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ 'a + b; a - b; a * b; a % b; a / b; a % b',
+ 'a + b; a - b; a * b; a % b; a * b; a % b', // mutates /
+ 'a + b; a - b; a * b; a % b; a / b; a * b', // mutates %
+ 'a + b; a - b; a * b; a * b; a / b; a % b', // mutates %
+ 'a - b; a - b; a * b; a % b; a / b; a % b', // mutates +
+ 'a + b; a + b; a * b; a % b; a / b; a % b', // mutates -
+ 'a + b; a - b; a / b; a % b; a / b; a % b', // mutates *
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/array-declaration-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/array-declaration-mutator.spec.ts
index 235879fd2f..8678168744 100644
--- a/packages/instrumenter/test/unit/mutators/array-declaration-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/array-declaration-mutator.spec.ts
@@ -1,7 +1,19 @@
import { expect } from 'chai';
import { arrayDeclarationMutator as sut } from '../../../src/mutators/array-declaration-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const arrayLiteralLevel: MutationLevel = {
+ name: 'ArrayLiteralLevel',
+ ArrayDeclaration: ['ArrayLiteralItemsFill', 'ArrayLiteralItemsRemoval'],
+};
+const arrayConstructorLevel: MutationLevel = {
+ name: 'ArrayConstructorLevel',
+ ArrayDeclaration: ['ArrayConstructorItemsFill', 'ArrayConstructorItemsRemoval'],
+};
+const arrayDeclarationOperatorUndefinedLevel: MutationLevel = { name: 'ArrayDeclarationcOperatorOperatorLevel', ArrayDeclaration: [] };
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "ArrayDeclaration"', () => {
@@ -30,12 +42,54 @@ describe(sut.name, () => {
});
it('should mutate empty array constructor call as a filled array', () => {
- expectJSMutation(sut, 'new Array()', 'new Array([])');
- expectJSMutation(sut, 'Array()', 'Array([])');
+ expectJSMutation(sut, 'new Array()', 'new Array("Stryker was here")');
+ expectJSMutation(sut, 'Array()', 'Array("Stryker was here")');
});
it('should not mutate other function call expressions', () => {
expectJSMutation(sut, 'window.Array(21, 2)');
expectJSMutation(sut, 'window["Array"](21, 2)');
});
+
+ describe('mutation level', () => {
+ it('should only mutate [], [x,y]', () => {
+ expectJSMutationWithLevel(
+ sut,
+ arrayLiteralLevel.ArrayDeclaration,
+ '[]; new Array(); new Array({x:"", y:""}); [{x:"", y:""}]',
+ '["Stryker was here"]; new Array(); new Array({x:"", y:""}); [{x:"", y:""}]', // mutates []
+ '[]; new Array(); new Array({x:"", y:""}); []', // mutates [x,y]
+ );
+ });
+
+ it('should only mutate new Array(), new Array({x:"", y:""}))', () => {
+ expectJSMutationWithLevel(
+ sut,
+ arrayConstructorLevel.ArrayDeclaration,
+ '[]; new Array(); new Array({x:"", y:""}); [{x:"", y:""}]',
+ '[]; new Array("Stryker was here"); new Array({x:"", y:""}); [{x:"", y:""}]', // mutates new Array()
+ '[]; new Array(); new Array(); [{x:"", y:""}]', // mutates new Array(x,y)
+ );
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ arrayDeclarationOperatorUndefinedLevel.ArrayDeclaration,
+ '[]; new Array(); new Array({x:"", y:""}); [{x:"", y:""}]',
+ );
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ '[]; new Array(); new Array({x:"", y:""}); [{x:"", y:""}]',
+ '["Stryker was here"]; new Array(); new Array({x:"", y:""}); [{x:"", y:""}]', // mutates []
+ '[]; new Array("Stryker was here"); new Array({x:"", y:""}); [{x:"", y:""}]', // mutates new Array()
+ '[]; new Array(); new Array(); [{x:"", y:""}]', // mutates new Array(x,y)
+ '[]; new Array(); new Array({x:"", y:""}); []', // mutates [x,y]
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/arrow-function-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/arrow-function-mutator.spec.ts
index 6be012de8e..d745905076 100644
--- a/packages/instrumenter/test/unit/mutators/arrow-function-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/arrow-function-mutator.spec.ts
@@ -1,7 +1,12 @@
import { expect } from 'chai';
import { arrowFunctionMutator as sut } from '../../../src/mutators/arrow-function-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const arrowFunctionLevel: MutationLevel = { name: 'ArrowFunctionLevel', ArrowFunction: ['ArrowFunctionRemoval'] };
+const arrowFunctionOperatorUndefinedLevel: MutationLevel = { name: 'ArrowFunctionLevel', ArrowFunction: [] };
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "ArrowFunction"', () => {
@@ -19,4 +24,18 @@ describe(sut.name, () => {
it('should not mutate an anonymous function with undefined as a body', () => {
expectJSMutation(sut, 'const b = () => undefined');
});
+
+ describe('mutation level', () => {
+ it('should remove ArrowFunction', () => {
+ expectJSMutationWithLevel(sut, arrowFunctionLevel.ArrowFunction, 'const b = () => 4;', 'const b = () => undefined;'); // ArrowFunctionRemoval
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, arrowFunctionOperatorUndefinedLevel.ArrowFunction, 'const b = () => 4;');
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, noLevel, 'const b = () => 4;', 'const b = () => undefined;');
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/assignment-operator-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/assignment-operator-mutator.spec.ts
index 8fe2a137b0..349cdf00a7 100644
--- a/packages/instrumenter/test/unit/mutators/assignment-operator-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/assignment-operator-mutator.spec.ts
@@ -1,7 +1,15 @@
import { expect } from 'chai';
import { assignmentOperatorMutator as sut } from '../../../src/mutators/assignment-operator-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const assignmentOperatorLevel: MutationLevel = {
+ name: 'AssignmentOperatorLevel',
+ AssignmentOperator: ['SubtractionAssignmentNegation', 'LeftShiftAssignmentNegation', 'LogicalAndAssignmentToLogicalOrReplacement'],
+};
+const assignmentOperatorUndefinedLevel: MutationLevel = { name: 'AssignmentOperatorLevel', AssignmentOperator: [] };
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "AssignmentOperator"', () => {
@@ -72,4 +80,35 @@ describe(sut.name, () => {
expectJSMutation(sut, 'a ||= `b`', 'a &&= `b`');
expectJSMutation(sut, 'a ??= `b`', 'a &&= `b`');
});
+
+ describe('mutation level', () => {
+ it('should only mutate -=, <<, &&=', () => {
+ expectJSMutationWithLevel(
+ sut,
+ assignmentOperatorLevel.AssignmentOperator,
+ 'a += b; a -= b; a *= b; a /= b; a <<= b; a &&= b;',
+ 'a += b; a += b; a *= b; a /= b; a <<= b; a &&= b;', // mutates -= to +=
+ 'a += b; a -= b; a *= b; a /= b; a >>= b; a &&= b;', // mutates <<= to >>=
+ 'a += b; a -= b; a *= b; a /= b; a <<= b; a ||= b;', // mutates &&= to ||=
+ );
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, assignmentOperatorUndefinedLevel.AssignmentOperator, 'a += b; a -= b; a *= b; a /= b; a <<= b; a &&= b;');
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ 'a += b; a -= b; a *= b; a /= b; a <<= b; a &&= b;',
+ 'a -= b; a -= b; a *= b; a /= b; a <<= b; a &&= b;', // mutates += to -=
+ 'a += b; a += b; a *= b; a /= b; a <<= b; a &&= b;', // mutates -= to +=
+ 'a += b; a -= b; a /= b; a /= b; a <<= b; a &&= b;', // mutates *= to /=
+ 'a += b; a -= b; a *= b; a *= b; a <<= b; a &&= b;', // mutates /= to *=
+ 'a += b; a -= b; a *= b; a /= b; a >>= b; a &&= b;', // mutates <<= to >>=
+ 'a += b; a -= b; a *= b; a /= b; a <<= b; a ||= b;', // mutates &&= to ||=
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/block-statement-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/block-statement-mutator.spec.ts
index 55566dd9c1..82670df9d9 100644
--- a/packages/instrumenter/test/unit/mutators/block-statement-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/block-statement-mutator.spec.ts
@@ -1,7 +1,12 @@
import { expect } from 'chai';
import { blockStatementMutator as sut } from '../../../src/mutators/block-statement-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const blockStatementLevel: MutationLevel = { name: 'BlockStatementLevel', BlockStatement: ['BlockStatementRemoval'] };
+const blockStatementUndefinedLevel: MutationLevel = { name: 'BlockStatementLevel', BlockStatement: [] };
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "BlockStatement"', () => {
@@ -71,4 +76,18 @@ describe(sut.name, () => {
expectJSMutation(sut, 'class Foo extends Bar { private baz = "qux"; constructor() { super(); } }');
});
});
+
+ describe('mutation level', () => {
+ it('should remove BlockStatement', () => {
+ expectJSMutationWithLevel(sut, blockStatementLevel.BlockStatement, 'class Foo { constructor() { bar(); } }', 'class Foo { constructor() {} }'); // BlockStatementRemoval
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, blockStatementUndefinedLevel.BlockStatement, 'class Foo { constructor() { bar(); } }');
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, noLevel, 'class Foo { constructor() { bar(); } }', 'class Foo { constructor() {} }');
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/boolean-literal-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/boolean-literal-mutator.spec.ts
index 1cf171553e..cff70b6a69 100644
--- a/packages/instrumenter/test/unit/mutators/boolean-literal-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/boolean-literal-mutator.spec.ts
@@ -1,7 +1,20 @@
import { expect } from 'chai';
import { booleanLiteralMutator as sut } from '../../../src/mutators/boolean-literal-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const booleanLiteralLevel: MutationLevel = {
+ name: 'BooleanLiteralLevel',
+ BooleanLiteral: ['TrueLiteralNegation', 'LogicalNotRemoval'],
+};
+
+const booleanLiteralUndefinedLevel: MutationLevel = {
+ name: 'BooleanLiteralLevel',
+ BooleanLiteral: [],
+};
+
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "BooleanLiteral"', () => {
@@ -19,4 +32,31 @@ describe(sut.name, () => {
it('should mutate !a to a', () => {
expectJSMutation(sut, '!a', 'a');
});
+
+ describe('mutation level', () => {
+ it('should only mutate TrueLiteralNegation, LogicalNotRemoval', () => {
+ expectJSMutationWithLevel(
+ sut,
+ booleanLiteralLevel.BooleanLiteral,
+ 'if (true) {}; if (false) {}; if (!value) {}',
+ 'if (false) {}; if (false) {}; if (!value) {}', // TrueLiteralNegation
+ 'if (true) {}; if (false) {}; if (value) {}', // LogicalNotRemoval
+ );
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, booleanLiteralUndefinedLevel.BooleanLiteral, 'if (true) {}; if (false) {}; if (!value) {}');
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ 'if (true) {}; if (false) {}; if (!value) {}',
+ 'if (false) {}; if (false) {}; if (!value) {}', // TrueLiteralNegation
+ 'if (true) {}; if (false) {}; if (value) {}', // LogicalNotRemoval
+ 'if (true) {}; if (true) {}; if (!value) {}', // FalseLiteralNegation
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/conditional-expression-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/conditional-expression-mutator.spec.ts
index 68b060c2f3..2046931b33 100644
--- a/packages/instrumenter/test/unit/mutators/conditional-expression-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/conditional-expression-mutator.spec.ts
@@ -1,7 +1,30 @@
import { expect } from 'chai';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
import { conditionalExpressionMutator as sut } from '../../../src/mutators/conditional-expression-mutator.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const conditionalLevel: MutationLevel = {
+ name: 'ConditionalLevel',
+ ConditionalExpression: [
+ 'ForLoopConditionToFalseReplacement',
+ 'IfConditionToFalseReplacement',
+ 'IfConditionToTrueReplacement',
+ 'SwitchStatementBodyRemoval',
+ ],
+};
+
+const booleanExpressionLevel: MutationLevel = {
+ name: 'ConditionalLevel',
+ ConditionalExpression: ['BooleanExpressionToFalseReplacement', 'BooleanExpressionToTrueReplacement'],
+};
+
+const conditionalUndefinedLevel: MutationLevel = {
+ name: 'ConditionLevelEmpty',
+ ConditionalExpression: [],
+};
+
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "ConditionalExpression"', () => {
@@ -140,4 +163,62 @@ describe(sut.name, () => {
it('should mutate the expression of a while statement', () => {
expectJSMutation(sut, 'while(a < b) { console.log(); }', 'while(false) { console.log(); }');
});
+
+ describe('mutation level', () => {
+ it('should only mutate for, if and switch statement', () => {
+ expectJSMutationWithLevel(
+ sut,
+ conditionalLevel.ConditionalExpression,
+ 'for (var i = 0; i < 10; i++) { };if(x > 2); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2',
+ 'for (var i = 0; false; i++) { };if(x > 2); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2', // mutates for loop
+ 'for (var i = 0; i < 10; i++) { };if(false); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2', // mutates if statement to false
+ 'for (var i = 0; i < 10; i++) { };if(true); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2', // mutates if statement to true
+ 'for (var i = 0; i < 10; i++) { };if(x > 2); switch (x) {case 0:}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2', // mutates switch statement
+ );
+ });
+
+ it('should only mutate && boolean expressions', () => {
+ expectJSMutationWithLevel(
+ sut,
+ booleanExpressionLevel.ConditionalExpression,
+ 'if (true) { }; for(let i=0;; i++) { }; if ((c1 && c2) || (c3 && c4)) { } ',
+ 'if (true) { }; for(let i=0;; i++) { }; if ((c1 && c2) || (false)) { } ', // mutates c3 && c4 to false
+ 'if (true) { }; for(let i=0;; i++) { }; if ((false) || (c3 && c4)) { } ', // mutates c1 && c2 to false
+ );
+ });
+
+ it('should only mutate || boolean expressions', () => {
+ expectJSMutationWithLevel(
+ sut,
+ booleanExpressionLevel.ConditionalExpression,
+ 'if (true) { }; for(let i=0;; i++) { }; if ((c1 || c2) && (c3 || c4)) { } ',
+ 'if (true) { }; for(let i=0;; i++) { }; if ((c1 || c2) && (true)) { } ', // mutates c3 || c4 to true
+ 'if (true) { }; for(let i=0;; i++) { }; if ((true) && (c3 || c4)) { } ', // mutates c1 || c2 to true
+ );
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ conditionalUndefinedLevel.ConditionalExpression,
+ 'for (var i = 0; i < 10; i++) { };if(x > 2); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2',
+ );
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ 'for (var i = 0; i < 10; i++) { };if(x > 2); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2',
+ 'for (var i = 0; false; i++) { };if(x > 2); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2', // mutates for loop
+ 'for (var i = 0; i < 10; i++) { };if(false); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2', // mutates if statement to false
+ 'for (var i = 0; i < 10; i++) { };if(true); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2', // mutates if statement to true
+ 'for (var i = 0; i < 10; i++) { };if(x > 2); switch (x) {case 0:}; while (a > b); { } do { } while (a > b); var x = a > b ? 1 : 2', // mutates switch statement
+ 'for (var i = 0; i < 10; i++) { };if(x > 2); switch (x) {case 0: 2}; while (false); { } do { } while (a > b); var x = a > b ? 1 : 2', // mutates while loop
+ 'for (var i = 0; i < 10; i++) { };if(x > 2); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = false ? 1 : 2', // mutates boolean expression to false
+ 'for (var i = 0; i < 10; i++) { };if(x > 2); switch (x) {case 0: 2}; while (a > b); { } do { } while (false); var x = a > b ? 1 : 2', // mutates do while loop
+ 'for (var i = 0; i < 10; i++) { };if(x > 2); switch (x) {case 0: 2}; while (a > b); { } do { } while (a > b); var x = true ? 1 : 2', // mutates boolean expression to true
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/equality-operator-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/equality-operator-mutator.spec.ts
index f62e11f988..9bd2c44157 100644
--- a/packages/instrumenter/test/unit/mutators/equality-operator-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/equality-operator-mutator.spec.ts
@@ -1,7 +1,26 @@
import { expect } from 'chai';
import { equalityOperatorMutator as sut } from '../../../src/mutators/equality-operator-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const equalityOperatorLevel: MutationLevel = {
+ name: 'EqualityOperatorLevel',
+ EqualityOperator: [
+ 'LessThanOperatorBoundary',
+ 'LessThanOperatorNegation',
+ 'GreaterThanEqualOperatorBoundary',
+ 'GreaterThanEqualOperatorNegation',
+ 'EqualityOperatorNegation',
+ ],
+};
+
+const equalityOperatorUndefinedLevel: MutationLevel = {
+ name: 'EqualityOperatorUndefinedLevel',
+ EqualityOperator: [],
+};
+
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "EqualityOperator"', () => {
@@ -27,4 +46,47 @@ describe(sut.name, () => {
expectJSMutation(sut, 'a != b', 'a == b');
expectJSMutation(sut, 'a !== b', 'a === b');
});
+
+ describe('mutation level', () => {
+ it('should only mutate <, >=, ==', () => {
+ expectJSMutationWithLevel(
+ sut,
+ equalityOperatorLevel.EqualityOperator,
+ 'a < b; a <= b; a > b; a >= b; a == b; a != b; a === b; a !== b',
+ 'a <= b; a <= b; a > b; a >= b; a == b; a != b; a === b; a !== b', // mutates < to <=
+ 'a >= b; a <= b; a > b; a >= b; a == b; a != b; a === b; a !== b', // mutates < to >=
+ 'a < b; a <= b; a > b; a > b; a == b; a != b; a === b; a !== b', // mutates >= To >
+ 'a < b; a <= b; a > b; a < b; a == b; a != b; a === b; a !== b', // mutates >= to <
+ 'a < b; a <= b; a > b; a >= b; a != b; a != b; a === b; a !== b', // mutates == to !=
+ );
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ equalityOperatorUndefinedLevel.EqualityOperator,
+ 'a < b; a <= b; a > b; a >= b; a == b; a != b; a === b; a !== b',
+ );
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ 'a < b; a <= b; a > b; a >= b; a == b; a != b; a === b; a !== b',
+ 'a < b; a < b; a > b; a >= b; a == b; a != b; a === b; a !== b', // mutates <= to <
+ 'a < b; a <= b; a <= b; a >= b; a == b; a != b; a === b; a !== b', // mutates > to <=
+ 'a < b; a <= b; a > b; a < b; a == b; a != b; a === b; a !== b', // mutates <= to <
+ 'a < b; a <= b; a > b; a > b; a == b; a != b; a === b; a !== b', // mutates >= to >
+ 'a < b; a <= b; a > b; a >= b; a != b; a != b; a === b; a !== b', // mutates == to !=
+ 'a < b; a <= b; a > b; a >= b; a == b; a != b; a !== b; a !== b', // mutates === to !==
+ 'a < b; a <= b; a > b; a >= b; a == b; a != b; a === b; a === b', // mutates !== to ===
+ 'a < b; a <= b; a > b; a >= b; a == b; a == b; a === b; a !== b', // mutates != to ==
+ 'a < b; a <= b; a >= b; a >= b; a == b; a != b; a === b; a !== b', // mutates > to >=
+ 'a < b; a > b; a > b; a >= b; a == b; a != b; a === b; a !== b', // mutates <= to >
+ 'a <= b; a <= b; a > b; a >= b; a == b; a != b; a === b; a !== b', // mutates < to <=
+ 'a >= b; a <= b; a > b; a >= b; a == b; a != b; a === b; a !== b', // mutates < to >=
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/logical-operator-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/logical-operator-mutator.spec.ts
index 8476e3731e..e46e5d0148 100644
--- a/packages/instrumenter/test/unit/mutators/logical-operator-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/logical-operator-mutator.spec.ts
@@ -1,7 +1,20 @@
import { expect } from 'chai';
import { logicalOperatorMutator as sut } from '../../../src/mutators/logical-operator-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const logicalOperatorLevel: MutationLevel = {
+ name: 'logicalOperatorLevel',
+ LogicalOperator: ['LogicalOrOperatorToLogicalAndReplacement', 'LogicalAndOperatorToLogicalOrReplacement'],
+};
+
+const logicalOperatorUndefinedLevel: MutationLevel = {
+ name: 'logicalOperatorUndefinedLevel',
+ LogicalOperator: [],
+};
+
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "LogicalOperator"', () => {
@@ -24,4 +37,31 @@ describe(sut.name, () => {
it('should mutate ?? to &&', () => {
expectJSMutation(sut, 'a ?? b', 'a && b');
});
+
+ describe('mutation level', () => {
+ it('should only mutate || and &&', () => {
+ expectJSMutationWithLevel(
+ sut,
+ logicalOperatorLevel.LogicalOperator,
+ 'a || b; a && b; a ?? b',
+ 'a && b; a && b; a ?? b', // mutates || to &&
+ 'a || b; a || b; a ?? b', // mutates && to ||
+ );
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, logicalOperatorUndefinedLevel.LogicalOperator, 'a || b; a && b; a ?? b');
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ 'a || b; a && b; a ?? b',
+ 'a && b; a && b; a ?? b', // mutates || to &&
+ 'a || b; a || b; a ?? b', // mutates && to ||
+ 'a || b; a && b; a && b', // mutates ?? to &&
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/method-expression-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/method-expression-mutator.spec.ts
index 1528dc4aa8..7943351c27 100644
--- a/packages/instrumenter/test/unit/mutators/method-expression-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/method-expression-mutator.spec.ts
@@ -1,7 +1,22 @@
import { expect } from 'chai';
import { methodExpressionMutator as sut } from '../../../src/mutators/method-expression-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const methodExpressionLevel: MutationLevel = {
+ name: 'methodExpressionLevel',
+ MethodExpression: ['EndsWithMethodCallNegation', 'StartsWithMethodCallNegation', 'SubstringMethodCallRemoval', 'ToLowerCaseMethodCallNegation'],
+};
+const methodExpressionUndefinedLevel: MutationLevel = {
+ name: 'methodExpressionUndefinedLevel',
+ MethodExpression: [],
+};
+
+const noLevel = undefined;
+
+const methodsCalls =
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()';
describe(sut.name, () => {
it('should have name "MethodExpression"', () => {
@@ -147,4 +162,54 @@ describe(sut.name, () => {
expectJSMutation(sut, 'new text.trim();');
});
});
+
+ describe('mutation level', () => {
+ it('should only mutate startsWith, toLowerCase, substring and endsWith', () => {
+ expectJSMutationWithLevel(
+ sut,
+ methodExpressionLevel.MethodExpression,
+ methodsCalls,
+ 'text.endsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates startsWith()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toUpperCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates toLowerCase()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text;text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // removes substring()
+ 'text.startsWith?.(); text.startsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates endsWith
+ );
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, methodExpressionUndefinedLevel.MethodExpression, methodsCalls);
+ });
+
+ it('should not mutate non-existing function', () => {
+ expectJSMutationWithLevel(sut, noLevel, 'Math.floor(5.95)');
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ methodsCalls,
+ 'text.endsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates startsWith()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimEnd();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates trimStart()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toLowerCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates toUpperCase()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleLowerCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates toLocaleUpperCase()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.every(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates some()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text;text.min();text.max()', // removes charAt()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text;text.charAt();text.min();text.max()', // removes slice()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text;text.slice();text.charAt();text.min();text.max()', // removes filter()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text;text.filter();text.slice();text.charAt();text.min();text.max()', // removes reverse()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.some();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates every()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text;text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // removes sort()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toUpperCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates toLowerCase()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text;text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // removes substring()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text;text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // removes substr()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimStart();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates trimEnd()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleUpperCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates toLocaleLowerCase()
+ 'text.startsWith?.(); text.endsWith(); text; text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // removes trim()
+ 'text.startsWith?.(); text.startsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.max()', // mutates endsWith()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.max();text.max()', // mutates min()
+ 'text.startsWith?.(); text.endsWith(); text.trim(); text.trimEnd();text.trimStart();text.substr();text.substring();text.toUpperCase();text.toLowerCase();text.toLocaleUpperCase();text.toLocaleLowerCase();text.sort();text.some(); text.every();text.reverse();text.filter();text.slice();text.charAt();text.min();text.min()', // mutates max()
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/mutate.spec.ts b/packages/instrumenter/test/unit/mutators/mutate.spec.ts
index 7aca905ce3..b5f1b7c6f6 100644
--- a/packages/instrumenter/test/unit/mutators/mutate.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/mutate.spec.ts
@@ -4,12 +4,14 @@ import { fileURLToPath, pathToFileURL } from 'url';
import { expect } from 'chai';
+import { MutatorDefinition } from '@stryker-mutator/api/core';
+
import { allMutators, NodeMutator } from '../../../src/mutators/index.js';
describe('allMutators', () => {
it('should include all mutators', async () => {
const resolveMutator = path.resolve.bind(path, path.dirname(fileURLToPath(import.meta.url)), '..', '..', '..', 'src', 'mutators');
- const blackList = ['index.js', 'node-mutator.js', 'mutator-options.js', 'mutate.js'];
+ const blackList = ['index.js', 'node-mutator.js', 'mutator-options.js', 'mutate.js', 'mutation-level-options.js'];
const actualMutators = (await Promise.all(
fs
.readdirSync(resolveMutator())
@@ -23,7 +25,7 @@ describe('allMutators', () => {
}
return mutatorModule[keys[0]];
}),
- )) as NodeMutator[];
+ )) as Array>;
actualMutators.forEach((mutator) => {
expect(allMutators.includes(mutator), `${mutator.name} is missing!`).ok;
});
diff --git a/packages/instrumenter/test/unit/mutators/object-literal-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/object-literal-mutator.spec.ts
index 516047d33e..61df32fcdf 100644
--- a/packages/instrumenter/test/unit/mutators/object-literal-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/object-literal-mutator.spec.ts
@@ -1,7 +1,12 @@
import { expect } from 'chai';
import { objectLiteralMutator as sut } from '../../../src/mutators/object-literal-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const objectLiteralLevel: MutationLevel = { name: 'ObjectLiteralLevel', ObjectLiteral: ['ObjectLiteralPropertiesRemoval'] };
+const objectLiteralUndefinedLevel: MutationLevel = { name: 'ObjectLiteralLevel', ObjectLiteral: [] };
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "ObjectLiteral"', () => {
@@ -23,4 +28,18 @@ describe(sut.name, () => {
it('shoud not mutate empty object declarations', () => {
expectJSMutation(sut, 'const o = {}');
});
+
+ describe('mutation level', () => {
+ it('should remove object literal', () => {
+ expectJSMutationWithLevel(sut, objectLiteralLevel.ObjectLiteral, 'const o = { ["foo"]: "bar" }', 'const o = {}');
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, objectLiteralUndefinedLevel.ObjectLiteral, "{ foo: 'bar' }");
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, noLevel, 'const o = { ["foo"]: "bar" }', 'const o = {}');
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/optional-chaining-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/optional-chaining-mutator.spec.ts
index d00d0f70ff..58a4a79edb 100644
--- a/packages/instrumenter/test/unit/mutators/optional-chaining-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/optional-chaining-mutator.spec.ts
@@ -2,7 +2,19 @@ import { expect } from 'chai';
import { optionalChainingMutator as sut } from '../../../src/mutators/optional-chaining-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const optionalChainingLevel: MutationLevel = {
+ name: 'OptionalChainingLevel',
+ OptionalChaining: ['OptionalMemberExpressionOptionalRemoval'],
+};
+const optionalChainingUndefinedLevel: MutationLevel = {
+ name: 'optionalChainingUndefinedLevel',
+ OptionalChaining: [],
+};
+
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "OptionalChaining"', () => {
@@ -29,4 +41,28 @@ describe(sut.name, () => {
expectJSMutation(sut, 'foo.bar?.()[0]', 'foo.bar()[0]');
expectJSMutation(sut, 'foo.bar()?.baz', 'foo.bar().baz');
});
+
+ describe('mutation level', () => {
+ it('should only mutate OptionalMemberExpression', () => {
+ expectJSMutationWithLevel(
+ sut,
+ optionalChainingLevel.OptionalChaining,
+ 'foo?.bar; foo?.[0]; foo?.()',
+ 'foo.bar; foo?.[0]; foo?.()', // removes .bar optional
+ );
+ });
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, optionalChainingUndefinedLevel.OptionalChaining, 'foo?.bar; foo?.[0]; foo?.()');
+ });
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ 'foo?.bar; foo?.[0]; foo?.()',
+ 'foo.bar; foo?.[0]; foo?.()', // removes .bar optional
+ 'foo?.bar; foo[0]; foo?.()', // removes [0] optional
+ 'foo?.bar; foo?.[0]; foo()', // removes () optional
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/regex-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/regex-mutator.spec.ts
index 129f6d6846..e1c4570974 100644
--- a/packages/instrumenter/test/unit/mutators/regex-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/regex-mutator.spec.ts
@@ -2,7 +2,12 @@ import { expect } from 'chai';
import sinon from 'sinon';
import { regexMutator as sut } from '../../../src/mutators/regex-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const regexLevel: MutationLevel = { name: 'RegexLevel', Regex: ['RegexRemoval'] };
+const regexUndefinedLevel: MutationLevel = { name: 'RegexLevel', Regex: [] };
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "Regex"', () => {
@@ -53,4 +58,18 @@ describe(sut.name, () => {
it('should only pass flags in new RegExp constructors if it is a string literal', () => {
expectJSMutation(sut, 'new RegExp("\\\\u{20}", foo)', 'new RegExp("\\\\u", foo)');
});
+
+ describe('mutation level', () => {
+ it('should remove regex', () => {
+ expectJSMutationWithLevel(sut, regexLevel.Regex, 'new RegExp("\\\\u{20}", foo)', 'new RegExp("\\\\u", foo)');
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, regexUndefinedLevel.Regex, 'new RegExp("\\\\u{20}", foo)');
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, noLevel, 'new RegExp("\\\\u{20}", foo)', 'new RegExp("\\\\u", foo)');
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/string-literal-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/string-literal-mutator.spec.ts
index fd12ec05e3..e592a5af05 100644
--- a/packages/instrumenter/test/unit/mutators/string-literal-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/string-literal-mutator.spec.ts
@@ -1,7 +1,15 @@
import { expect } from 'chai';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
import { stringLiteralMutator as sut } from '../../../src/mutators/string-literal-mutator.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const stringLiteralLevel: MutationLevel = {
+ name: 'stringLiteralLevel',
+ StringLiteral: ['FilledStringLiteralToEmptyReplacement', 'FilledInterpolatedStringToEmptyReplacement'],
+};
+const stringLiteralUndefinedLevel: MutationLevel = { name: 'stringLiteralUndefinedLevel', StringLiteral: [] };
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "StringLiteral"', () => {
@@ -112,4 +120,34 @@ describe(sut.name, () => {
expectJSMutation(sut, '');
});
});
+
+ describe('mutation level', () => {
+ it('should only mutate EmptyString and EmptyInterpolation', () => {
+ expectJSMutationWithLevel(
+ sut,
+ stringLiteralLevel.StringLiteral,
+ 'const bar = "bar"; const foo = `name: ${level_name}`; const emptyString=""; const emptyInterp=``',
+ 'const bar = ""; const foo = `name: ${level_name}`; const emptyString=""; const emptyInterp=``', // empties string
+ 'const bar = "bar"; const foo = ``; const emptyString=""; const emptyInterp=``', // empties interpolation
+ );
+ });
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ stringLiteralUndefinedLevel.StringLiteral,
+ 'const bar = "bar"; const foo = `name: ${level_name}`; const emptyString=""; const emptyInterp=``',
+ );
+ });
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ 'const bar = "bar"; const foo = `name: ${level_name}`; const emptyString=""; const emptyInterp=``',
+ 'const bar = ""; const foo = `name: ${level_name}`; const emptyString=""; const emptyInterp=``', // empties string literal
+ 'const bar = "bar"; const foo = ``; const emptyString=""; const emptyInterp=``', // empties interpolation
+ 'const bar = "bar"; const foo = `name: ${level_name}`; const emptyString="Stryker was here!"; const emptyInterp=``', // fills string literal
+ 'const bar = "bar"; const foo = `name: ${level_name}`; const emptyString=""; const emptyInterp=`Stryker was here!`', // fills interpolation
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/unary-operator-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/unary-operator-mutator.spec.ts
index a4bacf066f..22ea6d9d2c 100644
--- a/packages/instrumenter/test/unit/mutators/unary-operator-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/unary-operator-mutator.spec.ts
@@ -1,7 +1,15 @@
import { expect } from 'chai';
import { unaryOperatorMutator as sut } from '../../../src/mutators/unary-operator-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const unaryOperatorLevel: MutationLevel = {
+ name: 'unaryOperatorLevel',
+ UnaryOperator: ['UnaryPlusOperatorNegation', 'UnaryBitwiseNotRemoval'],
+};
+const unaryOperatorUndefinedLevel: MutationLevel = { name: 'unaryOperatorUndefinedLevel', UnaryOperator: [] };
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "UnaryOperator"', () => {
@@ -27,4 +35,31 @@ describe(sut.name, () => {
it('should not mutate a-a', () => {
expectJSMutation(sut, 'a-a');
});
+
+ describe('mutation level', () => {
+ it('should only mutate unary + and ~', () => {
+ expectJSMutationWithLevel(
+ sut,
+ unaryOperatorLevel.UnaryOperator,
+ '+a; -b; ~c;',
+ '-a; -b; ~c;', // mutates + to -
+ '+a; -b; c;', // removes ~
+ );
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, unaryOperatorUndefinedLevel.UnaryOperator, '+a; -b; ~c;');
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ '+a; -b; ~c;',
+ '-a; -b; ~c;', // mutates + to -
+ '+a; -b; c;', // removes ~
+ '+a; +b; ~c;', // mutates - to +
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/mutators/update-operator-mutator.spec.ts b/packages/instrumenter/test/unit/mutators/update-operator-mutator.spec.ts
index 9c5ac0c0b8..70b1a9c2af 100644
--- a/packages/instrumenter/test/unit/mutators/update-operator-mutator.spec.ts
+++ b/packages/instrumenter/test/unit/mutators/update-operator-mutator.spec.ts
@@ -1,7 +1,19 @@
import { expect } from 'chai';
import { updateOperatorMutator as sut } from '../../../src/mutators/update-operator-mutator.js';
-import { expectJSMutation } from '../../helpers/expect-mutation.js';
+import { expectJSMutation, expectJSMutationWithLevel } from '../../helpers/expect-mutation.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
+
+const updateLevel: MutationLevel = {
+ name: 'UpdateLevel',
+ UpdateOperator: ['PrefixDecrementOperatorNegation', 'PrefixIncrementOperatorNegation'],
+};
+
+const updateUndefinedLevel: MutationLevel = {
+ name: 'UpdateLevel3',
+ UpdateOperator: [],
+};
+const noLevel = undefined;
describe(sut.name, () => {
it('should have name "UpdateOperator"', () => {
@@ -23,4 +35,32 @@ describe(sut.name, () => {
it('should mutate --a to ++a', () => {
expectJSMutation(sut, '--a', '++a');
});
+
+ describe('mutation level', () => {
+ it('should only mutate --a and ++a', () => {
+ expectJSMutationWithLevel(
+ sut,
+ updateLevel.UpdateOperator,
+ '--a; ++a; a--; a++',
+ '++a; ++a; a--; a++', //mutates --a
+ '--a; --a; a--; a++', //mutates ++a
+ );
+ });
+
+ it('should not perform any ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(sut, updateUndefinedLevel.UpdateOperator, '--a; ++a; a--; a++');
+ });
+
+ it('should perform all ' + sut.name + ' mutations', () => {
+ expectJSMutationWithLevel(
+ sut,
+ noLevel,
+ '--a; ++a; a--; a++',
+ '++a; ++a; a--; a++', //mutates --a
+ '--a; --a; a--; a++', //mutates ++a
+ '--a; ++a; a--; a--', //mutates a++
+ '--a; ++a; a++; a++', //mutates a--
+ );
+ });
+ });
});
diff --git a/packages/instrumenter/test/unit/transformers/babel-transformer.spec.ts b/packages/instrumenter/test/unit/transformers/babel-transformer.spec.ts
index 9ca93afe64..c52516cf82 100644
--- a/packages/instrumenter/test/unit/transformers/babel-transformer.spec.ts
+++ b/packages/instrumenter/test/unit/transformers/babel-transformer.spec.ts
@@ -14,6 +14,7 @@ import { instrumentationBabelHeader } from '../../../src/util/index.js';
import { MutantPlacer } from '../../../src/mutant-placers/index.js';
import { NodeMutator } from '../../../src/mutators/index.js';
import { createJSAst, createTSAst } from '../../helpers/factories.js';
+import { MutationLevel } from '../../../src/mutation-level/mutation-level.js';
const generate = generator.default;
const { types } = babel;
@@ -26,23 +27,28 @@ const { types } = babel;
*/
describe('babel-transformer', () => {
let context: sinon.SinonStubbedInstance;
- let mutators: NodeMutator[];
+ let mutators: Array>;
let mutantPlacers: MutantPlacer[];
let mutantCollector: MutantCollector;
- const fooMutator: NodeMutator = {
+ const fooMutator: NodeMutator = {
name: 'Foo',
+ operators: { Foo: { mutationOperator: 'Foo' } },
*mutate(path) {
if (path.isIdentifier() && path.node.name === 'foo') {
- yield types.identifier('bar');
+ yield [types.identifier('bar'), this.operators.Foo.mutationOperator];
}
},
};
- const plusMutator: NodeMutator = {
+ const plusMutator: NodeMutator = {
name: 'Plus',
+ operators: { Plus: { mutationOperator: 'Plus' } },
*mutate(path) {
if (path.isBinaryExpression() && path.node.operator === '+') {
- yield types.binaryExpression('-', types.cloneNode(path.node.left, true), types.cloneNode(path.node.right, true));
+ yield [
+ types.binaryExpression('-', types.cloneNode(path.node.left, true), types.cloneNode(path.node.right, true)),
+ this.operators.Plus.mutationOperator,
+ ];
}
},
};
@@ -127,7 +133,7 @@ describe('babel-transformer', () => {
context.options.excludedMutations = ['Foo'];
act(ast);
expect(mutantCollector.mutants).lengthOf(1);
- expect(mutantCollector.mutants[0].ignoreReason).eq('Ignored because of excluded mutation "Foo"');
+ expect(mutantCollector.mutants[0].ignoreReason).eq('Ignored because the operator "Foo" is excluded from the mutation run');
});
});
@@ -622,9 +628,10 @@ describe('babel-transformer', () => {
});
mutators.push({
name: 'blockMutatorForTest',
+ operators: { BlockMutatorForTest: { mutationOperator: 'blockMutatorForTest' } },
*mutate(path) {
if (path.isBlockStatement()) {
- yield types.blockStatement([]);
+ yield [types.blockStatement([]), this.operators.BlockMutatorForTest.mutationOperator];
}
},
});
@@ -729,12 +736,10 @@ describe('babel-transformer', () => {
});
function act(ast: ScriptAst) {
- (transformBabel as (...args: [...Parameters>, mutators: NodeMutator[], mutantPlacers: MutantPlacer[]]) => void)(
- ast,
- mutantCollector,
- context,
- mutators,
- mutantPlacers,
- );
+ (
+ transformBabel as (
+ ...args: [...Parameters>, mutators: Array>, mutantPlacers: MutantPlacer[]]
+ ) => void
+ )(ast, mutantCollector, context, mutators, mutantPlacers);
}
});