alies-dev / psalm-tester
Test Psalm via phpt files!
Requires
- php: ^8.2
- phpunit/phpunit: ^11 || ^12 || ^13
Requires (Dev)
- ergebnis/composer-normalize: ^2.50
- friendsofphp/php-cs-fixer: ^3.94
- phpyh/coding-standard: ^2.6
- psalm/plugin-phpunit: ^0.19.5 || ^0.20.0
- rector/rector: ^2.3
- shipmonk/composer-dependency-analyser: ^1.8
- vimeo/psalm: ^6.10 || ^7.0.0-beta16
This package is auto-updated.
Last update: 2026-04-17 08:54:45 UTC
README
Test Psalm via phpt files!
Installation
composer require --dev alies-dev/psalm-tester
Basic usage
1. Write a test in phpt format
tests/array_values.phpt
--FILE-- <?php /** @psalm-trace $_list */ $_list = array_values(['a' => 1, 'b' => 2]); --EXPECT-- Trace on line 9: $_list: non-empty-list<1|2>
To avoid hardcoding error details, you can use EXPECTF:
--EXPECTF-- Trace on line %d: $_list: non-empty-list<%s>
2. Add a test suite
tests/MyPsalmTest.php
<?php use AliesDev\PsalmTester\PsalmTester; use AliesDev\PsalmTester\PsalmTest as PsalmTestFixture; use PHPUnit\Framework\Attributes\TestWith; use PHPUnit\Framework\TestCase; final class MyPsalmTest extends TestCase { private ?PsalmTester $psalmTester = null; #[TestWith([__DIR__ . '/array_values.phpt'])] public function testPhptFiles(string $phptFile): void { $this->psalmTester ??= PsalmTester::create(); $this->psalmTester->test(PsalmTestFixture::fromPhptFile($phptFile)); } }
Passing different arguments to Psalm
By default PsalmTester runs Psalm with --no-progress --no-diff --config=psalm.xml.
You can change this at the PsalmTester level:
use AliesDev\PsalmTester\PsalmTester; PsalmTester::create( defaultArguments: '--no-progress --no-cache --config=my_default_config.xml', );
or for each test individually using --ARGS-- section:
--ARGS-- --no-progress --config=my_special_config.xml --FILE-- ... --EXPECT-- ...
Skipping tests conditionally
Add a --SKIPIF-- section containing a PHP script that echoes a message starting with skip when the test should not run:
--SKIPIF-- <?php if (PHP_VERSION_ID < 80200) { echo 'skip requires PHP 8.2+'; } --FILE-- <?php ... --EXPECT-- ...
In your test suite, call PsalmTest::getSkipReason() before loading the test and pass the result to PHPUnit's markTestSkipped():
use PHPUnit\Framework\Attributes\TestWith; use PHPUnit\Framework\TestCase; use AliesDev\PsalmTester\PsalmTester; use AliesDev\PsalmTester\PsalmTest as PsalmTestFixture; final class MyPsalmTest extends TestCase { private ?PsalmTester $psalmTester = null; #[TestWith([__DIR__ . '/array_values.phpt'])] public function testPhptFiles(string $phptFile): void { $skipReason = PsalmTestFixture::getSkipReason($phptFile); if ($skipReason !== null) { $this->markTestSkipped($skipReason); } $this->psalmTester ??= PsalmTester::create(); $this->psalmTester->test(PsalmTestFixture::fromPhptFile($phptFile)); } }
The SKIPIF script runs in a separate PHP process, so exit()/die() calls in the script do not affect the test run. getSkipReason() returns the reason string with the leading skip token stripped (e.g. "requires PHP 8.2+") or null if the test should run.
Batch execution
By default, test() spawns a separate Psalm process per .phpt file.
For plugins with expensive boot costs (e.g., Laravel plugin boots a full application), this means each test pays the full startup overhead.
runBatch() groups tests by their argument string and runs one Psalm invocation per group,
then distributes results back to individual tests using the file_path field in Psalm's JSON output.
use PHPUnit\Framework\Assert; use PHPUnit\Framework\Attributes\DataProvider; use PHPUnit\Framework\TestCase; use AliesDev\PsalmTester\PsalmTester; use AliesDev\PsalmTester\PsalmTest; final class MyPsalmTest extends TestCase { /** @var array<string, string> */ private static array $batchResults = []; /** @var array<string, PsalmTest> */ private static array $testData = []; public static function setUpBeforeClass(): void { $tester = PsalmTester::create( defaultArguments: '--no-progress --no-diff --config=' . dirname(__DIR__) . '/psalm.xml', ); foreach (self::discoverPhptFiles() as $name => $path) { self::$testData[$name] = PsalmTest::fromPhptFile($path); } self::$batchResults = $tester->runBatch(self::$testData); } #[DataProvider('providePhptFiles')] public function testPhptFiles(string $name): void { Assert::assertThat( self::$batchResults[$name], self::$testData[$name]->constraint, ); } // ... data provider and discovery methods }
Important: Since all files in a batch group are analyzed in a single Psalm run, they share a global symbol table. Ensure that class and function names are unique across
.phptfiles within the same argument group, otherwise Psalm will reportDuplicateClass/DuplicateFunctionerrors.
See the source code in PsalmTester::runBatch() and related helper methods such as runGroup() for implementation details.