The go-to PHP IDE with extensive out-of-the-box support for Laravel and its ecosystem.

Spectator

hotmeteor/spectator image

Spectator stats

Downloads
380.7K
Stars
246
Open Issues
18
Forks
37

View on GitHub →

Testing helpers for your OpenAPI spec

Spectator

Spectator provides light-weight OpenAPI testing tools you can use within your existing Laravel test suite.

Write tests that verify your API spec doesn't drift from your implementation.

Installation

You can install the package through Composer.

composer require hotmeteor/spectator --dev

Then, publish the config file of this package with this command:

php artisan vendor:publish --provider="Spectator\SpectatorServiceProvider"

The config file will be published in config/spectator.php.

Sources

Sources are references to where your API spec lives. Depending on the way you or your team works, or where your spec lives, you may want to configure different sources for different environments.

As you can see from the config, there's three source types available: local, remote, and github. Each source requires the folder where your spec lives to be defined, not the spec file itself. This provides flexibility when working with multiple APIs in one project, or an API fragmented across multiple spec files.


Local Example

## Spectator config
 
SPEC_SOURCE=local
SPEC_PATH=/spec/reference

Remote Example

This is using the raw access link from Github, but any remote source can be specified. The SPEC_URL_PARAMS can be used to append any additional parameters required for the remote url.

## Spectator config
 
SPEC_PATH="https://raw.githubusercontent.com/path/to/repo"
SPEC_URL_PARAMS="?token=ABEDC3E5AQ3HMUBPPCDTTMDAFPMSM"

Github Example

This uses the Github Personal Access Token which allows you access to a remote repo containing your contract.

You can view instructions on how to obtain your Personal Access Token from Github at this link .

Important to note than the SPEC_GITHUB_PATH must included the branch (ex: main) and then the path to the directory containing your contract.

## Spectator config
 
SPEC_GITHUB_PATH='main/contracts'
SPEC_GITHUB_REPO='orgOruser/repo'
SPEC_GITHUB_TOKEN='your personal access token'

Specifying Your File In Your Tests

In your tests you will declare the spec file you want to test against:

