nycu-csit / laravel-restful-utils

This package provides utilities for building a restful API for Laravel projects.
MIT License
1 stars 0 forks source link

Laravel Restful Utils

This package provides utilities for building a restful API for Laravel projects.

This package requires PHP >= 8.0, Laravel ^8.0, ^9.0, ^10.0 and ^11.0.

Setup

In order to enable error wrapping feature, you should update your app/Exceptions/Handler.php. Make Handler extends NycuCsit\LaravelRestfulUtils\Exceptions\Handler instead of original one.

And we suggest you add HttpApiException into the $dontReport list:

use NycuCsit\LaravelRestfulUtils\Exceptions\HttpApiException;

protected $dontReport = [
    HttpApiException::class,
];

Exceptions

We suggest throwing exceptions instead of returning a response to indicate the status.

These exceptions help you to give a well-wrapped error response:

Error Wrapping

When enabled, error responses are represented like this:

{
    "error": {
        "code": "CONFLICT_STATUS",
        "message": "You cannot update this resource because it was locked."
    }
}

The code should be a string to simply indicate the type of error. The client may use it as error identifier.

The message should be a human-readable string to show the reason or what happened. You may provide localized message.

API Controller

This package provides controllers for you to reduce many duplicated logics. Publish these controllers' base classes to your app/Http/Controller:

php artisan vendor:publish --tag=restful-controllers

Now, you can use ApiResourceController and ApiNestedResourceController in app/Http/Controller in your application. These controllers are designed to integrate with Laravel's built-in features:

So, we suggested you create a resource controller like this:

php artisan make:restful-controller UserController --model=User --requests

or for nested resource:

php artisan make:restful-controller PhotoController --model=Photo --parent=User --requests

Now, you may implement method stubs in your controller.

There are some properties can help you reduce passing arguments:

These properties are defined in Context trait.

Be careful!

Because of the route implicit binding, the action methods (show(), store(), update(), destroy(), index() if nested resource is used) from the API Controllers cannot get bindings, so you need write these action methods and call the parent version:

class PhotoController extends ApiResourceController
{

    public function index(Request $request)
    {
        parent::indexAction($request);
    }

    public function store(StorePhotoRequest $request)
    {
        parent::storeAction($request);
    }

    public function show(Request $request, Photo $photo)
    {
        parent::showAction($request, $photo);
    }

    public function update(UpdatePhotoRequest $request, Photo $photo)
    {
        parent::updateAction($request, $photo);
    }

    public function destroy(Request $request, Photo $photo)
    {
        parent::destroyAction($request, $photo)
    }

    // ...

If you use the make:restful-controller command, these methods are generated for you.

There are many methods in the box, you can find each action method in Concerns folder, extend and override these methods to build something great!

Nested resource:

API Resource

For action method index() returns a collection of resources, action methods show(), store(), update() returns a single resource. You may need a transformation layer that sits between your Eloquent models and the JSON responses that are actually returned to your application's users. So, these methods may return API Resources instead of the model.

Class HasResourceActions and HasNestedResourceActions provides function constructJsonResource() and constructResourceCollection() to construct resource instance that just wraps the model.

To use your own resource class, you may use the make:resource artisan command:

php artisan make:resource PhotoResource

and override these methods in your controller:

use App\Http\Resources\PhotoResource;

class PhotoController extends ApiResourceController
{
    // ...

    public function constructJsonResource($resource): JsonResource
    {
        return new PhotoResource($resource);
    }

    public function constructResourceCollection($resource): ResourceCollection
    {
        return new PhotoResource::collection($resource);
    }
}

In most case, simply create a PhotoResource and use its ::collection() method to construct a resource collection.

In addition to generating resources that transform individual models, you may generate resources that are responsible for transforming collections of models.

To use your own collection class, you may use the artisan command to generate a collection:

php artisan make:resource User --collection

and override these methods in your controller:

use App\Http\Resources\PhotoResource;
use App\Http\Resources\PhotoCollection;

class PhotoController extends ApiResourceController
{
    // ...

    public function constructJsonResource($resource): JsonResource
    {
        return new PhotoResource($resource);
    }

    public function constructResourceCollection($resource): ResourceCollection
    {
        return new PhotoCollection($resource);
    }
}

Pagination

See: Pagination

You can add these properties in the controller to override the default value for pagination:

    protected bool $alwaysPaginate = true;
    protected bool $enablePaginate = true;
    protected int $defaultLimit = 20;
    protected int $maxLimit = 100;

When the pagination enabled, query params page and limit may be used in the URL.

Datetime store in local timezone

Laravel just stores the date in the model by the original hour/time of the date but without preserving the timezone information. You may use the custom cast LocalDatetime to transform date to local timezone (config('app.timezone')) date for setting model attribute.

Attach LocalDatetime to a model attribute:

    // import it
    use NycuCsit\LaravelRestfulUtils\Casts\LocalDatetime;

    // Attach it in the model class
    protected $casts = [
        'my_column' => LocalDatetime::class,
    ];

Once you attached LocalDatetime, you may assign an ISO 8601 string or a Carbon instance to my_column attribute.

If you need immutable instance or a date only (without time) attribute, here are parameters for you:

    protected $casts = [
        'my_column1' => LocalDatetime::class . LocalDatetime::DATE_ONLY,
        'my_column2' => LocalDatetime::class . LocalDatetime::IMMUTABLE,
        'my_column3' => LocalDatetime::class . LocalDatetime::DATE_ONLY_IMMUTABLE,
    ];