Compare commits

...

10 Commits

Author SHA1 Message Date
ef120125e0 release: 2.34.9 2024-07-11 09:36:26 +01:00
8a9a416133 chore: bumps dependencies 2024-07-11 09:35:43 +01:00
4783334f15 chore: style 2024-07-11 09:35:38 +01:00
e8f122bf47 release: 2.34.8 2024-06-10 23:02:16 +01:00
9fc607a2b8 Fixes wrong stub 2024-06-10 22:52:21 +01:00
b33af71036 Merge pull request #1157 from ExeQue/2.x
[2.x] Added `toBeList` expectation
2024-05-27 13:08:05 +01:00
3c6c89a6ad Added test to toBeList 2024-05-21 08:15:32 +02:00
55f6b5696e Added toBeList expectation 2024-05-21 08:13:20 +02:00
303f4c0113 Adds sponsor 2024-04-19 20:47:46 +01:00
35a1fcd0cf chore: updates readme 2024-04-08 12:28:43 +01:00
54 changed files with 108 additions and 181 deletions

View File

@ -21,6 +21,8 @@ We cannot thank our sponsors enough for their incredible support in funding Pest
### Platinum Sponsors ### Platinum Sponsors
- **[LaraJobs](https://larajobs.com)**
- **[Brokerchooser](https://brokerchooser.com)**
- **[Forge](https://forge.laravel.com)** - **[Forge](https://forge.laravel.com)**
- **[Spatie](https://spatie.be)** - **[Spatie](https://spatie.be)**
- **[Worksome](https://www.worksome.com/)** - **[Worksome](https://www.worksome.com/)**
@ -30,9 +32,7 @@ We cannot thank our sponsors enough for their incredible support in funding Pest
- [Akaunting](https://akaunting.com/?ref=pestphp) - [Akaunting](https://akaunting.com/?ref=pestphp)
- [Codecourse](https://codecourse.com/?ref=pestphp) - [Codecourse](https://codecourse.com/?ref=pestphp)
- [Laracasts](https://laracasts.com/?ref=pestphp) - [Laracasts](https://laracasts.com/?ref=pestphp)
- [Laradir](https://laradir.com/?ref=pestphp)
- [Localazy](https://localazy.com/?ref=pestphp) - [Localazy](https://localazy.com/?ref=pestphp)
- [Stormlikes](https://www.stormlikes.net/?ref=pestphp)
- [Zapiet](https://www.zapiet.com/?ref=pestphp) - [Zapiet](https://www.zapiet.com/?ref=pestphp)
Pest is an open-sourced software licensed under the **[MIT license](https://opensource.org/licenses/MIT)**. Pest is an open-sourced software licensed under the **[MIT license](https://opensource.org/licenses/MIT)**.

View File

@ -52,8 +52,8 @@
}, },
"require-dev": { "require-dev": {
"pestphp/pest-dev-tools": "^2.16.0", "pestphp/pest-dev-tools": "^2.16.0",
"pestphp/pest-plugin-type-coverage": "^2.8.1", "pestphp/pest-plugin-type-coverage": "^2.8.4",
"symfony/process": "^6.4.0|^7.0.4" "symfony/process": "^6.4.0|^7.1.1"
}, },
"minimum-stability": "dev", "minimum-stability": "dev",
"prefer-stable": true, "prefer-stable": true,

View File

@ -32,8 +32,7 @@ final class BootSubscribers implements Bootstrapper
*/ */
public function __construct( public function __construct(
private readonly Container $container, private readonly Container $container,
) { ) {}
}
/** /**
* Boots the list of Subscribers. * Boots the list of Subscribers.

View File

@ -24,9 +24,7 @@ final class EachExpectation
* *
* @param Expectation<TValue> $original * @param Expectation<TValue> $original
*/ */
public function __construct(private readonly Expectation $original) public function __construct(private readonly Expectation $original) {}
{
}
/** /**
* Creates a new expectation. * Creates a new expectation.

View File

@ -36,9 +36,7 @@ final class OppositeExpectation
* *
* @param Expectation<TValue> $original * @param Expectation<TValue> $original
*/ */
public function __construct(private readonly Expectation $original) public function __construct(private readonly Expectation $original) {}
{
}
/** /**
* Asserts that the value array not has the provided $keys. * Asserts that the value array not has the provided $keys.

View File

@ -9,7 +9,5 @@ namespace Pest\Factories\Covers;
*/ */
final class CoversClass final class CoversClass
{ {
public function __construct(public string $class) public function __construct(public string $class) {}
{
}
} }

View File

@ -9,7 +9,5 @@ namespace Pest\Factories\Covers;
*/ */
final class CoversFunction final class CoversFunction
{ {
public function __construct(public string $function) public function __construct(public string $function) {}
{
}
} }

View File

@ -7,6 +7,4 @@ namespace Pest\Factories\Covers;
/** /**
* @internal * @internal
*/ */
final class CoversNothing final class CoversNothing {}
{
}

View File

@ -17,8 +17,7 @@ final class ServiceMessage
public function __construct( public function __construct(
private readonly string $type, private readonly string $type,
private readonly array $parameters, private readonly array $parameters,
) { ) {}
}
public function toString(): string public function toString(): string
{ {

View File

@ -14,9 +14,7 @@ abstract class Subscriber
/** /**
* Creates a new Subscriber instance. * Creates a new Subscriber instance.
*/ */
public function __construct(private readonly TeamCityLogger $logger) public function __construct(private readonly TeamCityLogger $logger) {}
{
}
/** /**
* Creates a new TeamCityLogger instance. * Creates a new TeamCityLogger instance.

View File

@ -7,6 +7,4 @@ namespace Pest\Matchers;
/** /**
* @internal * @internal
*/ */
final class Any final class Any {}
{
}

View File

@ -467,6 +467,18 @@ final class Expectation
return $this; return $this;
} }
/**
* Asserts that the value is a list.
*
* @return self<TValue>
*/
public function toBeList(string $message = ''): self
{
Assert::assertIsList($this->value, $message);
return $this;
}
/** /**
* Asserts that the value is of type bool. * Asserts that the value is of type bool.
* *

View File

@ -6,7 +6,7 @@ namespace Pest;
function version(): string function version(): string
{ {
return '2.34.7'; return '2.34.9';
} }
function testDirectory(string $file = ''): string function testDirectory(string $file = ''): string

View File

@ -63,8 +63,7 @@ final class ResultPrinter
{ {
public function __construct( public function __construct(
private readonly OutputInterface $output, private readonly OutputInterface $output,
) { ) {}
}
public function print(string $buffer): void public function print(string $buffer): void
{ {
@ -79,9 +78,7 @@ final class ResultPrinter
$this->output->write(OutputFormatter::escape($buffer)); $this->output->write(OutputFormatter::escape($buffer));
} }
public function flush(): void public function flush(): void {}
{
}
}; };
$this->compactPrinter = CompactPrinter::default(); $this->compactPrinter = CompactPrinter::default();

View File

@ -21,8 +21,7 @@ final class SnapshotRepository
public function __construct( public function __construct(
readonly private string $testsPath, readonly private string $testsPath,
readonly private string $snapshotsPath, readonly private string $snapshotsPath,
) { ) {}
}
/** /**
* Checks if the snapshot exists. * Checks if the snapshot exists.

View File

@ -24,8 +24,7 @@ final class EnsureTeamCityEnabled implements ConfiguredSubscriber
private readonly InputInterface $input, private readonly InputInterface $input,
private readonly OutputInterface $output, private readonly OutputInterface $output,
private readonly TestSuite $testSuite, private readonly TestSuite $testSuite,
) { ) {}
}
/** /**
* Runs the subscriber. * Runs the subscriber.

View File

@ -30,8 +30,7 @@ final class ExpectationPipeline
*/ */
public function __construct( public function __construct(
private readonly Closure $closure private readonly Closure $closure
) { ) {}
}
/** /**
* Creates a new instance of Expectation Pipeline with given closure. * Creates a new instance of Expectation Pipeline with given closure.

View File

@ -16,7 +16,6 @@ final class NullClosure
*/ */
public static function create(): Closure public static function create(): Closure
{ {
return Closure::fromCallable(function (): void { return Closure::fromCallable(function (): void {});
});
} }
} }

View File

@ -6,5 +6,5 @@ use Illuminate\Foundation\Testing\TestCase as BaseTestCase;
abstract class TestCase extends BaseTestCase abstract class TestCase extends BaseTestCase
{ {
use CreatesApplication; //
} }

View File

@ -1,5 +1,5 @@
Pest Testing Framework 2.34.7. Pest Testing Framework 2.34.8.
USAGE: pest <file> [options] USAGE: pest <file> [options]

View File

@ -1,3 +1,3 @@
Pest Testing Framework 2.34.7. Pest Testing Framework 2.34.8.

View File

@ -511,6 +511,12 @@
✓ passes with strings ✓ passes with strings
✓ failures ✓ failures
✓ failures with custom message ✓ failures with custom message
✓ not failures
PASS Tests\Features\Expect\toBeList
✓ pass
✓ failures
✓ failures with custom message
✓ not failures ✓ not failures
PASS Tests\Features\Expect\toBeLowercase PASS Tests\Features\Expect\toBeLowercase
@ -1418,4 +1424,4 @@
WARN Tests\Visual\Version WARN Tests\Visual\Version
- visual snapshot of help command output - visual snapshot of help command output
Tests: 2 deprecated, 4 warnings, 5 incomplete, 2 notices, 13 todos, 20 skipped, 1009 passed (2395 assertions) Tests: 2 deprecated, 4 warnings, 5 incomplete, 2 notices, 13 todos, 20 skipped, 1013 passed (2405 assertions)

View File

@ -10,9 +10,7 @@ use Tests\Fixtures\Covers\CoversTrait;
$runCounter = 0; $runCounter = 0;
function testCoversFunction() function testCoversFunction() {}
{
}
it('uses the correct PHPUnit attribute for class', function () { it('uses the correct PHPUnit attribute for class', function () {
$attributes = (new ReflectionClass($this))->getAttributes(); $attributes = (new ReflectionClass($this))->getAttributes();

View File

@ -3,8 +3,7 @@
use PHPUnit\Framework\ExpectationFailedException; use PHPUnit\Framework\ExpectationFailedException;
test('pass', function () { test('pass', function () {
expect(function () { expect(function () {})->toBeCallable();
})->toBeCallable();
expect(null)->not->toBeCallable(); expect(null)->not->toBeCallable();
}); });

View File

@ -0,0 +1,21 @@
<?php
use PHPUnit\Framework\ExpectationFailedException;
test('pass', function () {
expect([1, 2, 3])->toBeList();
expect(['a' => 1, 'b' => 2, 'c' => 3])->not->toBeList();
expect('1, 2, 3')->not->toBeList();
});
test('failures', function () {
expect(null)->toBeList();
})->throws(ExpectationFailedException::class);
test('failures with custom message', function () {
expect(null)->toBeList('oh no!');
})->throws(ExpectationFailedException::class, 'oh no!');
test('not failures', function () {
expect(['a', 'b', 'c'])->not->toBeList();
})->throws(ExpectationFailedException::class);

View File

@ -4,13 +4,9 @@ use PHPUnit\Framework\ExpectationFailedException;
$object = new class $object = new class
{ {
public function foo(): void public function foo(): void {}
{
}
public function bar(): void public function bar(): void {}
{
}
}; };
test('pass', function () use ($object) { test('pass', function () use ($object) {

View File

@ -43,9 +43,7 @@ test('pass with `__toString`', function () {
$object = new class($this->snapshotable) $object = new class($this->snapshotable)
{ {
public function __construct(protected string $snapshotable) public function __construct(protected string $snapshotable) {}
{
}
public function __toString() public function __toString()
{ {
@ -61,9 +59,7 @@ test('pass with `toString`', function () {
$object = new class($this->snapshotable) $object = new class($this->snapshotable)
{ {
public function __construct(protected string $snapshotable) public function __construct(protected string $snapshotable) {}
{
}
public function toString() public function toString()
{ {
@ -97,9 +93,7 @@ test('pass with `toArray`', function () {
$object = new class($this->snapshotable) $object = new class($this->snapshotable)
{ {
public function __construct(protected string $snapshotable) public function __construct(protected string $snapshotable) {}
{
}
public function toArray() public function toArray()
{ {
@ -125,9 +119,7 @@ test('pass with `toSnapshot`', function () {
$object = new class($this->snapshotable) $object = new class($this->snapshotable)
{ {
public function __construct(protected string $snapshotable) public function __construct(protected string $snapshotable) {}
{
}
public function toSnapshot() public function toSnapshot()
{ {

View File

@ -2,9 +2,7 @@
use PHPUnit\Framework\ExpectationFailedException; use PHPUnit\Framework\ExpectationFailedException;
class CustomException extends Exception class CustomException extends Exception {}
{
}
test('passes', function () { test('passes', function () {
expect(function () { expect(function () {
@ -15,15 +13,13 @@ test('passes', function () {
})->toThrow(Exception::class); })->toThrow(Exception::class);
expect(function () { expect(function () {
throw new RuntimeException(); throw new RuntimeException();
})->toThrow(function (RuntimeException $e) { })->toThrow(function (RuntimeException $e) {});
});
expect(function () { expect(function () {
throw new RuntimeException('actual message'); throw new RuntimeException('actual message');
})->toThrow(function (Exception $e) { })->toThrow(function (Exception $e) {
expect($e->getMessage())->toBe('actual message'); expect($e->getMessage())->toBe('actual message');
}); });
expect(function () { expect(function () {})->not->toThrow(Exception::class);
})->not->toThrow(Exception::class);
expect(function () { expect(function () {
throw new RuntimeException('actual message'); throw new RuntimeException('actual message');
})->toThrow('actual message'); })->toThrow('actual message');
@ -35,22 +31,18 @@ test('passes', function () {
})->toThrow(RuntimeException::class, 'actual message'); })->toThrow(RuntimeException::class, 'actual message');
expect(function () { expect(function () {
throw new RuntimeException('actual message'); throw new RuntimeException('actual message');
})->toThrow(function (RuntimeException $e) { })->toThrow(function (RuntimeException $e) {}, 'actual message');
}, 'actual message');
expect(function () { expect(function () {
throw new CustomException('foo'); throw new CustomException('foo');
})->toThrow(new CustomException('foo')); })->toThrow(new CustomException('foo'));
}); });
test('failures 1', function () { test('failures 1', function () {
expect(function () { expect(function () {})->toThrow(RuntimeException::class);
})->toThrow(RuntimeException::class);
})->throws(ExpectationFailedException::class, 'Exception "'.RuntimeException::class.'" not thrown.'); })->throws(ExpectationFailedException::class, 'Exception "'.RuntimeException::class.'" not thrown.');
test('failures 2', function () { test('failures 2', function () {
expect(function () { expect(function () {})->toThrow(function (RuntimeException $e) {});
})->toThrow(function (RuntimeException $e) {
});
})->throws(ExpectationFailedException::class, 'Exception "'.RuntimeException::class.'" not thrown.'); })->throws(ExpectationFailedException::class, 'Exception "'.RuntimeException::class.'" not thrown.');
test('failures 3', function () { test('failures 3', function () {
@ -77,8 +69,7 @@ test('failures 5', function () {
})->throws(ExpectationFailedException::class, 'Failed asserting that \'actual message\' [ASCII](length: 14) contains "expected message" [ASCII](length: 16).'); })->throws(ExpectationFailedException::class, 'Failed asserting that \'actual message\' [ASCII](length: 14) contains "expected message" [ASCII](length: 16).');
test('failures 6', function () { test('failures 6', function () {
expect(function () { expect(function () {})->toThrow('actual message');
})->toThrow('actual message');
})->throws(ExpectationFailedException::class, 'Exception with message "actual message" not thrown'); })->throws(ExpectationFailedException::class, 'Exception with message "actual message" not thrown');
test('failures 7', function () { test('failures 7', function () {
@ -106,15 +97,11 @@ test('not failures', function () {
})->throws(ExpectationFailedException::class); })->throws(ExpectationFailedException::class);
test('closure missing parameter', function () { test('closure missing parameter', function () {
expect(function () { expect(function () {})->toThrow(function () {});
})->toThrow(function () {
});
})->throws(InvalidArgumentException::class, 'The given closure must have a single parameter type-hinted as the class string.'); })->throws(InvalidArgumentException::class, 'The given closure must have a single parameter type-hinted as the class string.');
test('closure missing type-hint', function () { test('closure missing type-hint', function () {
expect(function () { expect(function () {})->toThrow(function ($e) {});
})->toThrow(function ($e) {
});
})->throws(InvalidArgumentException::class, 'The given closure\'s parameter must be type-hinted as the class string.'); })->throws(InvalidArgumentException::class, 'The given closure\'s parameter must be type-hinted as the class string.');
it('can handle a non-defined exception', function () { it('can handle a non-defined exception', function () {

View File

@ -6,8 +6,5 @@ namespace Tests\Fixtures\Arch\ToBeInvokable\IsInvokable;
class InvokableClass class InvokableClass
{ {
public function __invoke(): void public function __invoke(): void {}
{
}
} }

View File

@ -6,8 +6,5 @@ namespace Tests\Fixtures\Arch\ToBeInvokable\IsInvokable;
class ParentInvokableClass class ParentInvokableClass
{ {
public function __invoke(): void public function __invoke(): void {}
{
}
} }

View File

@ -6,8 +6,5 @@ namespace Tests\Fixtures\Arch\ToBeInvokable\IsNotInvokable;
class IsNotInvokableClass class IsNotInvokableClass
{ {
public function handle(): void public function handle(): void {}
{
}
} }

View File

@ -7,6 +7,4 @@ namespace Tests\Fixtures\Arch\ToHaveAttribute\Attributes;
use Attribute; use Attribute;
#[Attribute()] #[Attribute()]
class AsAttribute class AsAttribute {}
{
}

View File

@ -7,6 +7,4 @@ namespace Tests\Fixtures\Arch\ToHaveAttribute\HaveAttribute;
use Tests\Fixtures\Arch\ToHaveAttribute\Attributes\AsAttribute; use Tests\Fixtures\Arch\ToHaveAttribute\Attributes\AsAttribute;
#[AsAttribute] #[AsAttribute]
class HaveAttributeClass class HaveAttributeClass {}
{
}

View File

@ -4,6 +4,4 @@ declare(strict_types=1);
namespace Tests\Fixtures\Arch\ToHaveAttribute\NotHaveAttribute; namespace Tests\Fixtures\Arch\ToHaveAttribute\NotHaveAttribute;
class NotHaveAttributeClass class NotHaveAttributeClass {}
{
}

View File

@ -6,8 +6,5 @@ namespace Tests\Fixtures\Arch\ToHaveConstructor\HasConstructor;
class HasConstructor class HasConstructor
{ {
public function __construct() public function __construct() {}
{
}
} }

View File

@ -4,6 +4,4 @@ declare(strict_types=1);
namespace Tests\Fixtures\Arch\ToHaveConstructor\HasNoConstructor; namespace Tests\Fixtures\Arch\ToHaveConstructor\HasNoConstructor;
class HasNoConstructor class HasNoConstructor {}
{
}

View File

@ -6,8 +6,5 @@ namespace Tests\Fixtures\Arch\ToHaveDestructor\HasDestructor;
class HasDestructor class HasDestructor
{ {
public function __destruct() public function __destruct() {}
{
}
} }

View File

@ -4,6 +4,4 @@ declare(strict_types=1);
namespace Tests\Fixtures\Arch\ToHaveDestructor\HasNoDestructor; namespace Tests\Fixtures\Arch\ToHaveDestructor\HasNoDestructor;
class HasNoDestructor class HasNoDestructor {}
{
}

View File

@ -6,8 +6,5 @@ namespace Tests\Fixtures\Arch\ToHaveMethod\HasMethod;
class HasMethod class HasMethod
{ {
public function foo(): void public function foo(): void {}
{
}
} }

View File

@ -6,8 +6,5 @@ namespace Tests\Fixtures\Arch\ToHaveMethod\HasMethod;
trait HasMethodTrait trait HasMethodTrait
{ {
public function foo(): void public function foo(): void {}
{
}
} }

View File

@ -6,8 +6,5 @@ namespace Tests\Fixtures\Arch\ToHaveMethod\HasMethod;
class ParentHasMethodClass class ParentHasMethodClass
{ {
public function foo(): void public function foo(): void {}
{
}
} }

View File

@ -6,8 +6,5 @@ namespace Tests\Fixtures\Arch\ToHaveMethod\HasNoMethod;
class HasNoMethodClass class HasNoMethodClass
{ {
public function bar(): void public function bar(): void {}
{
}
} }

View File

@ -4,6 +4,4 @@ declare(strict_types=1);
namespace Tests\Fixtures\Arch\ToHavePrefix\HasNoPrefix; namespace Tests\Fixtures\Arch\ToHavePrefix\HasNoPrefix;
class ClassWithout class ClassWithout {}
{
}

View File

@ -4,6 +4,4 @@ declare(strict_types=1);
namespace Tests\Fixtures\Arch\ToHavePrefix\HasPrefix; namespace Tests\Fixtures\Arch\ToHavePrefix\HasPrefix;
class PrefixClassWith class PrefixClassWith {}
{
}

View File

@ -4,6 +4,4 @@ declare(strict_types=1);
namespace Tests\Fixtures\Arch\ToHaveSuffix\HasNoSuffix; namespace Tests\Fixtures\Arch\ToHaveSuffix\HasNoSuffix;
class ClassWithout class ClassWithout {}
{
}

View File

@ -4,6 +4,4 @@ declare(strict_types=1);
namespace Tests\Fixtures\Arch\ToHaveSuffix\HasSuffix; namespace Tests\Fixtures\Arch\ToHaveSuffix\HasSuffix;
class ClassWithSuffix class ClassWithSuffix {}
{
}

View File

@ -4,7 +4,5 @@ namespace Tests\Fixtures\Covers;
class CoversClass1 class CoversClass1
{ {
public function foo() public function foo() {}
{
}
} }

View File

@ -2,6 +2,4 @@
namespace Tests\Fixtures\Covers; namespace Tests\Fixtures\Covers;
class CoversClass2 class CoversClass2 {}
{
}

View File

@ -2,6 +2,4 @@
namespace Tests\Fixtures\Covers; namespace Tests\Fixtures\Covers;
class CoversClass3 class CoversClass3 {}
{
}

View File

@ -2,6 +2,4 @@
namespace Tests\Fixtures\Covers; namespace Tests\Fixtures\Covers;
trait CoversTrait trait CoversTrait {}
{
}

View File

@ -50,21 +50,15 @@ it('cannot resolve a parameter without type', function () {
class ClassWithDependency class ClassWithDependency
{ {
public function __construct(Container $container) public function __construct(Container $container) {}
{
}
} }
class ClassWithSubDependency class ClassWithSubDependency
{ {
public function __construct(ClassWithDependency $param) public function __construct(ClassWithDependency $param) {}
{
}
} }
class ClassWithoutTypeParameter class ClassWithoutTypeParameter
{ {
public function __construct($param) public function __construct($param) {}
{
}
} }

View File

@ -3,8 +3,7 @@
use Pest\Support\Reflection; use Pest\Support\Reflection;
it('gets file name from closure', function () { it('gets file name from closure', function () {
$fileName = Reflection::getFileNameFromClosure(function () { $fileName = Reflection::getFileNameFromClosure(function () {});
});
expect($fileName)->toBe(__FILE__); expect($fileName)->toBe(__FILE__);
}); });

View File

@ -19,8 +19,7 @@ it('does not allow to add the same test description twice', function () {
it('alerts users about tests with arguments but no input', function () { it('alerts users about tests with arguments but no input', function () {
$testSuite = new TestSuite(getcwd(), 'tests'); $testSuite = new TestSuite(getcwd(), 'tests');
$method = new TestCaseMethodFactory('foo', 'bar', function (int $arg) { $method = new TestCaseMethodFactory('foo', 'bar', function (int $arg) {});
});
$testSuite->tests->set($method); $testSuite->tests->set($method);
})->throws( })->throws(

View File

@ -16,7 +16,7 @@ $run = function () {
test('parallel', function () use ($run) { test('parallel', function () use ($run) {
expect($run('--exclude-group=integration')) expect($run('--exclude-group=integration'))
->toContain('Tests: 1 deprecated, 4 warnings, 5 incomplete, 2 notices, 13 todos, 16 skipped, 994 passed (2348 assertions)') ->toContain('Tests: 1 deprecated, 4 warnings, 5 incomplete, 2 notices, 13 todos, 16 skipped, 998 passed (2358 assertions)')
->toContain('Parallel: 3 processes'); ->toContain('Parallel: 3 processes');
})->skipOnWindows(); })->skipOnWindows();