public function testBasicExample()
{
Spectator::using('Api.v1.json');
 
// ...

Testing

Paradigm Shift

Now, on to the good stuff.

At first, spec testing, or contract testing, may seem counter-intuitive, especially when compared with "feature" or "functional" testing as supported by Laravel's HTTP Tests. While functional tests are ensuring that your request validation, controller behavior, events, responses, etc. all behave the way you expect when people interact with your API, contract tests are ensuring that requests and responses are spec-compliant, and that's it.

Writing Tests

Spectator adds a few simple tools to the existing Laravel testing toolbox.

Here's an example of a typical JSON API test:

<?php
 
class ExampleTest extends TestCase
{
/**
* A basic functional test example.
*
* @return void
*/
public function testBasicExample()
{
$response = $this->postJson('/user', ['name' => 'Sally']);
 
$response
->assertStatus(201)
->assertJson([
'created' => true,
]);
}
}

And here's an example of a contract test:

<?php
 
use Spectator\Spectator;
 
class ExampleTest extends TestCase
{
/**
* A basic functional test example.
*
* @return void
*/
public function testBasicExample()
{
Spectator::using('Api.v1.json');
 
$response = $this->postJson('/user', ['name' => 'Sally']);
 
$response
->assertValidRequest()
->assertValidResponse(201);
}
}

The test is verifying that both the request and the response are valid according to the spec, in this case located in Api.v1.json. This type of testing promotes TDD: you can write endpoint contract tests against your endpoints first, and then ensure your spec and implementation are aligned.

Within your spec, each possible response should be documented. For example, a single POST endpoint may result in a 2xx, 4xx, or even 5xx code response. Additionally, your endpoints will likely have particular parameter validation that needs to be adhered to. This is what makes contract testing different from functional testing: in functional testing, successful and failed responses are tested for outcomes; in contract testing, requests and responses are tested for conformity and outcomes don't matter.

Debugging

For certain validation errors, a special exception message is thrown which shows error message(s) displayed alongside the expected schema. For example:

---
 
The properties must match schema: data
All array items must match schema
The required properties (name) are missing
 
object++ <== The properties must match schema: data
status*: string
data*: array <== All array items must match schema
object <== The required properties (name) are missing
id*: string
name*: string
slug: string?
 
---

A few custom symbols are used:

  • "++": Object supports additionalProperties
  • "*": Item is required
  • "?": Item can be nullable

Usage

Define the spec file to test against. This can be defined in your setUp() method or in a specific test method.

<?php
 
use Spectator\Spectator;
 
class ExampleTest extends TestCase
{
public function setUp(): void
{
parent::setUp();
 
Spectator::using('Api.v1.json');
}
 
public function testApiEndpoint()
{
// Test request and response...
}
 
public function testDifferentApiEndpoint()
{
Spectator::using('Other.v1.json');
 
// Test request and response...
}
}

When testing endpoints, there are a few new methods:

$this->assertValidRequest();
$this->assertValidResponse($status = null);
$this->assertValidationMessage('Expected validation message');
$this->assertErrorsContain('Check for single error');
$this->assertErrorsContain(['Check for', 'Multiple Errors']);

Of course, you can continue to use all existing HTTP test methods:

$this
->actingAs($user)
->postJson('/comments', [
'message' => 'Just over here spectating',
])
->assertCreated()
->assertValidRequest()
->assertValidResponse();

That said, mixing functional and contract testing may become more difficult to manage and read later.

Instead of using the built-in ->assertStatus($status) method, you may also verify the response that is valid is actually the response you want to check. For example, you may receive a 200 or a 202 from a single endpoint, and you want to ensure you're validating the correct response.

$this
->actingAs($user)
->postJson('/comments', [
'message' => 'Just over here spectating',
])
->assertValidRequest()
->assertValidResponse(201);

When exceptions are thrown that are not specific to this package's purpose, e.g. typos or missing imports, the output will be formatted by default with a rather short message and no stack trace. This can be changed by disabling Laravel's built-in validation handler which allows for easier debugging when running tests.

This can be done in a few different ways:

class ExampleTestCase
{
public function setUp(): void
{
parent::setUp();
 
Spectator::using('Api.v1.json');
 
// Disable exception handling for all tests in this file
$this->withoutExceptionHandling();
}
 
// ...
}
class ExampleTestCase
{
public function test_some_contract_test_example(): void
{
// Only disable exception handling for this test
$this->withouthExceptionHandling();
 
// Test request and response ...
 
}
}

Core Concepts

Approach

Spectator works by registering a custom middleware that performs request and response validation against a spec.

Dependencies

For those interested in contributing to Spectator, it is worth familiarizing yourself with the core dependencies used for spec testing:

  • cebe/php-openapi: Used to parse specs into usable arrays
  • opis/json-schema: Used to perform validation of an object/array against a spec

Sponsors

A huge thanks to all our sponsors who help push Spectator development forward! In particular, we'd like to say a special thank you to our partners:

If you'd like to become a sponsor, please see here for more information. 💪

Credits

Made with contributors-img.

License

The MIT License (MIT). Please see License File for more information.

hotmeteor photo

I build software that scratches my own itches, and then love to share it with others.

Cube

Laravel Newsletter

Join 40k+ other developers and never miss out on new tips, tutorials, and more.


Hotmeteor Spectator Related Articles

Regex Helpers for Laravel image

Regex Helpers for Laravel

Read article
Immutable Date Casting in Laravel 8.53 image

Immutable Date Casting in Laravel 8.53

Read article
Inertia.js Adapter for Statamic image

Inertia.js Adapter for Statamic

Read article
Manage Environment Variables with Eco image

Manage Environment Variables with Eco

Read article
Harpoon: Next generation time tracking and invoicing logo

Harpoon: Next generation time tracking and invoicing

The next generation time-tracking and billing software that helps your agency plan and forecast a profitable future.

Harpoon: Next generation time tracking and invoicing
Honeybadger logo

Honeybadger

Simple developer-focused application monitoring for Laravel. Error tracking, log management, uptime monitoring, status pages, and more!

Honeybadger
Shift logo

Shift

Running an old Laravel version? Instant, automated Laravel upgrades and code modernization to keep your applications fresh.

Shift
CodeKudu logo

CodeKudu

Stand-ups, Retrospectives, and 360° Feedback for the entire team. 50% off with code LARAVELNEWS.

CodeKudu
Tighten logo

Tighten

We help companies turn great ideas into amazing apps, products, and services.

Tighten
Curotec logo

Curotec

World class Laravel experts with GenAI dev skills. LATAM-based, embedded engineers that ship fast, communicate clearly, and elevate your product. No bloat, no BS.

Curotec