# Routing
# Introduction
Laravel JSON:API routing provides a fluent interface for defining the
the resource routes defined in the JSON:API specification. Routes
are added using the JsonApiRoute
facade.
TIP
Our routing implementation expects to access the Schema
class for each
resource type. So before you add resource routes, make sure you have
generated a Schema for the resource type.
# Defining Servers
To define routes available in a JSON:API server, register the API in
your routes/api.php
file as follows:
use App\Http\Controllers\Api\V1\PostController;
use App\Http\Controllers\Api\V1\TagController;
use App\Http\Controllers\Api\V1\UserController;
use LaravelJsonApi\Laravel\Facades\JsonApiRoute;
use LaravelJsonApi\Laravel\Routing\ResourceRegistrar;
JsonApiRoute::server('v1')
->prefix('v1')
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', PostController::class);
$server->resource('tags', TagController::class);
$server->resource('users', UserController::class);
});
As you can see with this example, this registers the JSON:API server called
v1
. This must match the key of the server in your jsonapi.servers
configuration array.
The object returned by the JsonApiRoute::server()
method allows a number
of typical Laravel routing methods to be called. In the example above,
we call prefix
and namespace
to set the URL prefix for the server,
and the controller namespace. The other available methods are described
below.
After calling any of these methods, we finish with a call to the resources
method. This receives a Closure
in which we register the routes for each
resource type in our server. This is similar to a
Laravel routing group. (opens new window)
# Controllers and Namespaces
On a fresh installation of a Laravel 8 application, you will need to provide the fully-qualified namespace of the controller when register JSON:API resource routes. For example:
use App\Http\Controllers\Api\V1\PostController;
use App\Http\Controllers\Api\V1\TagController;
use LaravelJsonApi\Laravel\Facades\JsonApiRoute;
use LaravelJsonApi\Laravel\Routing\ResourceRegistrar;
JsonApiRoute::server('v1')
->prefix('v1')
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', PostController::class);
$server->resource('tags', TagController::class);
});
As controllers are optional, it is also possible to use the default
JsonApiController
. For example:
use LaravelJsonApi\Laravel\Facades\JsonApiRoute;
use LaravelJsonApi\Laravel\Http\Controllers\JsonApiController;
use LaravelJsonApi\Laravel\Routing\ResourceRegistrar;
JsonApiRoute::server('v1')
->prefix('v1')
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', JsonApiController::class);
$server->resource('tags', JsonApiController::class);
});
Both the above examples work if the $namespace
property of your application's
RouteServiceProvider
is not set. This is the case in a fresh installation
of a Laravel >=8 application.
Traditionally, Laravel's route groups have allowed controller namespaces
to be set via groups. This works if the $namespace
property on your
RouteServiceProvider
is set to the base namespace of your controllers,
e.g. App\Http\Controllers
. Your application may be set up like this if it
was created before Laravel 8.
In this scenario you should call the namespace()
method when registering the
routes for a JSON:API sever. Providing the controller name to the resource()
method then becomes optional. In the following example, the namespace()
method is called, instructing Laravel that controllers for our server are in
the App\Http\Controllers\Api\V1
namespace:
JsonApiRoute::server('v1')
->prefix('v1')
->namespace('Api\V1')
->resources(function (ResourceRegistrar $server) {
// Expects controller to be `App\Http\Api\V1\PostController`
$server->resource('posts');
});
In this case, the controller is assumed to be the singular form of the resource
type. For example, the blog-posts
resource type would be expected to have a
BlogPostController
in the specified namespace.
If your controller does not conform to this convention, provide the
controller name as the second argument to the resource()
method:
JsonApiRoute::server('v1')
->prefix('v1')
->namespace('Api\V1')
->resources(function (ResourceRegistrar $server) {
// Controller is `App\Http\Api\V1\BlogPostController`
$server->resource('posts', 'BlogPostController');
});
When using controller namespacing, if you want to use the generic
JsonApiController
you must qualify the controller when providing it to the
resource()
method. For example:
use LaravelJsonApi\Laravel\Facades\JsonApiRoute;
use LaravelJsonApi\Laravel\Http\Controllers\JsonApiController;
use LaravelJsonApi\Laravel\Routing\ResourceRegistrar;
JsonApiRoute::server('v1')
->prefix('v1')
->namespace('Api\V1')
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', '\\' . JsonApiController::class);
});
# Server Domain
If you need to set a domain for your server, use the domain
method.
For example:
JsonApiRoute::server('v1')
->domain('api.myapp.com')
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', PostController::class);
$server->resource('tags', TagController::class);
$server->resource('users', UserController::class);
});
Or if you had wildcard sub-domains:
JsonApiRoute::server('v1')
->domain('{account}.myapp.com')
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', PostController::class);
$server->resource('tags', TagController::class);
$server->resource('users', UserController::class);
});
# Server Middleware
When you call JsonApiRoute::server()
, what you are effectively doing is
adding server routes that all run within the jsonapi
middleware.
This middleware receives the name of the JSON:API server the routes belong
to, so it knows which server to bootstrap.
If you call the middleware
method after the server
method, your
middleware will be added after the jsonapi
middleware.
In this example:
JsonApiRoute::server('v1')
->middleware('my-middleware1', 'my-middleware2')
->prefix('v1')
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', PostController::class);
});
All the server's routes will be wrapped in the following middleware:
jsonapi:v1
my-middleware1
my-middleware2
If you need to add middleware to run before the jsonapi
middleware,
use a Laravel route group. For example:
Route::middleware('my-middleware1')->group(function () {
JsonApiRoute::server('v1')
->middleware('my-middleware2')
->prefix('v1')
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', PostController::class);
});
});
In this example, the middleware order will be as follows:
my-middleware1
jsonapi:v1
my-middleware2
# Server Route Names
By default, the JsonApiRoute::server
method will set the route name to
the name of the server. You do not therefore need to call the name
method unless you want to override this behaviour.
In the following example, we override the default route name of v1
to
api:v1
:
JsonApiRoute::server('v1')
->name('api:v1')
->prefix('v1')
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', PostController::class);
});
# Defining Resources
To define routes for a specific resource type, we call the resource
method on the $server
variable passed to our resources
closure.
For example, the following registers routes for the posts
, tags
and
users
resources:
JsonApiRoute::server('v1')
->prefix('v1')
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', PostController::class);
$server->resource('tags', TagController::class);
$server->resource('users', UserController::class);
});
# Resource Actions
For each resource type, this registers the following actions:
Verb | URI | Action | Route Name |
---|---|---|---|
GET | /posts | index | posts.index |
POST | /posts | store | posts.store |
GET | /posts/{post} | show | posts.show |
PATCH | /posts/{post} | update | posts.update |
DELETE | /posts/{post} | destroy | posts.destroy |
TIP
The naming of these actions mirrors those used by Laravel's resource controllers. (opens new window)
# Partial Resource Routes
When declaring a resource route, you may specific a subset of actions the controller should handle instead of the full set of default actions:
$server->resource('posts')->only('index', 'show');
$server->resource('posts')->except('store', 'destroy');
We also include the readOnly
helper, which ensures that only the index
and show
actions are registered:
$server->resource('posts')->readOnly();
# Resource URI and Parameter
When registering the resource URIs, we use the dash-case form of the resource type by default. If you need to use something else, you can override this on your resource's schema.
For the parameter, Laravel does not allow dashes for parameter names.
We therefore underscore (snake-case) the singular form of the resource
type for the parameter name. E.g. blog-posts
will become blog_post
.
The parameter can be customised using the parameter
method:
$server->resource('blogPosts')->parameter('post');
The above example would register routes using /blog-posts
and
/blog-posts/{post}
.
# ID Constraints
We will automatically add ID contraints for a resource type. This is worked out from your schema's ID pattern.
# Naming Resource Routes
As shown in the table above, all resource routes are named. However, you
can override these names by calling the name
or names
methods.
For example:
$server->resource('posts')
->name('store', 'posts.build')
->name('update', 'posts.modify');
// is identical to...
$server->resource('posts')->names([
'store' => 'posts.build',
'update' => 'posts.modify',
]);
# Resource Middleware
It is possible to add middleware for all of a resource's routes. Just
use the middleware
method:
$server->resource('posts')->middleware('my_middleware1', 'my_middleware2');
Alternatively, you can specify middleware per resource action. To do that,
provide an array to the middleware()
method. Middleware that applies to
every action should use the "*"
key. Middleware for a specific action
should be keyed by that action.
For example:
$server->resource('posts')->middleware([
'*' => 'my_middleware1', // applies to all actions
'show' => 'my_middleware2', // apples to just the "show" action
'store' => ['my_middleware3', 'my_middleware4'], // use arrays for multiple
]);
# Defining Relationships
To define relationship routes for a specific resource type, call the
relationships
method when registering the resource type. The method
is provided with a closure that allows you to fluently define resource
relationship routes.
For example, if we want to register routes for a posts
resource's
author
, comments
and tags
relationships:
use LaravelJsonApi\Laravel\Routing\Relationships;
$server->resource('posts')->relationships(function (Relationships $relationships) {
$relationships->hasOne('author');
$relationships->hasMany('comments');
$relationships->hasMany('tags');
});
# To-One Actions
For each to-one relationship, the following actions are registered:
Verb | URI | Action | Route Name |
---|---|---|---|
GET | /author | showRelated | posts.author |
GET | /relationships/author | showRelationship | posts.author.show |
PATCH | /relationships/author | updateRelationship | posts.author.update |
# To-Many Actions
For each to-many relationship, the following actions are registered:
Verb | URI | Action | Route Name |
---|---|---|---|
GET | /tags | showRelated | posts.tags |
GET | /relationships/tags | showRelationship | posts.tags.show |
PATCH | /relationships/tags | updateRelationship | posts.tags.update |
POST | /relationships/tags | attachRelationship | posts.tags.attach |
DELETE | /relationships/tags | detachRelationship | posts.tags.detach |
# Partial Relationship Routes
When declaring a relationship, you may specify a subset of actions the
controller should handle instead of the full set of default actions.
To do this, use the short-hand action names of related
, show
,
update
, attach
and detach
.
For example:
$relationships->hasOne('author')->only('related', 'show');
$relationships->hasMany('tags')->except('update');
We also include the readOnly
helper, which ensures only the related
and
show
actions are registered:
$relationships->hasOne('author')->readOnly();
$relationships->hasMany('comments')->readOnly();
# Relationship URI
When registering relationship routes, we use the relationship field name
to work out the URI. We follow our convention of dash-casing relationships
field names in URIs. So if the relationship is called blogPost
,
the URI will be blog-post
.
If you need to use something else, you can configure this on the relation field in the resource's schema.
# Naming Relationship Routes
As shown in the table above, all relationship routes are named. However, you
can override these names by calling the name
or names
methods.
Use our short-hands of related
, show
, update
, attach
and detach
for the actions:
$relationships
->hasOne('author')
->name('related', 'author.related')
->name('show', 'author.relationships.show');
// is identical to...
$relationships->hasOne('author')->names([
'related' => 'author.related',
'show' => 'author.relationships.show',
]);
# Relationship Middleware
It is possible to add middleware for all of a relationship's routes. Just
use the middleware
method.
The following example adds middleware to our tags
relationship routes:
$relationships->hasMany('tags')->middleware('my_middleware1', 'my_middleware2');
Alternatively, you can specify middleware per relationship action. To do that,
provide an array to the middleware()
method. Middleware that applies to
every relationship action should use the "*"
key. Middleware for a specific
action should be keyed by that action. Use our short-hands of related
,
show
, update
, attach
and detach
for the actions:
For example:
$relationships->hasMany('tags')->middleware([
'*' => 'my_middleware1', // applies to all actions
'show' => 'my_middleware2', // apples to just the "show" action
'update' => ['my_middleware3', 'my_middleware4'], // use arrays for multiple
]);
# Route Model Binding
By default Laravel takes care of substituting parameter values for models using
its Route Model Binding implementation. (opens new window)
Laravel does this in the Illuminate\Routing\Middleware\SubstituteBindings
middleware.
In a fresh Laravel installation, this middleware is already included in the
api
middleware group. This means when you use JsonApiRoute::server()
helper
method to define JSON:API routes within your routes/api.php
file, the
JSON:API server routes are defined after the SubstituteBindings
middleware
runs.
The JSON:API implementation however does work without the SubstituteBindings
middleware. This is because the JSON:API middleware is able to substitute the
resource binding for the route without resorting to Laravel's implementation.
For example, when you define a route for GET /api/v1/posts/{post}
, the
JSON:API implementation can substitute the post
parameter for a Post
model
itself.
In fact, it is preferrable that the JSON:API implementation takes care of
substituting the binding. This is because JSON:API bindings are substituted
after your server's serving()
hook is called - which means if you apply
any global scopes in that hook, they will affect whether or not a model can
be found and therefore whether a 404 Not Found
response is sent.
If your API routes have no other bindings to substitute, we therefore
recommend that you remove Laravel's SubstituteBindings
middleware from your
JSON:API routes.
You can do this using the withoutMiddleware()
method when registering your
JSON:API routes:
JsonApiRoute::server('v1')
->prefix('v1')
->withoutMiddleware(\Illuminate\Routing\Middleware\SubstituteBindings::class)
->resources(function (ResourceRegistrar $server) {
$server->resource('posts', PostController::class);
});
# Writing Regular Laravel Routes
You are not constrained to just using the routing from this package!
If you need to create some routes and actions that deviate from JSON:API conventions, you can just create an endpoint in exactly the way you would when using Laravel - i.e. register a route, write a controller.
If you want to return a JSON:API response from the controller, you can use one of the package's response classes.