Implementation of Repository pattern for Laravel (on top of Eloquent)
Install the saritasa/laravel-repositories
package:
$ composer require saritasa/laravel-repositories
- Publish configuration file:
php artisan vendor:publish --tag=laravel_repositories
- Register custom repositories implementation:
return [
'bindings' => [
\App\Models\User::class => \App\Repositories\UserRepository::class,
],
];
Note: Your custom repository must implement IRepository
contract.
To get specific repository in your code you can just build with DI container repositories factory and then
build needed for your repository in this factory.
Example:
$repositoryFactory = app(\Saritasa\LaravelRepositories\Repositories\IRepositoryFactory::class);
$userRepository = $repositoryFactory->getRepository(\App\Models\User::class);
Methods findWhere/getWhere/getWith/getPage/getCursorPage/count/ can receive criteria as params Here the examples of available syntax:
- Criterion without operator:
$criteria = [
'field1' => 'value1',
'field2' => 1,
];
In this case =
operator and and
boolean between them will be used.
Example: ... 'field1 = 'value1' and 'field2' = 1 ...
- Criterion with operator:
$criteria = [
['field1', '<>', 'value1'],
['field2', '>', 1, 'or'],
['field3', 'in', [1, 2]],
['field4', 'not in', new \Illuminate\Support\Collection([1, 2])],
];
Important:
arrays and collection can be used only with in
and not in
operators.
Note: As 4th parameter you can pass boolean or
/and
(and
uses by default).
But you should remember that boolean used between current and previous criterion
Example: ... 'field1 <> 'value1' or 'field2' > 1 and 'field3' in (1, 2) and 'field4' not in (1, 2) ...
- Criterion as DTO:
$criteria = [
new Criterion([
Criterion::OPERATOR => '<>',
Criterion::VALUE => 'value1',
Criterion::ATTRIBUTE => 'field1',
]),
new Criterion([
Criterion::OPERATOR => '>',
Criterion::VALUE => 1,
Criterion::ATTRIBUTE => 'field2',
Criterion::BOOLEAN => 'or',
]),
new Criterion([
Criterion::OPERATOR => 'in',
Criterion::VALUE => [1, 2],
Criterion::ATTRIBUTE => 'field3',
]),
new Criterion([
Criterion::OPERATOR => 'not in',
Criterion::VALUE => [1, 2],
Criterion::ATTRIBUTE => 'field4',
]),
];
Result will be the same as in previous example.
- Nested criteria:
You can group different conditions that gives flexibility in getting data from repository
$criteria = [
[
['field1', '<>', 'value1'],
['field2', '>', 1, 'or'],
],
[
['field3', 'in', [1, 2]],
['field4', 'not in', [1, 2],
'boolean' => 'or',
],
];
Note: you can add nesting level in any depth what you want. To use or
condition between one group
and other(group and non-group condition) you can pass 'boolean' parameter in the same level as other conditions.
Example:... ('field1 <> 'value1' or 'field2' > 1) or ('field3' in (1, 2) and 'field4' not in (1, 2)) ...
- Relation existence criterion:
You can build queries to check on existence any model relations
$criteria = [
new RelationCriterion('roles', [['slug', 'in', [1, 2]]], 'or'),
];
Method getWith() method allows to retrieve list of entities with
eager loaded related models and related models counts. Also allows to filter this list by given criteria
and sort in requested order.
Example:
$usersRepository->getWith(
['role', 'supervisors'],
['phones'],
[],
new Saritasa\LaravelRepositories\DTO\SortOptions('name', 'DESC')
);
- Each user will be retrieved with pre-loaded role and supervisors models.
- Each user will be retrieved with pre-loaded phones relation count.
- List of users will be ordered by requested sort options.
Base exception for repository layer.
Throws when can not register custom repository.
Throws in case when some model not exists in storage.
Throws when provided criteria has incorrect format at least in one criterion inside.
- Create fork, checkout it
- Develop locally as usual. Code must follow PSR-1, PSR-2 -
run PHP_CodeSniffer to ensure, that code follows style guides - Cover added functionality with unit tests and run PHPUnit to make sure, that all tests pass
- Update README.md to describe new or changed functionality
- Add changes description to CHANGES.md file. Use Semantic Versioning convention to determine next version number.
- When ready, create pull request
If you have GNU Make installed, you can use following shortcuts:
make cs
(instead ofphp vendor/bin/phpcs
) -
run static code analysis with PHP_CodeSniffer
to check code stylemake csfix
(instead ofphp vendor/bin/phpcbf
) -
fix code style violations with PHP_CodeSniffer
automatically, where possible (ex. PSR-2 code formatting violations)make test
(instead ofphp vendor/bin/phpunit
) -
run tests with PHPUnitmake install
- instead ofcomposer install
make all
or justmake
without parameters -
invokes described above install, cs, test tasks sequentially -
project will be assembled, checked with linter and tested with one single command