Skip to content

Commit

Permalink
Add new CatchBlockRemoval mutator (#1742)
Browse files Browse the repository at this point in the history
  • Loading branch information
sidz committed Oct 18, 2022
1 parent 6bc6cfe commit f9bd211
Show file tree
Hide file tree
Showing 4 changed files with 263 additions and 0 deletions.
1 change: 1 addition & 0 deletions resources/schema.json
Original file line number Diff line number Diff line change
Expand Up @@ -330,6 +330,7 @@
}
]
},
"CatchBlockRemoval": { "$ref": "#/definitions/default-mutator-config" },
"FunctionCallRemoval": { "$ref": "#/definitions/default-mutator-config" },
"MethodCallRemoval": { "$ref": "#/definitions/default-mutator-config" },
"CloneRemoval": { "$ref": "#/definitions/default-mutator-config" },
Expand Down
2 changes: 2 additions & 0 deletions src/Mutator/ProfileList.php
Original file line number Diff line number Diff line change
Expand Up @@ -176,6 +176,7 @@ final class ProfileList

public const REMOVAL_PROFILE = [
Mutator\Removal\ArrayItemRemoval::class,
Mutator\Removal\CatchBlockRemoval::class,
Mutator\Removal\CloneRemoval::class,
Mutator\Removal\ConcatOperandRemoval::class,
Mutator\Removal\FunctionCallRemoval::class,
Expand Down Expand Up @@ -381,6 +382,7 @@ final class ProfileList

// Removal
'ArrayItemRemoval' => Mutator\Removal\ArrayItemRemoval::class,
'CatchBlockRemoval' => Mutator\Removal\CatchBlockRemoval::class,
'CloneRemoval' => Mutator\Removal\CloneRemoval::class,
'ConcatOperandRemoval' => Mutator\Removal\ConcatOperandRemoval::class,
'FunctionCallRemoval' => Mutator\Removal\FunctionCallRemoval::class,
Expand Down
117 changes: 117 additions & 0 deletions src/Mutator/Removal/CatchBlockRemoval.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,117 @@
<?php
/**
* This code is licensed under the BSD 3-Clause License.
*
* Copyright (c) 2017, Maks Rafalko
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* * Neither the name of the copyright holder nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/

declare(strict_types=1);

namespace Infection\Mutator\Removal;

use function count;
use Infection\Mutator\Definition;
use Infection\Mutator\GetMutatorName;
use Infection\Mutator\Mutator;
use Infection\Mutator\MutatorCategory;
use PhpParser\Node;

/**
* @internal
*
* @implements Mutator<Node\Stmt\TryCatch>
*/
final class CatchBlockRemoval implements Mutator
{
use GetMutatorName;

public static function getDefinition(): ?Definition
{
return new Definition(
'Removes `catch` block when more than one defined in `try-catch`.',
MutatorCategory::SEMANTIC_REDUCTION,
null,
<<<'DIFF'
try {
$callback();
- } catch (\DomainException $ex) {
- $logger->log($ex);
} catch (\LogicException $e) {
throw $e;
}
DIFF
);
}

public function canMutate(Node $node): bool
{
if (!$node instanceof Node\Stmt\TryCatch || count($node->catches) < 2) {
return false;
}

foreach ($node->catches as $catch) {
if ($catch->stmts !== []) {
return true;
}
}

return false;
}

/**
* @psalm-mutation-free
*
* @return iterable<Node\Stmt\TryCatch>
*/
public function mutate(Node $node): iterable
{
foreach ($node->catches as $i => $catch) {
if (!$this->hasAtLeastOneNonNopStatements(...$catch->stmts)) {
continue;
}

$catches = $node->catches;

unset($catches[$i]);

yield new Node\Stmt\TryCatch($node->stmts, $catches, $node->finally, $node->getAttributes());
}
}

private function hasAtLeastOneNonNopStatements(Node\Stmt ...$stmts): bool
{
foreach ($stmts as $stmt) {
if (!$stmt instanceof Node\Stmt\Nop) {
return true;
}
}

return false;
}
}
143 changes: 143 additions & 0 deletions tests/phpunit/Mutator/Removal/CatchBlockRemovalTest.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,143 @@
<?php
/**
* This code is licensed under the BSD 3-Clause License.
*
* Copyright (c) 2017, Maks Rafalko
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* * Neither the name of the copyright holder nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/

declare(strict_types=1);

namespace Infection\Tests\Mutator\Removal;

use Infection\Tests\Mutator\BaseMutatorTestCase;

final class CatchBlockRemovalTest extends BaseMutatorTestCase
{
/**
* @dataProvider mutationsProvider
*
* @param string|string[] $expected
*/
public function test_it_can_mutate(string $input, array|string $expected = []): void
{
$this->doTest($input, $expected);
}

public function mutationsProvider(): iterable
{
yield 'It removes catch block' => [
<<<'PHP'
<?php
try {
$callback();
} catch (\DomainException $e) {
$logger->log($e);
} catch (\LogicException $e) {
throw $e;
} catch (\Throwable $e) {
throw new \RuntimeException();
}
PHP
,
[
<<<'PHP'
<?php
try {
$callback();
} catch (\LogicException $e) {
throw $e;
} catch (\Throwable $e) {
throw new \RuntimeException();
}
PHP,
<<<'PHP'
<?php
try {
$callback();
} catch (\DomainException $e) {
$logger->log($e);
} catch (\Throwable $e) {
throw new \RuntimeException();
}
PHP,
<<<'PHP'
<?php
try {
$callback();
} catch (\DomainException $e) {
$logger->log($e);
} catch (\LogicException $e) {
throw $e;
}
PHP,
],
];

yield 'It does not mutate with one catch block' => [
<<<'PHP'
<?php
try {
$callback();
} catch (\DomainException $e) {
$logger->log($e);
}
PHP
,
];

yield 'It does not mutate if catch block does not contain statements' => [
<<<'PHP'
<?php
try {
$callback();
} catch (\DomainException $e) {
// DO nothing
} catch (\Throwable $e) {
throw new \RuntimeException();
}
PHP
,
<<<'PHP'
<?php
try {
$callback();
} catch (\DomainException $e) {
// DO nothing
}
PHP
];
}
}

0 comments on commit f9bd211

Please sign in to comment.