Skip to main content
Version: Laravel: 4.x (current)

Responses

Overview

Scribe gives you multiple ways to provide example responses for your endpoint:

  • manually:
    • you can use the @response tag, followed by an example response
    • you can place an example in a file and reference it with the @responseFile tag
  • automatically:
    • Scribe can generate a response by faking a request to your endpoint (a "response call")
    • Scribe can generate a response from the @apiResource tags (if you're using Eloquent API resources) or @transformer tags (if you're using Fractal transformers)

You can use all of these strategies within the same endpoint. Scribe will display all the responses it finds.

Additionally, you can add descriptions for fields in your responses.

tip

We'll describe all of these here, but for more details and valid values, you can see the list of supported annotations.

@response/#[Response]

You can provide an example response for an endpoint by using the @response tag with valid JSON (on one line or multiple). Alternatively, you can use the #[Response] attribute with either a JSON string or a PHP array.

👉Full reference and more examples

/**
* @response {
* "id": 4,
* "name": "Jessica Jones",
* "roles": ["admin"]
* }
*/
public function show($id)
{
return User::findOrFail($id);
}

By default, a status code of 200 is assumed, but you can specify a different one:

/**
* @response 201 {"id": 4, "name": "Jessica Jones"}
*/

This means you can define multiple responses from the same endpoint. With @response, you can use the status and scenario fields to add context to each response. With #[Response], use the status and description arguments.

/**
* @response scenario=success {
* "id": 4,
* "name": "Jessica Jones"
* }
* @response status=404 scenario="user not found" {"message": "User not found"}
*/

If an endpoint returns a file or some other binary response, you can use <<binary>> as the value of the response, followed by an optional description.

/**
* @response <<binary>> The resized image
*/

@responseFile/#[ResponseFromFile]

@responseFile works similarly to @response, but instead of inlining the response, you pass a file containing your JSON response. This can be helpful for large responses.

To use this annotation, place the response as a JSON string in a file somewhere in your project directory and specify the relative path to it. For instance, we can put this response in a file named users.get.json in storage/responses/:

{"id":4,"name":"Jessica Jones"}

Then in the controller:

/**
* @responseFile storage/responses/users.get.json
*/
public function getUser(int $id)
{
// ...
}

👉Full reference and more examples

tip

You can specify an absolute path, a path relative to your project root, or a path relative to your Laravel storage directory.

Like with @response. you can include a status code, or have multiple @responseFile tags on a single method, distinguished by status code and/or scenario/description.

/**
* @responseFile responses/users.get.json
* @responseFile status=200 scenario="when authenticated as admin" responses/user.get.admin.json
* @responseFile 404 responses/user.not_found.json
*/

You can also "merge" responses into one. To do this, add the JSON you want to merge after the file path. For instance, supposing our generic "not found" response located in storage/responses/model.not_found.json says:

{
"type": "Model",
"result": "not found"
}

We can change the type to User on the fly like this:

/**
* @responseFile 404 responses/model.not_found.json {"type": "User"}
*/

This will be parsed and merged with the response from the file to give:

{
"type": "User",
"result": "not found"
}

Response calls

If Scribe doesn't find any 2xx responses for your endpoint, it will attempt to make a fake HTTP request to the endpoint to get a response (known as a "response call"). We say a fake request, because it doesn't call any actual URLs; it just uses the information it's extracted about your endpoints (headers, body parameters, etc) to build a HTTP request and passes it to your Laravel app as though it came via a URL.

tip

If you don't want a parameter in your docs to be included in a response call, you can specify No-example at the end of its description.

Some key things about response calls:

  1. Response calls aim to be "safe" by default. We don't want to accidentally delete a user or anything like that. By default:

    • Scribe will try to start a database transaction and roll it back afterwards (see database transactions for some important details about that).
    • Scribe will only make response calls for GET endpoints.
  2. Response calls are configurable. Most of the configuration is located in the apply.response_calls section for each route group in your Scribe config. You can:

    • choose which HTTP methods are safe for response calls, or disable it entirely (methods)
    • specify Laravel config values that should be set for the response call (config)
    • specify query, body and file parameters to be included in the response call (queryParams, bodyParams, and fileParams)

Authentication and customization

If your endpoints are authenticated, Scribe will use the configured use_value in your auth config. However, if you need more customization, you can use the beforeResponseCall() method to manually set it up. Typically, you'd do this in the boot() method of your AppServiceProvider.

app\Providers\AppServiceProvider.php

use Knuckles\Camel\Extraction\ExtractedEndpointData;
use Symfony\Component\HttpFoundation\Request;
use Knuckles\Scribe\Scribe;

public function boot()
{
// Be sure to wrap in a `class_exists()`,
// so production doesn't break if you installed Scribe as dev-only
if (class_exists(\Knuckles\Scribe\Scribe::class)) {
Scribe::beforeResponseCall(function (Request $request, ExtractedEndpointData $endpointData) {
$token = User::first()->api_token;
$request->headers->add(["Authorization" => "Bearer $token"]);
});
}
// ...
}

