Ресурсы API Laravel 5.5 для коллекций (автономные данные)
Мне было интересно, можно ли определить разные данные для ресурса элемента и ресурса коллекции.
Для сбора я хочу отправить только ['id', 'title', 'slug']
, но ресурс элемента будет содержать дополнительные сведения ['id', 'title', 'slug', 'user', etc.]
Я хочу достичь чего-то вроде:
class PageResource extends Resource
{
/**
* Transform the resource into an array.
*
* @param \Illuminate\Http\Request
* @return array
*/
public function toArray($request)
{
return [
'id' => $this->id,
'title' => $this->title,
'slug' => $this->slug,
'user' => [
'id' => $this->user->id,
'name' => $this->user->name,
'email' => $this->user->email,
],
];
}
}
class PageResourceCollection extends ResourceCollection
{
/**
* Transform the resource collection into an array.
*
* @param \Illuminate\Http\Request
* @return array
*/
public function toArray($request)
{
return [
'id' => $this->id,
'title' => $this->title,
'slug' => $this->slug,
];
}
}
Pageresourcecollection не будет работать должным образом, потому что он использует pageresource, поэтому ему нужен
return [
'data' => $this->collection,
];
Я мог бы скопировать ресурс в PageFullResource
/ PageListResource
и PageFullResourceCollection
/ PageListResourceCollection
но я пытаюсь найти лучший способ добиться того же результата.
3 answers
В классе ресурсов есть метод сбора данных. Вы можете вернуть это в качестве входного параметра в свою коллекцию ресурсов, а затем указать свои преобразования в коллекции.
Контроллер:
class PageController extends Controller
{
public function index()
{
return new PageResourceCollection(PageResource::collection(Page::all()));
}
public function show(Page $page)
{
return new PageResource($page);
}
}
Ресурсы:
class PageResource extends Resource
{
public function toArray($request)
{
return [
'id' => $this->id,
'title' => $this->title,
'slug' => $this->slug,
'user' => [
'id' => $this->user->id,
'name' => $this->user->name,
'email' => $this->user->email,
],
];
}
}
class PageResourceCollection extends ResourceCollection
{
public function toArray($request)
{
return [
'data' => $this->collection->transform(function($page){
return [
'id' => $page->id,
'title' => $page->title,
'slug' => $page->slug,
];
}),
];
}
}
Принятый ответ работает, если вы не заинтересованы в использовании ссылок и метаданных. Если вы хотите, просто верните:
return new PageResourceCollection(Page::paginate(10));
В вашем контроллере. Вам также следует позаботиться о том, чтобы загрузить другие зависимые отношения, прежде чем переходить к сбору ресурсов.
Если вы хотите, чтобы поля ответа имели одинаковое значение в Ресурсе и Коллекции, вы можете повторно использовать Ресурс внутри Коллекции
PersonResource.php
<?php
namespace App\Http\Resources;
use Illuminate\Http\Resources\Json\Resource;
class PersonResource extends Resource
{
/**
* Transform the resource into an array.
*
* @param \Illuminate\Http\Request $request
* @return array
*/
public function toArray($request)
{
// return parent::toArray($request);
return [
'id' => $this->id,
'person_type' => $this->person_type,
'first_name' => $this->first_name,
'last_name' => $this->last_name,
'created_at' => (string) $this->created_at,
'updated_at' => (string) $this->updated_at,
];
}
}
PersonCollection.php
<?php
namespace App\Http\Resources;
use Illuminate\Http\Resources\Json\ResourceCollection;
class PersonCollection extends ResourceCollection
{
/**
* Transform the resource collection into an array.
*
* @param \Illuminate\Http\Request $request
* @return \Illuminate\Http\Resources\Json\AnonymousResourceCollection
*/
public function toArray($request)
{
// return parent::toArray($request);
return PersonResource::collection($this->collection);
}
}