laravel-testing
Original:🇺🇸 English
Translated
Laravel 13 testing with Pest PHP 4 or PHPUnit 12. Use when writing feature tests, unit tests, or any test code in a Laravel application. Triggers on tasks involving HTTP tests, model factories, database assertions, mocking facades, authentication testing, or test organisation patterns.
2installs
Added on
NPX Install
npx skill4agent add asyrafhussin/agent-skills laravel-testingTags
Translated version includes tags in frontmatterSKILL.md Content
View Translation Comparison →Laravel 13 Testing — Pest PHP 4 & PHPUnit 12
Supports both Pest PHP 4 and PHPUnit 12. See Framework Detection below.PHPUnit version note: Laravel 13 ships within its defaultphpunit/phpunit: ^12.5.12. All patterns in this skill are compatible with PHPUnit 11, 12, and 13.composer.json
Comprehensive testing guide for Laravel 13 applications. Contains 24 rules across 6 categories for writing fast, readable, and reliable tests. Supports both Pest PHP 4 and PHPUnit 12 (Laravel 13 default).
Framework Detection
Before writing or reviewing any test code, detect which testing framework the project uses:
Step 1 — Check composer.json
bash
# Look for these in require-dev:
# "pestphp/pest" → Pest
# "phpunit/phpunit" (without pest) → PHPUnit- If is present → use Pest syntax
pestphp/pest - If only is present → use PHPUnit syntax
phpunit/phpunit - If both are present → Pest takes priority (Pest runs on top of PHPUnit)
Step 2 — Check for tests/Pest.php
- If exists → Pest is configured, use Pest syntax
tests/Pest.php
Step 3 — If still unclear, ask the user
"I couldn't detect the testing framework. Does this project use Pest PHP or PHPUnit?"
Syntax Reference
Test Declaration
| Pest | PHPUnit | |
|---|---|---|
| Test function | | |
| Readable name | | |
| Grouping | | Test class name / nested classes |
| Trait application | | |
| Before each | | |
| After each | | |
| Parameterised | | |
| Global setup | | Base |
Core assertions (identical in both frameworks)
assertStatusassertJsonassertJsonPathassertDatabaseHasassertModelExistsactingAsMail::fake()Queue::fake()Event::fake()Notification::fake()Storage::fake()When to Apply
Reference these guidelines when:
- Writing feature or unit tests for Laravel
- Testing HTTP endpoints and API responses
- Creating factories and test data
- Asserting database state after operations
- Faking Mail, Queue, Notification, or Event facades
- Testing authenticated routes and API tokens
- Organising tests with describe blocks, datasets, or test classes
Rule Categories by Priority
| Priority | Category | Impact | Prefix |
|---|---|---|---|
| 1 | HTTP & Feature Tests | CRITICAL | |
| 2 | Model Factories | CRITICAL | |
| 3 | Database Assertions | HIGH | |
| 4 | Faking Services | HIGH | |
| 5 | Authentication Testing | HIGH | |
| 6 | Test Organisation Patterns | MEDIUM | |
Quick Reference
1. HTTP & Feature Tests (CRITICAL)
- - Arrange/Act/Assert with factories — Pest + PHPUnit examples
http-test-structure - - assertStatus, assertJson, assertRedirect, assertJsonMissing
http-assert-response - - Fluent assertJson with AssertableJson closure
http-assert-json-fluent - - RefreshDatabase vs DatabaseTransactions — when to use each
http-refresh-database
2. Model Factories (CRITICAL)
- - Define factories with typed fake data and PHP 8.3 syntax
factory-define - - Factory states for distinct test scenarios
factory-states - - sequence() for varied data across multiple records
factory-sequences - - has(), for(), recycle(), afterCreating()
factory-relationships
3. Database Assertions (HIGH)
- - assertDatabaseHas, assertModelExists for presence checks
db-assert-has - - assertDatabaseMissing, assertModelMissing for deletion
db-assert-missing - - assertSoftDeleted, trashed() factory state
db-assert-soft-deletes
4. Faking Services (HIGH)
- - Mail::fake(), assertSent vs assertQueued, assertNothingSent
fake-mail - - Queue::fake(), assertPushed, assertPushedOn
fake-queue - - Notification::fake(), assertSentTo, assertCount
fake-notification - - Event::fake(), assertDispatched, assertNotDispatched
fake-event - - Storage::fake(), UploadedFile::fake(), assertExists
fake-storage - - Agent::fake(), assertPrompted, preventStrayPrompts (Laravel 13+)
fake-ai-agent - - Image::fake(), Audio::fake(), Transcription::fake() (Laravel 13+)
fake-ai-media - - Embeddings::fake(), Reranking::fake(), Files::fake(), Stores::fake() (Laravel 13+)
fake-ai-data
5. Authentication Testing (HIGH)
- - actingAs() for session/web authenticated tests
auth-acting-as - - Sanctum::actingAs() for API token authentication
auth-sanctum
6. Test Organisation Patterns (MEDIUM)
- - describe()/it() (Pest) or test class organisation (PHPUnit)
pest-describe-it - - with() datasets (Pest) or #[DataProvider] (PHPUnit)
pest-datasets - - beforeEach/afterEach (Pest) or setUp/tearDown (PHPUnit)
pest-hooks
Essential Patterns
Pest
php
<?php
use App\Models\User;
use Illuminate\Foundation\Testing\RefreshDatabase;
uses(RefreshDatabase::class);
test('authenticated user can create a post', function () {
$user = User::factory()->create();
$this->actingAs($user)
->postJson('/api/posts', ['title' => 'Hello World', 'body' => 'Content.'])
->assertStatus(201)
->assertJsonPath('data.title', 'Hello World');
$this->assertDatabaseHas('posts', ['title' => 'Hello World', 'user_id' => $user->id]);
});PHPUnit
php
<?php
namespace Tests\Feature;
use App\Models\User;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Tests\TestCase;
class PostControllerTest extends TestCase
{
use RefreshDatabase;
public function test_authenticated_user_can_create_a_post(): void
{
$user = User::factory()->create();
$this->actingAs($user)
->postJson('/api/posts', ['title' => 'Hello World', 'body' => 'Content.'])
->assertStatus(201)
->assertJsonPath('data.title', 'Hello World');
$this->assertDatabaseHas('posts', ['title' => 'Hello World', 'user_id' => $user->id]);
}
}How to Use
Read individual rule files for detailed explanations and code examples.
Each rule file contains:
- YAML frontmatter with metadata (title, impact, tags)
- Brief explanation of why it matters
- Bad Example with explanation
- Good Example with both Pest and PHPUnit where syntax differs
- Laravel 13 specific context and references
Full Compiled Document
For the complete guide with all rules expanded:
AGENTS.md