The callback you provide will receive the current Symfony\Component\HttpFoundation\Request instance and the details of the current endpoint being extracted. If you have database transactions configured, they will already be activated at that point, allowing you to modify your database freely, and have your changes rolled back after the request.

tip

You can use beforeResponseCall() to modify anything about the outgoing request, not just authentication.

Database transactions

Response calls involve invoking your endpoint, which may lead to changes in your database. To avoid any permanent changes, Scribe tries to run response calls within a database transaction.

However, Scribe only knows of your default database connection. If you're using multiple database connections, you'll need to add them to the database_connections_to_transact array in your config.

Alternatively, you can switch to your test database.

Recommendations

To get the best value from response calls, you should make sure to configure your environment to return production-type responses. For instance, you'll want to turn debug mode off, so that 404/500 responses will return formatted error messages rather than exceptions and stack traces. You might also want to switch the app database to your test database. Additionally, if the endpoints that will be called trigger external services (for example, sending email), you'll want to use dummy service providers.

There are two ways to do this:

  1. Use the apply.response_calls.config key in your Scribe config to override the Laravel config:

    config/scribe.php
    'response_calls' => [
    'config' => [
    'app.debug' => false,
    'database.default' => 'sqlite',
    ],
    ],
  2. Add a .env.docs file to your repo with the desired config. This env file will be loaded when you run php artisan scribe:generate --env docs, and will apply to all of Scribe, not just response calls, so you can use this to configure a database where models will be fetched from when using @apiResource or @transformer (see How model instances are generated).

    APP_DEBUG=false
    DB_DATABASE=sqlite

API resources

If your endpoint uses Eloquent API resources for its response, you can use the @apiResource annotations to tell Scribe how to generate a sample response without actually calling the endpoint. To do this, you'll need two annotations:

  • @apiResource/#[ResponseFromApiResource]: the name of the resource class.
    • If you're returning a resource collection via YourResource::collection($things), use @apiResourceCollection or pass collection: true to #[ResponseFromApiResource]. If you're using new YourResourceCollection($things), this is optional.
  • @apiResourceModel: the Eloquent model to be passed to the resource. You should use @apiResourceModel alongside either @apiResource or @apiResourceCollection.

👉Full reference and more examples

Examples:

App\Http\Resources\UserResource.php
class UserResource extends JsonResource
{
public function toArray($request)
{
return [ 'id' => $this->id ];
}
}
UserController.php
/**
* @apiResource App\Http\Resources\UserResource
* @apiResourceModel App\Models\User
*/
public function showUser(User $user)
{
return new UserResource($user);
}

/**
* @apiResourceCollection App\Http\Resources\UserResource
* @apiResourceModel App\Models\User
*/
public function listUsers()
{
return UserResource::collection(User::all());
}

/**
* @apiResourceCollection App\Http\Resources\UserCollection
* @apiResourceModel App\Models\User
*/
public function listMoreUsers()
{
return new UserCollection(User::all());
}

Scribe will generate an instance (or instances) of the model and pass the model(s) to the resource class to get the example response. To understand how Scribe generates an instance of your model and how you can customize that, you should check out the section on How model instances are generated.

As with @response, you can also specify a status code:

/**
* @apiResource 201 App\Http\Resources\UserResource
* @apiResourceModel App\Models\User
*/

Pagination

If your endpoint returns a paginated resource response, you can tell Scribe how to paginate by:

  • using the paginate field on @apiResourceModel or .
  • using either the paginate, simplePaginate, or cursorPaginate argument on #[ResponseFromApiResource].
/**
* @apiResourceCollection App\Http\Resources\UserCollection
* @apiResourceModel App\Models\User paginate=10
*/
public function listMoreUsers()
{
return new UserCollection(User::paginate(10));
}

/**
* @apiResourceCollection App\Http\Resources\UserCollection
* @apiResourceModel App\Models\User paginate=15,simple
*/
public function listMoreUsers()
{
return new UserCollection(User::simplePaginate(15));
}

/**
* @apiResourceCollection App\Http\Resources\UserCollection
* @apiResourceModel App\Models\User paginate=15,cursor
*/
public function listMoreUsers()
{
return new UserCollection(User::cursorPaginate(15));
}

Additional Data

If your endpoint returns additional fields using the API resource's additional() method, you can indicate this with @apiResourceAdditional or the merge: argument to #[ResponseFromApiResource]:

class UserController extends Controller
{
/**
* @apiResource App\Http\Resources\UserResource
* @apiResourceModel App\Models\User
* @apiResourceAdditional result=success message="User created successfully"
*/
public function store($request): JsonResponse
{
// ... some store code ...
return UserResource::make($event)->additional([
'result' => __('success'),
'message' => __('User created successfully'),
]);
}
}

Produces output (with default data-wrapper):

{
"data": {
"id": 1
},
"result": "success",
"message": "User created successfully"
}

Transformers

If your endpoint uses "transformers" (via the league/fractal package) for its response, you can use the transformer annotations to tell Scribe how to generate a sample response without actually calling the endpoint. To do this, you'll need two annotations:

  • @transformer: the name of the transformer class. Use @transformerCollection instead if you're returning a collection.
  • @transformerModel: the model to be passed to the resource. You should use @transformerModel alongside either of the other two.
  • #[ResponseFromTransformer]: attribute alternative to all three

