Ресурсы 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 но я пытаюсь найти лучший способ добиться того же результата.

Author: Jeff Puckett, 2017-09-24

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,
                ];
            }),
        ];
    }
}
 13
Author: Jeff Puckett, 2017-09-25 20:01:03

Принятый ответ работает, если вы не заинтересованы в использовании ссылок и метаданных. Если вы хотите, просто верните:

return new PageResourceCollection(Page::paginate(10));

В вашем контроллере. Вам также следует позаботиться о том, чтобы загрузить другие зависимые отношения, прежде чем переходить к сбору ресурсов.

 0
Author: Wale, 2018-01-29 14:38:40

Если вы хотите, чтобы поля ответа имели одинаковое значение в Ресурсе и Коллекции, вы можете повторно использовать Ресурс внутри Коллекции

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);
    }
}
 0
Author: cyberfly, 2018-07-11 13:57:41