mirror of
https://github.com/pestphp/pest.git
synced 2026-03-06 07:47:22 +01:00
Adds tap for Higher Order tests
This commit is contained in:
33
src/Support/HigherOrderCallables.php
Normal file
33
src/Support/HigherOrderCallables.php
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Pest\Support;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
final class HigherOrderCallables
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @var object
|
||||||
|
*/
|
||||||
|
private $target;
|
||||||
|
|
||||||
|
public function __construct(object $target)
|
||||||
|
{
|
||||||
|
$this->target = $target;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @template TValue
|
||||||
|
*
|
||||||
|
* @param callable(): TValue $callable
|
||||||
|
*
|
||||||
|
* @return TValue|object
|
||||||
|
*/
|
||||||
|
public function tap(callable $callable)
|
||||||
|
{
|
||||||
|
return Reflection::bindCallable($callable) ?? $this->target;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -70,6 +70,10 @@ final class HigherOrderMessage
|
|||||||
*/
|
*/
|
||||||
public function call(object $target)
|
public function call(object $target)
|
||||||
{
|
{
|
||||||
|
if (($value = $this->retrieveHigherOrderCallable($target)) !== null) {
|
||||||
|
return $value;
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
return Reflection::call($target, $this->methodName, $this->arguments);
|
return Reflection::call($target, $this->methodName, $this->arguments);
|
||||||
} catch (Throwable $throwable) {
|
} catch (Throwable $throwable) {
|
||||||
@ -88,6 +92,21 @@ final class HigherOrderMessage
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempts to call one of the available Higher Order callables if it exists.
|
||||||
|
*
|
||||||
|
* @return mixed|null
|
||||||
|
*/
|
||||||
|
private function retrieveHigherOrderCallable(object $target)
|
||||||
|
{
|
||||||
|
if (in_array($this->methodName, get_class_methods(HigherOrderCallables::class), true)) {
|
||||||
|
/* @phpstan-ignore-next-line */
|
||||||
|
return (new HigherOrderCallables($target))->{$this->methodName}(...$this->arguments);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
private static function getUndefinedMethodMessage(object $target, string $methodName): string
|
private static function getUndefinedMethodMessage(object $target, string $methodName): string
|
||||||
{
|
{
|
||||||
if (\PHP_MAJOR_VERSION >= 8) {
|
if (\PHP_MAJOR_VERSION >= 8) {
|
||||||
|
|||||||
@ -41,15 +41,25 @@ final class Reflection
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (is_callable($method)) {
|
if (is_callable($method)) {
|
||||||
return Closure::fromCallable($method)->bindTo(
|
return static::bindCallable($method, $args);
|
||||||
TestSuite::getInstance()->test
|
|
||||||
)(...$args);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
throw $exception;
|
throw $exception;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Bind a callable to the TestCase and return the result.
|
||||||
|
*
|
||||||
|
* @param array<int, mixed> $args
|
||||||
|
*
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
public static function bindCallable(callable $callable, array $args = [])
|
||||||
|
{
|
||||||
|
return Closure::fromCallable($callable)->bindTo(TestSuite::getInstance()->test)(...$args);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Infers the file name from the given closure.
|
* Infers the file name from the given closure.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@ -409,6 +409,8 @@
|
|||||||
PASS Tests\Features\HigherOrderTests
|
PASS Tests\Features\HigherOrderTests
|
||||||
✓ it proxies calls to object
|
✓ it proxies calls to object
|
||||||
✓ it is capable doing multiple assertions
|
✓ it is capable doing multiple assertions
|
||||||
|
✓ it can tap into the test
|
||||||
|
✓ it can use the returned instance from a tap
|
||||||
|
|
||||||
WARN Tests\Features\Incompleted
|
WARN Tests\Features\Incompleted
|
||||||
… incompleted
|
… incompleted
|
||||||
@ -578,5 +580,5 @@
|
|||||||
✓ it is a test
|
✓ it is a test
|
||||||
✓ it uses correct parent class
|
✓ it uses correct parent class
|
||||||
|
|
||||||
Tests: 4 incompleted, 7 skipped, 362 passed
|
Tests: 4 incompleted, 7 skipped, 364 passed
|
||||||
|
|
||||||
@ -1,5 +1,7 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
|
use PHPUnit\Framework\TestCase;
|
||||||
|
|
||||||
beforeEach()->assertTrue(true);
|
beforeEach()->assertTrue(true);
|
||||||
|
|
||||||
it('proxies calls to object')->assertTrue(true);
|
it('proxies calls to object')->assertTrue(true);
|
||||||
@ -8,7 +10,14 @@ it('is capable doing multiple assertions')
|
|||||||
->assertTrue(true)
|
->assertTrue(true)
|
||||||
->assertFalse(false);
|
->assertFalse(false);
|
||||||
|
|
||||||
//it('can tap into the test')
|
it('can tap into the test')
|
||||||
|
->expect('foo')->toBeString()->toBe('foo')
|
||||||
|
->tap(function () { expect($this)->toBeInstanceOf(TestCase::class); })
|
||||||
|
->and('hello world')->toBeString();
|
||||||
|
|
||||||
|
it('can use the returned instance from a tap')
|
||||||
|
->expect('foo')->toBeString()->toBe('foo')
|
||||||
|
->tap(function () { return expect($this); })
|
||||||
|
->toBeInstanceOf(TestCase::class);
|
||||||
|
|
||||||
afterEach()->assertTrue(true);
|
afterEach()->assertTrue(true);
|
||||||
|
|||||||
Reference in New Issue
Block a user