👉Full reference and more examples

tip

@transformerModel is optional. If you don't specify it, Scribe will attempt to use the class of the first parameter to the transformer's transform(MyModel $model) method (ie MyModel).

For example:

/**
* @transformer App\Transformers\UserTransformer
*/
public function showUser(int $id)
{
$resource = new Fractal\Resource\Item(User::find($id), new UserTransformer());
return (new Fractal\Manager)->createData($resource)->toArray();
}

/**
* @transformerCollection App\Transformers\UserTransformer
* @transformerModel App\Models\User
*/
public function listUsers()
{
$resource = new Fractal\Resource\Collection(User::all(), new UserTransformer());
return (new Fractal\Manager)->createData($resource)->toArray();
}

Scribe will generate an instance (or instances) of the model and pass the model(s) to the transformer to get the example response. To understand how Scribe generates an instance of your model and how you can customize that, you should check out the section on How model instances are generated.

If your response data is nested within a Fractal resource key, you can specify it with the resourceKey field on @transformerModel:

/**
* @transformer App\Transformers\UserTransformer
* @transformerModel App\Models\User resourceKey=user
*/

You can also specify a status code:

/**
* @transformer 201 App\Transformers\UserTransformer
* @transformerModel App\Models\User
*/

Pagination

If your endpoint uses a paginator with the transformer, you can tell Scribe how to paginate via an additional tag, @transformerPaginator, or with the paginate argument of the #[ResponseFromTransformer] attribute.

/**
* @transformerCollection App\Transformers\UserTransformer
* @transformerModel App\Models\User
* @transformerPaginator League\Fractal\Pagination\IlluminatePaginatorAdapter 15
*/
public function listMoreUsers()
{
$users = User::paginate(15)->getCollection();
$transformer = new Fractal\Resource\Collection($users, new UserTransformer(), 'data');
$transformer->setPaginator(new IlluminatePaginatorAdapter($users));
return (new Fractal\Manager)->createData($users)->toArray();
}

How model instances are generated

When generating responses from @apiResourceModel and @transformerModel tags, Scribe needs to generate a sample model to pass to the resource or transformer. Here's the process Scribe follows:

  1. First, it tries the Eloquent model factory: YourModel::factory()->create().

    note

    create() saves the model to your database. But no worries! Scribe will use a database transaction, using the same database transaction rules as for response calls.

  2. If that fails, it tries YourModel::factory()->make().

  3. If that fails, Scribe calls YourModel::first() to retrieve the first model from the database.

  4. If that fails, Scribe creates an instance using new YourModel().

You can customise this process in a few ways:

Changing the strategies

You can change the order of these strategies, or remove the ones you don't need by editing the examples.models_source config item. By default, it's set to ['factoryCreate', 'factoryMake', 'databaseFirst'], corresponding to the three main strategies above.

Applying states

If you want specific states to be applied to your model factory, you can use the states field on @apiResourceModel or @transformerModel. Separate multiple states with a comma. If you're using attributes, use the factoryStates: argument with an array of states.

/**
* @apiResourceCollection App\Http\Resources\UserCollection
* @apiResourceModel App\Models\User states=student,verified
*/

Loading relations

If you want specific relations to be loaded with your model, you can use the with attribute on @apiResourceModel or @transformerModel. Separate multiple relations with a comma.

/**
* @apiResource App\Http\Resources\UserResource
* @apiResourceModel App\Models\User with=teacher,classmates
*/

Nested relations can be specified using dot notation. HasMany and BelongsToMany relations are supported for factories:

/**
* @apiResource App\Http\Resources\AuthorCollection
* @apiResourceModel App\Models\Author with=posts.categories
*/
note

If your many-to-many relation has extra columns in its pivot table, you'll need to create a helper method named pivot<RelationName> in both factories. For instance, if our post_category table has a priority column:

class PostFactory extends Factory
{
protected $model = Post::class;

public function definition(): array
{
// ...
}

public function pivotCategories(): array
{
return [
'priority' => $this->faker->randomNumber(),
];
}
}

class CategoryFactory extends Factory
{
protected $model = Category::class;

public function definition(): array
{
// ...
}

public function pivotCategories(): array
{
return [
'priority' => $this->faker->randomNumber(),
];
}
}

Response fields

You can add descriptions for fields in your response by adding @responseField/#[ResponseField] to your controller method. If you're using an Eloquent @apiResource to generate your response, you can place the annotation on your resource's toArray() method instead.

👉Full reference and more examples

/**
* @responseField id The id of the newly created word
*/

You can leave out the type, and Scribe will figure it out from your responses.

tip

You don't need to specify the full field path if the field is inside an array of objects or wrapped in pagination data. For instance, the above annotation will work fine for all of these responses:

{ "id": 3 }
// Array of objects
[
{ "id": 3 }
]
// Inside "data" key
{
"data": [
{ "id": 3 }
]
}

However, you can also specify the type of the field if you wish:

/**
* @responseField id integer The id of the newly created word
*/