camilord / laravel-api-controller
A laravel api base controller with basic CRUD mapped to the model. Fork from https://github.com/phpsa/laravel-api-controller
Installs: 2 437
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 1
Forks: 0
Open Issues: 0
Requires
- php: >=8.0|>=8.1|>=8.2|>=8.3
- laravel/framework: >=8.0|>=9.0|>=10.0
Requires (Dev)
- orchestra/testbench: >=8.0|>=9.0
- phpunit/phpunit: >=9.0
README
This repository is required from one of my work projects I maintained and unfortunately the maintainer remove the hasRepository
and BaseRepository
after version 2.0.0.
Laravel Api Controller
Basic CRUD API Methods that can be extended for your models by default has a list, show, update, add and delete endpoint to interact with your model.
Installation
Install via composer
composer require phpsa/laravel-api-controller
Publish Configuration File (optional - if you need to change any of the default configurations)
php artisan vendor:publish --provider="Phpsa\LaravelApiController\ServiceProvider" --tag="config"
Usage
Generate a new Api Controller, Repository and Route via php artisan make:api {ModelName}
This will create a Api/ModelNameController for you and you will have the basic routes in place as follows:
- GET
api/v1/{model_name}
- list all/paged/filtered (class::index) - GET
api/v1/{model_name}/$id
- Show a specified id (class::show) - POST
api/v1/{model_name}
- Insert a new record (class::store) - PUT
api/v1/{model_name}/$id
- Update an existing record (class::update) - DELETE
api/v1/{model_name}/$id
- Delete an existing record (class::destroy)
You can override the methods by simply putting in your own methods to override - method names in braces above
Events
- POST (class::store) - triggers a new
Phpsa\LaravelApiController\Events\Created
Event which has the new record available as$record
- PUT (class::update) - triggers a new
Phpsa\LaravelApiController\Events\Updated
Event which has the updated record available as$record
- DELETE (class::destry) - triggers a new
Phpsa\LaravelApiController\Events\Deleted
Event which has the deleted record available as$record
Policies
Policies: https://laravel.com/docs/6.x/authorization#generating-policies
Generate with php artisan make:policy PostPolicy --model=Post
- Get list - calls the
viewAny
policy - Get single - calls the
view
policy - Post New - calls the
create
policy - Put Update - calls the
update
policy - Delete item - calls the
delete
policy
Query/Data modifiers in policies for the api endpoints
qualifyCollectionQueryWithUser($user, $repository)
-> return void - add any queries to the repository (ie ->where('x','))qualifyItemQueryWithUser($user, $repository)
-> return void - add any queries to the repository (ie ->where('x','))qualifyStoreDataWithUser($data)
- return the updated data arrayqualifyUpdateDataWithUser($data)
- return the updated data array
Resources / Collections (Transforming)
Resources: https://laravel.com/docs/6.x/eloquent-resources
Generate with
php artisan make:apiresource UserResource
and php artisan make:resource UserCollection
Change the Resource to extend from:
use Phpsa\LaravelApiController\Http\Resources\ApiResource
for your resource
use Phpsa\LaravelApiController\Http\Resources\ApiCollection
for your resource collection
in your controller override the following params:
protected $resourceSingle = UserResource::class; protected $resourceCollection = UserCollection::class;
Snake vs Camel
- middleware to convert all camel to snake:
Phpsa\LaravelApiController\Http\Middleware\SnakeCaseInputs
- set request header
X-Accept-Case-Type
to eithersnake
orcamel
to alter your data response
Filtering
For the get command you can filter by using the following url patterns
In / Not In
You can pass to the filters an array of values
ie: filter[user_id]=1||2||||4||7
or filter[user_id!]=55||33
Null / Not Null (introduced 1.23.0)
If you need to filter on whether a field is null or not null, you can use the filter param as of version 1.23.0 EG: filter[age]=NULL
or filter[age!]=NULL
. Note that NULL must be uppercase.
Older versions Add a scope to your model: eg
public function scopeAgeNull(Builder $builder, $isNull = true){ $isNull ? $builder->whereNull('age') : $builder->whereNotNull('age'); }
Add to your allowedScopes and can then be called in url as ?ageNull=1
for where null and ?ageNull=0
for where age not null
Scopes
In addition to filtering, you can use Laravel's Eloquent Query Scopes to do more complex searches or filters.
Simply add an $allowedScopes
to your ApiResource
, and that scope will be exposed as a query parameter.
Assuming you have a scopeFullname
defined on your Eloquent Model, you can expose this scope to your API as follows:
protected static $allowedScopes = [ 'fullname' ];
Given the above $allowedScopes
array, your API consumers will now be able to request ?fullname=John
. The query parameter value will be passed to your scope function in your Eloquent Model.
Filtering on related models
You can easily filter using any related model that is configured for include
. Simply specify ?filter[model.field]=123
in your query string. The same filter options above apply to related fields.
Fields, Relationships, Sorting & Pagination
Fields
By default all fields are returned, you can limit that to specific fields in the following ways:
- Api Controller parameter
$defaultFields
default asprotected $defaultFields = ['*'];
- switch to include an array of fields - fields param in url querystring: ie
fields=id,name,age
= will only return those, this will also override the above. - in your response resource you can set the static::allowedFields to lock down which fields are returnable
addfields
andremovefields
params in url querystring will work with these.- Use laravel eloquent model
$appends
property to automatically include custom attribute accessors.
Relationships
- Using the relationships defined in your models, you can pass a comma delimited list eg
include=join1,join2
which will return those joins (one or many).
Simply add a protected static $mapResources
to your Resource
to define which resources to assign your related data. E.e., for a one to many relationship, you should specify a collection, and a one-to-one relationship specify the related resource directly. This will allow the API to properly format the related record.
protected static $mapResources = [
'notes' => NotesCollection::class,
'owner' => OwnerResource::class
];
- You can automatically update and create related records for most types of relationships. Just include the related resource name in your POST or PUT request.
For BelongsToMany
or MorphToMany
relationships, you can choose the sync strategy. By default, this will take an additive strategy. That is to say, related records sent will be ADDED to any existing related records. On a request-by-request basis, you can opt for a sync strategy which will remove the pivot for any related records not listed in the request. Note the actual related record will not be removed, just the pivot entry.
To opt for the sync behavaiour, set ?sync[field]=true
in your request.
Sorting
- Sorts can be passed as comma list aswell, ie
sort=age asc
orsort=age asc,name desc,eyes
- generates sql ofsort age asc
andsort age asc, name desc, eyes asc
respectively - Default sort can also be added on the controller using by overrideing the
protected $defaultSort = null;
parameter
Pagination
- pagination can be enabled/disbled on the controller by overriding the
protected $defaultLimit = 25;
on the controller - pagination can also be passed via the url using
limit=xx&page=y
- pagination can also be limited to a max per page by overriding the
protected $maximumLimit = false;
parameter
Validation
- When Posting a new record, validation can be done by adding a
rulesForCreate
method to your controller returning an array eg
[ 'email' => 'required|email', 'games' => 'required|numeric', ]
see https://laravel.com/docs/5.8/validation#conditionally-adding-rules
- for updating a record, add a method
rulesForUpdate
per above.
Defaults
The following parameters are set in the Base Api controller and can be overwritten by your Controller on a case by case basis:
-
DEPRECATED
protected $resourceKeySingular = 'data';
-
DEPRECATED
protected $resourceKeyPlural = 'data';
-
protected $resourceSingle = JsonResource::class;
Collection to use for your single resource -
protected $resourceCollection = ResourceCollection::class;
Collection to use for your resource collection -
protected $defaultFields = ['*'];
Default Fields to respond with -
protected $defaultSort = null;
Set the default sorting for queries. -
protected $defaultLimit = 25;
Number of items displayed at once if not specified. (0 = maximumLimit) -
protected $maximumLimit = 0;
Maximum limit that can be set via $_GET['limit']. - this ties in with the defaultLimit aswell, and if wanting to disable pagination , both should be 0. ) will allow all records to be returned in a single call. -
protected $unguard = false;
Do we need to unguard the model before create/update?
Scopes
SoftDeleted Records
add the Phpsa\LaravelApiController\Model\Scopes\WithSoftDeletes
trait to your model,
add to your resource file:
class MyModelResource extends ApiResource { protected static $allowedScopes = [ 'withTrashed', 'onlyTrashed' ];
you can now append withTrashed=1
or onlyTrashed=1
to your query.
Responses
you can override responses for each point by overriding the following protected methods:
- handleIndexResponse
- handleStoreResponse
- handleShowResponse
- handleUpdateResponse
- handleDestroyResponse
Security
If you discover any security related issues, please email instead of using the issue tracker.