Code review at scale is broken. Here’s how Augment Code is fixing it.

JSON API Resources in Laravel

Published on by

JSON API Resources in Laravel image

Building APIs in Laravel is a passion of mine, and I have spent a lot of time searching for the perfect way to return consistent JSON:API friendly resources so that I can find and use a standard.

In the past, I have used a cobbled-together solution that would just about manage to achieve what I needed, but it was quite a bit of work. The negatives outweighed the benefits of this approach, as the development time spent achieving it just felt like it wasn't worth it.

Luckily for you and me, Tim MacDonald built a fantastic package for this use case. It allows us to build and return JSON:API compliant resources that are easy to use. Let's walk through how this works.

Typically when building an API resource, we would extend Laravels JsonResource or CollectionResource depending on what we wanted to achieve. Our typical Resource might look like the following:

class PostResource extends JsonResource
{
public function toArray($request): array
{
return [
'id' => $this->id,
'type' => 'post',
'attributes' => [
'title' => $this->title,
'slug' => $this->slug,
'content' => $this->content,
]
];
}
}

We are adding a simple implementation to "fake" a JSON:API resource at a basic level. But as you can see, this is a little - yuck. Let's install the package by Tim:

composer require timacdonald/json-api

Now we can refactor the above resource to follow JSON:API standards. Let me walk you through the changes.

Firstly, we need to change what class we are extending from JsonResource to JsonApiResource:

class PostResource extends JsonApiResource

The next thing we need to make sure that we change is to remove the toArray method, as this package will handle this under the hood for you - instead, we use different methods that are useful for JSON:API standards. For example, to add attributes to your resource, you can use the following method:

protected function toAttributes(Request $request): array
{
return [
'title' => $this->title,
'slug' => $this->slug,
'content' => $this->content,
];
}

Now let's look at relationships. Previously we would do something similar to:

return [
'id' => $this->id,
'type' => 'post',
'attributes' => [
'title' => $this->title,
'slug' => $this->slug,
'content' => $this->content,
],
'relationships' => [
'category' => new CategoryResource(
resource: $this->whenLoaded('category'),
),
]
];

This isn't a bad way to do it by any means; however, let us have a look at how we would add these on the JSON:API package:

protected function toRelationships(Request $request): array
{
return [
'category' => fn () => new CategoryResource(
resource: $this->category,
),
];
}

So this time, we are passing through a closure to be evaluated to return the relationship. This is a very powerful way to do this, as it opens us up to add very custom behavior to relationship loading - meaning that we can load different conditional relationships or run authorization on the resource. Another point to note is that the closure is only evaluated when the client has included the relationship - making it a little cleaner.

Taking this another step further, adding links to your API resources is something that I feel is an important step. Adding these links makes your API navigatable, allowing clients to follow links as required programmatically. Previously I would add another array entry to add these and use the route helper to echo these out. The JSON:API package has an alternative approach, which is particularly fluent:

protected function toLinks(Request $request): array
{
return [
Link::self(route('api:v1:posts:show', $this->resource)),
];
}

As you can see - it is fluent, simple, and will generate the correct link for you. Of course, you are welcome to add what you need here, but it will add the links in a JSON:API standardized way - so you don't have to.

Finally, meta-data, in JSON:API, you can add additional information within the meta-object so you can add documentation links or anything you might need to pass back with an API resource (depending on your API design). There aren't a million use cases for this, but the package does support it. So let's have a look at this to understand it.

protected function toMeta(Request $request): array
{
return [
'depreciated' => false,
'docs' => 'https://docs.domain/com/resources/posts',
];
}

As you can see above, we can add depreciation warnings so that clients can be notified of resources they need to consider changing - and a link to the docs explaining the replacement approach.

How do you build your API resources? Are you following any specific standards? Let us know on Twitter!

Steve McDougall photo

Educator and Content creator, freelance consultant, API evangelist

Cube

Laravel Newsletter

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

image
Curotec

Curotec helps software teams hire the best Laravel developers in Latin America. Click for rates and to learn more about how it works.

Visit Curotec
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
Bacancy logo

Bacancy

Supercharge your project with a seasoned Laravel developer with 4-6 years of experience for just $3200/month. Get 160 hours of dedicated expertise & a risk-free 15-day trial. Schedule a call now!

Bacancy
Tinkerwell logo

Tinkerwell

The must-have code runner for Laravel developers. Tinker with AI, autocompletion and instant feedback on local and production environments.

Tinkerwell
Cut PHP Code Review Time & Bugs into Half with CodeRabbit logo

Cut PHP Code Review Time & Bugs into Half with CodeRabbit

CodeRabbit is an AI-powered code review tool that specializes in PHP and Laravel, running PHPStan and offering automated PR analysis, security checks, and custom review features while remaining free for open-source projects.

Cut PHP Code Review Time & Bugs into Half with CodeRabbit
Get expert guidance in a few days with a Laravel code review logo

Get expert guidance in a few days with a Laravel code review

Expert code review! Get clear, practical feedback from two Laravel devs with 10+ years of experience helping teams build better apps.

Get expert guidance in a few days with a Laravel code review
Kirschbaum logo

Kirschbaum

Providing innovation and stability to ensure your web application succeeds.

Kirschbaum
Shift logo

Shift

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

Shift
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
Lucky Media logo

Lucky Media

Get Lucky Now - the ideal choice for Laravel Development, with over a decade of experience!

Lucky Media
Lunar: Laravel E-Commerce logo

Lunar: Laravel E-Commerce

E-Commerce for Laravel. An open-source package that brings the power of modern headless e-commerce functionality to Laravel.

Lunar: Laravel E-Commerce
SaaSykit: Laravel SaaS Starter Kit logo

SaaSykit: Laravel SaaS Starter Kit

SaaSykit is a Multi-tenant Laravel SaaS Starter Kit that comes with all features required to run a modern SaaS. Payments, Beautiful Checkout, Admin Panel, User dashboard, Auth, Ready Components, Stats, Blog, Docs and more.

SaaSykit: Laravel SaaS Starter Kit

The latest

View all →
JSON:API Resource in Laravel 12.45 image

JSON:API Resource in Laravel 12.45

Read article
Caching With MongoDB for Faster Laravel Apps image

Caching With MongoDB for Faster Laravel Apps

Read article
Laravel 12.44 Adds HTTP Client afterResponse() Callbacks image

Laravel 12.44 Adds HTTP Client afterResponse() Callbacks

Read article
Handle Nested Data Structures in PHP with the Data Block Package image

Handle Nested Data Structures in PHP with the Data Block Package

Read article
Detect and Clean Up Unchanged Vendor Files with Laravel Vendor Cleanup image

Detect and Clean Up Unchanged Vendor Files with Laravel Vendor Cleanup

Read article
Seamless PropelAuth Integration in Laravel with Earhart image

Seamless PropelAuth Integration in Laravel with Earhart

Read article