Subresources
Keep on Learning!
If you liked what you've learned so far, dive in! Subscribe to get access to this tutorial plus video, code and script downloads.
With a Subscription, click any sentence in the script to jump to that part of the video!
Login SubscribeWe have two different ways to get the dragon treasures for a user. First, we could fetch the User
and read its dragonTreasures
property. The second is via the filter that we added a moment ago. In the API, that looks like owner=/api/users/4
on the GET
collection operation for treasures.
This is my go-to way of getting the data... because if I want to fetch treasures, it make sense to use a treasures
endpoint. Besides, if a user owns a lot of treasures, that'll give us pagination!
But you may sometimes choose to add a special way to fetch a resource or collection of resources... almost like a vanity URL. For example, imagine that, to get this same collection, we want the user to be able to go to /api/users/4/treasures.jsonld
. That, of course, doesn't work. But it can be done. This is called a subresource, and subresources are much nicer in API platform 3.
Adding a Subresource via Another ApiResource
Okay, let's think. This endpoint will return treasures. So to add this subresource, we need to update the DragonTreasure
class.
How? By adding a second ApiResource
attribute. We already have this main one, so now add a new one. But this time, control the URL with a uriTemplate
option set to exactly what we want: /users/{user_id}
for the wildcard part (we'll see how that's used in a moment) then /treasures
.
That's it! Well... also add .{_format}
. This is optional, but that's the magic that lets us "cheat" and add this .jsonld
to the end of the URL.
// ... lines 1 - 54 | |
( | |
uriTemplate: '/users/{user_id}/treasures.{_format}', | |
// ... line 57 | |
) | |
// ... lines 59 - 62 | |
class DragonTreasure | |
{ | |
// ... lines 65 - 222 | |
} |
Next, add operations
... because we don't need all six... we really need just one. So, say [new GetCollection()]
because we will return a collection of treasures.
// ... lines 1 - 54 | |
( | |
uriTemplate: '/users/{user_id}/treasures.{_format}', | |
operations: [new GetCollection()], | |
) | |
// ... lines 59 - 62 | |
class DragonTreasure | |
{ | |
// ... lines 65 - 222 | |
} |
Ok, let's see what this did! Head back to the documentation and refresh. Suddenly we have... three resources and this one has the correct URL!
Oh, and we have three resources because, if you recall, we customized the shortName
. Copy that and paste it onto the new ApiResource
so they match. And to make PhpStorm happy, I'll put them in order.
// ... lines 1 - 54 | |
( | |
uriTemplate: '/users/{user_id}/treasures.{_format}', | |
shortName: 'Treasure', | |
operations: [new GetCollection()], | |
) | |
// ... lines 60 - 63 | |
class DragonTreasure | |
{ | |
// ... lines 66 - 223 | |
} |
Now when we refresh... perfect! That's what we want!
Understanding uriVariables
We now have a new operation for fetching treasures. But does it work? It says that it will retrieve a collection of treasure resources, so that's good. But... we have a problem. It thinks that we need to pass the id
of a DragonTreasure
... but it should be the id of a User
! And even if we pass something, like 4
... and hit "Execute"... look at the URL! It didn't even use the 4
: it still has {user_id}
in the URL! So of course it comes back with a 404 error.
The problem is that we need to help API Platform understand what {user_id}
means. We need to tell it that this is the id of the user and that it should use that to query WHERE owner_id
equals the value.
To do that, add a new option called uriVariables
. This is where we describe any "wildcards" in your URL. Pass user_id
set to a new Link()
object. There are multiple... we want the one from ApiPlatform\Metadata
.
// ... lines 1 - 11 | |
use ApiPlatform\Metadata\Link; | |
// ... lines 13 - 55 | |
( | |
uriTemplate: '/users/{user_id}/treasures.{_format}', | |
shortName: 'Treasure', | |
operations: [new GetCollection()], | |
uriVariables: [ | |
'user_id' => new Link( | |
// ... lines 62 - 63 | |
), | |
], | |
) | |
// ... lines 67 - 70 | |
class DragonTreasure | |
{ | |
// ... lines 73 - 230 | |
} |
This object needs two things. First, point to the class that the {user_id}
is referring to. Do that by passing a fromClass
option set to User::class
.
// ... lines 1 - 11 | |
use ApiPlatform\Metadata\Link; | |
// ... lines 13 - 55 | |
( | |
uriTemplate: '/users/{user_id}/treasures.{_format}', | |
shortName: 'Treasure', | |
operations: [new GetCollection()], | |
uriVariables: [ | |
'user_id' => new Link( | |
// ... line 62 | |
fromClass: User::class, | |
), | |
], | |
) | |
// ... lines 67 - 70 | |
class DragonTreasure | |
{ | |
// ... lines 73 - 230 | |
} |
Second, we need to define which property on User
points to DragonTreasure
so that it can figure out how to structure the query. To do this, set fromProperty
to treasures
. So, inside User
, we're saying that this property describes the relationship. Oh, but I totally messed that up: the property is dragonTreasures
.
// ... lines 1 - 11 | |
use ApiPlatform\Metadata\Link; | |
// ... lines 13 - 55 | |
( | |
uriTemplate: '/users/{user_id}/treasures.{_format}', | |
shortName: 'Treasure', | |
operations: [new GetCollection()], | |
uriVariables: [ | |
'user_id' => new Link( | |
fromProperty: 'dragonTreasures', | |
fromClass: User::class, | |
), | |
], | |
) | |
// ... lines 67 - 70 | |
class DragonTreasure | |
{ | |
// ... lines 73 - 230 | |
} |
Ok, cruise back over and refresh. Under the endpoint... yea! It says "User identifier". Let's put 4
in there again, hit "Execute" and... got it. There are the five treasures for this user!
And in the other browser tab... if we refresh... it works!
How the Query is Made
Behind the scenes, thanks to the Link
, API Platform basically makes the following query:
SELECT * FROM dragon_treasure WHERE owner_id =
whatever we pass for {user_id}
. It knows how to make that query by looking at the Doctrine relationship and figuring out which column to use. It's super smart.
We can actually see this in the profiler. Go to /_profiler
, click on our request... and, down here, we see 2 queries... which are basically the same: the 2nd is used for the "total items" for pagination.
If you click "View formatted query" on the main query... it's even more complex than I expected! It has an INNER JOIN
... but it's basically selecting all the dragon treasures data where owner_id
= the ID of that user.
What about toProperty?
By the way, if you look at the documentation, there's also a way to set all of this up via the other side of the relationship: by saying toProperty: 'owner'
.
This still works... and works exactly the same. But I recommend sticking with fromProperty
, which is consistent and, I think, more clear. The toProperty
is needed only if you didn't map the inverse side of a relationship... like if there was no dragonTreasures
property on User
. Unless you have that situation, stick with fromProperty
.
Don't Forget normalizationContext!
This is all working nicely except for one small problem. If you look back at the data, it shows the wrong fields! It's returning everything, like id
and isPublished
.
Those aren't supposed to be included thanks of our normalization groups. But since we haven't specified any normalization groups on the new ApiResource
, the serializer returns everything.
To fix this, copy the normalizationContext
and paste it down here. We don't need to worry about denormalizationContext
because we don't have any operations that do any denormalizing.
// ... lines 1 - 11 | |
use ApiPlatform\Metadata\Link; | |
// ... lines 13 - 55 | |
( | |
uriTemplate: '/users/{user_id}/treasures.{_format}', | |
shortName: 'Treasure', | |
operations: [new GetCollection()], | |
uriVariables: [ | |
'user_id' => new Link( | |
fromProperty: 'dragonTreasures', | |
fromClass: User::class, | |
), | |
], | |
normalizationContext: [ | |
'groups' => ['treasure:read'], | |
], | |
) | |
// ... lines 70 - 73 | |
class DragonTreasure | |
{ | |
// ... lines 76 - 233 | |
} |
If we refresh now... got it!
A Single Subresource Endpoint
Let's add one more subresource to see a slightly different case. I'll show you the URL I want first. We have a treasure with ID 11
. This means we can go to /api/treasures/11.jsonld
to see that. Now I want to be able to add /owner
to the end to get the user that owns this treasure. Right now, that doesn't work.... so let's get to work!
Because the resource that will be returned is a User
, that's the class that needs the new API Resource.
Above it, add #[ApiResource()]
with uriTemplate
set to /treasures/{treasure_id}
for the wildcard (though this can be called anything), followed by /owner.{_format}
.
// ... lines 1 - 24 | |
( | |
uriTemplate: '/treasures/{treasure_id}/owner.{_format}', | |
// ... lines 27 - 34 | |
) | |
// ... lines 36 - 38 | |
class User implements UserInterface, PasswordAuthenticatedUserInterface | |
{ | |
// ... lines 41 - 187 | |
} |
Next pass uriVariables
with treasure_id
set to a new Link()
- the one from ApiPlatform\Metadata
. Inside, set fromClass
to DragonTreasure::class
. And since the property inside DragonTreasure
that refers to this relationship is owner
, add fromProperty: 'owner'
.
// ... lines 1 - 7 | |
use ApiPlatform\Metadata\Link; | |
// ... lines 9 - 24 | |
( | |
uriTemplate: '/treasures/{treasure_id}/owner.{_format}', | |
// ... line 27 | |
uriVariables: [ | |
'treasure_id' => new Link( | |
fromProperty: 'owner', | |
fromClass: DragonTreasure::class, | |
), | |
], | |
// ... line 34 | |
) | |
// ... lines 36 - 38 | |
class User implements UserInterface, PasswordAuthenticatedUserInterface | |
{ | |
// ... lines 41 - 187 | |
} |
We also know that we're going to need the normalizationContext
... so copy that... and paste it here. Finally, we only want one operation: a GET
operation to return a single User
. So, add operations
set to [new Get()]
.
// ... lines 1 - 6 | |
use ApiPlatform\Metadata\Get; | |
use ApiPlatform\Metadata\Link; | |
// ... lines 9 - 24 | |
( | |
uriTemplate: '/treasures/{treasure_id}/owner.{_format}', | |
operations: [new Get()], | |
uriVariables: [ | |
'treasure_id' => new Link( | |
fromProperty: 'owner', | |
fromClass: DragonTreasure::class, | |
), | |
], | |
normalizationContext: ['groups' => ['user:read']], | |
) | |
// ... lines 36 - 38 | |
class User implements UserInterface, PasswordAuthenticatedUserInterface | |
{ | |
// ... lines 41 - 187 | |
} |
That should do it! Move back over to the documentation, refresh, and take a look under "User". Yep! We have a new operation! And it even sees that the wildcard is a "DragonTreasure identifier".
If we go refresh the other tab... it works!
Ok team, I lied about this being the last topic because... it's bonus topic time! Next: let's create a React-based admin area automatically from our API docs. Woh.
Just a nitpick at Api platform I guess (Using Symfony 6.2 from the project code but api platform 3.2 as that is what is installed with 'composer require api'):
The subresource works, I can enter '4' in the docs for the user_id field and I get all treasures that have /api/users/4 as owner...
... but it is still called 'Treasure identifier' in my docs (even after a cache clear), while it's called 'User identifier' in your tutorial?