chubbyphp /
chubbyphp-mock
| 1 | <?php |
||
| 2 | |||
| 3 | declare(strict_types=1); |
||
| 4 | |||
| 5 | namespace Chubbyphp\Mock; |
||
| 6 | |||
| 7 | use Chubbyphp\Mock\Argument\ArgumentInterface; |
||
| 8 | use PHPUnit\Framework\MockObject\MockObject; |
||
| 9 | |||
| 10 | trait MockByCallsTrait |
||
| 11 | { |
||
| 12 | /** |
||
| 13 | * @param string[]|string $class |
||
| 14 | * @param Call[] $calls |
||
| 15 | */ |
||
| 16 | private function getMockByCalls($class, array $calls = []): MockObject |
||
| 17 | { |
||
| 18 | 11 | $mock = $this->prepareMock($class); |
|
| 19 | |||
| 20 | 11 | $mockName = (new \ReflectionObject($mock))->getShortName(); |
|
| 21 | |||
| 22 | 11 | $className = $this->getMockClassAsString($class); |
|
| 23 | |||
| 24 | 11 | $options = JSON_PRETTY_PRINT | JSON_PRESERVE_ZERO_FRACTION | JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES; |
|
| 25 | |||
| 26 | 11 | $callIndex = -1; |
|
| 27 | |||
| 28 | 11 | $mock->expects(self::exactly(count($calls)))->method(self::anything())->willReturnCallback( |
|
| 29 | function () use ($className, $mock, $mockName, &$callIndex, &$calls, $options) { |
||
| 30 | 11 | ++$callIndex; |
|
| 31 | 11 | ||
| 32 | 10 | $call = array_shift($calls); |
|
| 33 | |||
| 34 | 10 | $method = $call->getMethod(); |
|
| 35 | $mockedMethod = $this->getMockedMethod($mockName); |
||
| 36 | 10 | ||
| 37 | 10 | if ($mockedMethod !== $method) { |
|
| 38 | self::fail( |
||
| 39 | 10 | sprintf( |
|
| 40 | 1 | 'Call at index %d on class "%s" expected method "%s", "%s" given', |
|
| 41 | 1 | $callIndex, |
|
| 42 | 1 | $className, |
|
| 43 | 1 | $method, |
|
| 44 | 1 | $mockedMethod |
|
| 45 | 1 | ) |
|
| 46 | 1 | .PHP_EOL |
|
| 47 | .json_encode($this->getStackTrace($mock), $options) |
||
| 48 | 1 | ); |
|
| 49 | 1 | } |
|
| 50 | |||
| 51 | return $this->getMockCallback($className, $callIndex, $call, $mock)(...func_get_args()); |
||
| 52 | } |
||
| 53 | 9 | ); |
|
| 54 | 11 | ||
| 55 | return $mock; |
||
| 56 | } |
||
| 57 | 11 | ||
| 58 | /** |
||
| 59 | * @param string[]|string $class |
||
| 60 | */ |
||
| 61 | private function prepareMock($class): MockObject |
||
| 62 | { |
||
| 63 | $mockBuilder = $this->getMockBuilder($class) |
||
|
0 ignored issues
–
show
Bug
introduced
by
Loading history...
|
|||
| 64 | ->disableOriginalConstructor() |
||
| 65 | 11 | ->disableOriginalClone() |
|
| 66 | ; |
||
| 67 | 11 | ||
| 68 | 11 | return $mockBuilder->getMock(); |
|
| 69 | 11 | } |
|
| 70 | |||
| 71 | 11 | /** |
|
| 72 | * @param string[]|string $class |
||
| 73 | */ |
||
| 74 | private function getMockClassAsString($class): string |
||
| 75 | { |
||
| 76 | if (is_array($class)) { |
||
| 77 | return implode('|', $class); |
||
| 78 | } |
||
| 79 | 11 | ||
| 80 | return $class; |
||
| 81 | 11 | } |
|
| 82 | 1 | ||
| 83 | private function getMockedMethod(string $mockName): string |
||
| 84 | { |
||
| 85 | 10 | foreach (debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS) as $trace) { |
|
| 86 | if ($mockName === $trace['class']) { |
||
| 87 | return $trace['function']; |
||
| 88 | } |
||
| 89 | } |
||
|
0 ignored issues
–
show
In this branch, the function will implicitly return
null which is incompatible with the type-hinted return string. Consider adding a return statement or allowing null as return value.
For hinted functions/methods where all return statements with the correct type are only reachable via conditions, ?null? gets implicitly returned which may be incompatible with the hinted type. Let?s take a look at an example: interface ReturnsInt {
public function returnsIntHinted(): int;
}
class MyClass implements ReturnsInt {
public function returnsIntHinted(): int
{
if (foo()) {
return 123;
}
// here: null is implicitly returned
}
}
Loading history...
|
|||
| 90 | } |
||
| 91 | |||
| 92 | private function getMockCallback( |
||
| 93 | 10 | string $class, |
|
| 94 | int $callIndex, |
||
| 95 | 10 | Call $call, |
|
| 96 | 10 | MockObject $mock |
|
| 97 | 10 | ): \Closure { |
|
| 98 | return function () use ($class, $callIndex, $call, $mock) { |
||
| 99 | if ($call->hasWith()) { |
||
| 100 | $this->compareArguments($class, $call->getMethod(), $callIndex, $call->getWith(), func_get_args()); |
||
| 101 | } |
||
| 102 | |||
| 103 | if (null !== $exception = $call->getException()) { |
||
| 104 | throw $exception; |
||
| 105 | } |
||
| 106 | |||
| 107 | if ($call->hasReturnSelf()) { |
||
| 108 | return $mock; |
||
| 109 | } |
||
| 110 | |||
| 111 | if ($call->hasReturn()) { |
||
| 112 | return $call->getReturn(); |
||
| 113 | } |
||
| 114 | |||
| 115 | if ($call->hasReturnCallback()) { |
||
| 116 | 9 | $callback = $call->getReturnCallback(); |
|
| 117 | 9 | ||
| 118 | 9 | return $callback(...func_get_args()); |
|
| 119 | } |
||
| 120 | }; |
||
| 121 | 8 | } |
|
| 122 | 1 | ||
| 123 | private function compareArguments( |
||
| 124 | string $class, |
||
| 125 | 7 | string $method, |
|
| 126 | 2 | int $at, |
|
| 127 | array $expectedArguments, |
||
| 128 | array $arguments |
||
| 129 | 5 | ) { |
|
| 130 | 1 | $expectedArgumentsCount = count($expectedArguments); |
|
| 131 | $argumentsCount = count($arguments); |
||
| 132 | |||
| 133 | 4 | self::assertSame( |
|
| 134 | 1 | $expectedArgumentsCount, |
|
| 135 | $argumentsCount, |
||
| 136 | 1 | sprintf( |
|
| 137 | 'Method "%s" on class "%s" at call %d, got %d arguments, but %d are expected', |
||
| 138 | 9 | $method, |
|
| 139 | $class, |
||
| 140 | $at, |
||
| 141 | $expectedArgumentsCount, |
||
| 142 | $argumentsCount |
||
| 143 | ) |
||
| 144 | ); |
||
| 145 | |||
| 146 | foreach ($expectedArguments as $index => $expectedArgument) { |
||
| 147 | if ($expectedArgument instanceof ArgumentInterface) { |
||
| 148 | 9 | $expectedArgument->assert( |
|
| 149 | $arguments[$index], |
||
| 150 | ['class' => $class, 'method' => $method, 'at' => $at, 'index' => $index] |
||
| 151 | ); |
||
| 152 | |||
| 153 | continue; |
||
| 154 | } |
||
| 155 | 9 | ||
| 156 | 9 | self::assertSame( |
|
| 157 | $expectedArgument, |
||
| 158 | 9 | $arguments[$index], |
|
| 159 | 9 | sprintf( |
|
| 160 | 9 | 'Method "%s" on class "%s" at call %d, argument %d', |
|
| 161 | 9 | $method, |
|
| 162 | 9 | $class, |
|
| 163 | 9 | $at, |
|
| 164 | 9 | $index |
|
| 165 | 9 | ) |
|
| 166 | 9 | ); |
|
| 167 | 9 | } |
|
| 168 | } |
||
| 169 | |||
| 170 | private function getStackTrace(MockObject $mock): array |
||
| 171 | 9 | { |
|
| 172 | 9 | $mockName = (new \ReflectionObject($mock))->getShortName(); |
|
| 173 | 3 | ||
| 174 | 3 | $trace = []; |
|
| 175 | 3 | $enableTrace = false; |
|
| 176 | foreach (debug_backtrace() as $i => $row) { |
||
| 177 | if (isset($row['class']) && $mockName === $row['class']) { |
||
| 178 | 2 | $enableTrace = true; |
|
| 179 | } |
||
| 180 | |||
| 181 | 8 | if ($enableTrace) { |
|
| 182 | 8 | $traceRow = ''; |
|
| 183 | 8 | ||
| 184 | 8 | if (isset($row['class'])) { |
|
| 185 | 8 | $traceRow .= $row['class']; |
|
| 186 | 8 | } |
|
| 187 | 8 | ||
| 188 | 8 | if (isset($row['type'])) { |
|
| 189 | 8 | $traceRow .= $row['type']; |
|
| 190 | } |
||
| 191 | |||
| 192 | if (isset($row['function'])) { |
||
| 193 | 8 | $traceRow .= $row['function']; |
|
| 194 | } |
||
| 195 | |||
| 196 | if (isset($row['file'])) { |
||
| 197 | $traceRow .= sprintf(' (%s:%d)', $row['file'], $row['line']); |
||
| 198 | } |
||
| 199 | |||
| 200 | 1 | $trace[] = $traceRow; |
|
| 201 | } |
||
| 202 | 1 | } |
|
| 203 | |||
| 204 | 1 | krsort($trace); |
|
| 205 | 1 | ||
| 206 | 1 | return array_values($trace); |
|
| 207 | 1 | } |
|
| 208 | } |
||
| 209 |