Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/pacerit/laravel-core

Package of core classes that provide standardization in usage of Services, Repositories etc. for Laravel framework.
https://github.com/pacerit/laravel-core

entity formatter laravel laravel-framework php repository services trait

Last synced: 4 months ago
JSON representation

Package of core classes that provide standardization in usage of Services, Repositories etc. for Laravel framework.

Awesome Lists containing this project

README

        

# Laravel Core
![GitHub tag (latest by date)](https://img.shields.io/github/tag-date/pacerit/laravel-core?label=Version)
![GitHub](https://img.shields.io/github/license/pacerit/laravel-core?label=License)
![Packagist](https://img.shields.io/packagist/dt/pacerit/laravel-core?label=Downloads)
![PHP from Packagist](https://img.shields.io/packagist/php-v/pacerit/laravel-core?label=PHP)
[![StyleCI](https://github.styleci.io/repos/199045038/shield?branch=master)](https://github.styleci.io/repos/199045038)
[![Build Status](https://travis-ci.com/pacerit/laravel-core.svg?branch=master)](https://travis-ci.com/pacerit/laravel-core)

Package of core classes that provide standardization in usage of Services, Repositories etc. for Laravel and
Lumen framework.

## Installation
You can install this package by composer:

composer require pacerit/laravel-core

### Version compatibility
#### Laravel
Framework | Package | Note
:---------|:--------|:------
5.8.x | ^1.x.x | No longer maintained.
6.0.x | ^2.x.x | Bug fixes only.
7.x.x | ^3.x.x | Bug fixes only.
8.x.x | ^4.x.x | PHP ^8.0 Supported from 4.0.4
9.x.x | ^5.x.x |

#### Lumen
Framework | Package | Note
:---------|:--------|:------
5.8.x | ^1.1.x | No longer maintained.
6.0.x | ^2.x.x | Bug fixes only.
7.x.x | ^3.x.x | Bug fixes only.
8.x.x | ^4.x.x | PHP ^8.0 Supported from 4.0.4
9.x.x | ^5.x.x |

## Classes included

### Basic classes
CoreEntity::class - base class models
CoreRepository::class - base class of repositories (provided by package)
CoreService::class - base class of services
CoreFormatter::class - base class of formatter
CoreException::class - base class of exceptions

### Traits
ScenatioTrait::class
StringTrait::class
NumberTrait::class
XMLTrait::class

## Implementation

### Entities implementation
In order to use Service, Repositories or Formatter classes, first you must prepare your entities. Every entity
in you application must:
- Extend CoreEntity class
- Implements interface that extend CoreEntityInterface

For example, this is implementation of Example entity:

Example class:
```php
class Example extends CoreEntity implements ExampleInterface
{}
```

ExampleInterface:
```php
interface ExampleInterface extends CoreEntityInterface
{}
```
Interface and entity class must be bind in you app ServiceProvider:
```php
/**
* Register any application services.
*
* @return void
*/
public function register()
{
$this->app->bind(ExampleInterface::class, Example::class);
}
```

#### Using UUID's in Entity
**To use UUID feature install suggested ramsey/uuid package**

This package provides possibility to use UUID's in Entities as secondary key, for external use (i.e. in routes).
It still requires to use integer type ID's as Primary keys in your database.

To use UUID's in your Entity, it must:
- Implements UUIDInterface
- Use UsesUUID trait

In your migration create field with "uuid" key using uuid() method.
```php
$table->uuid(UsesUUID::UUID);
```
UUID's will bu automatically generated when new entity is created.
If you use Services, you can add WithUUID trait to your implementation of CoreService class to add methods that
are helpful when using UUID's.
```php
class MyEntityService extends CoreService implements MyEntityServiceInterface
use WithUUID;
(...)
```

### Repositories implementation and usage

Documentation of Repository can be found [here.](https://github.com/pacerit/laravel-repository)

### Services implementation
To use Service, create service class that:
- Extend CoreService class
- Implements interface that extend CoreServiceInterface

For example, this is implementation of service for Example entity:

ExampleServiceInterface:
```php
interface ExampleServiceInterface extends CoreRepositoryInterface
{}
```
ExampleService class. In __construct() function of class, provide class of repository for entity, and set it by
setRepository() function. Optionally you can pass formatter for entity and set if by setFormattter() function
to provide formatting functionality.
Provided classes must be implementation of CoreRepository/CoreFormatter.
```php
class ExampleService extends CoreService implements ExampleServiceInterface
{
/**
* ExampleService constructor.
*
* @param ExampleRepositoryInterface $exampleRepository
* @param ExampleFormatterInterface $exampleFormatter
*/
public function __construct(
ExampleRepositoryInterface $exampleRepository,
ExampleFormatterInterface $exampleFormatter
) {
$this->setRepository($exampleRepository)
->setFormatter($exampleFormatter);
}

}
```
#### Using services
To use Service in controller or other class you can use dependency injection or Container. Below is sample code of
using service in controller.
```php
class ExampleController extends Controller {

/**
* @var ExampleServiceInterface $exampleService
*/
protected $exampleService;

public function __construct(ExampleServiceInterface $exampleService){
$this->exampleService = $exampleService;
}

....
}
```
#### Available methods
* setRepository() - set repository to use in service. Passing object must be implementation of CoreRepositoryInterface
* getRepository() - return previously set repository class instance
* setFormatter() - set formatter class to use in service. Passing object must be implementation of
CoreFormatterInterface
* getFormatter() - return previously set formatter class instance
* setModel() - set model you want to work on i.e. if you already have instance
* setModelByID() - set model by given id (integer). If model with given ID exist, it will be set, otherwise exception
will by thrown
* setModelByKey() - set model by given key (column in table) and given value. Helpful when looking for record by column
other than ID (i.e foreign key)
* setNewMode() - set new model class instance
* getModel() - get previously set model class instance
* format() - get previously set model class instance and pass it to formatter class. CoreFormatterInterface instance
will be return. Example of use.
```php
// Return Example entity record with ID 1 as an array.
return $this->exampleService->setModelByID(1)->format()->toArray();
```
* create() - create new entity record based on actually set model, of given parameters. Example:
```php
For this example, we assume that Example entity class have setFirstValue() and setSecondValue() functions (setters)
and const. Both of this example, create the same records in database.

// Create based on previously set entity.
$this->exampleService
->setNewModel()
->getModel()
->setFirstValue(1)
->setSecondValue(2);

$this->exampleService->create();

// Create model based on parameters.
$this-exampleService->create(
[
ExampleInterface::FIRST_VALUE => 1,
ExampleInterface::SECOND_VALUE => 2,
]
);

```
* update() - update model. Like in create method it base on previously set model or parameters, but we must set existing
model record before. Otherwise exception will be thrown. Example:
```php
// Update based on previously set entity.
$this->exampleService
->setModelByID(1)
->getModel()
->setFirstValue(2);

$this->exampleService->update();

// Update model based on parameters.
$this-exampleService->update(
1,
[
ExampleInterface::FIRST_VALUE => 2,
]
);
```
* delete() - delete model. Like in update() method, this also required to set existing model before.
```php
$this->exampleService->setModelByID(1)->delete();
```
## Traits
### ScenarioTrait
ScenarioTrait contains very useful functions, if we want use different class, based by some key (i.e. "type" field
in entity).

For example - we have abstract "PaymentService" class with not implemented "create()" method.
We create two classes that extend this class and implement create() function - each class for different
payment provider:

* DummyProviderPaymentService::class - for "DummyProvider"
* OtherDummyProviderPaymentService::class - for "OtherDummyProvider"

each of them implements the create() function a little differently.

Normally if we want call create() function in matching class we will be do it like this:
```php
(...)
switch ($type) {
case 'DummyProvider':
$service = new DummyProviderPaymentService::class;
break;

case 'OtherDummyProvider':
$service = new OtherDummyProviderService::class;
break;

default:
// Dunno what to do..
}

$service->create();
(...)
```

But if this class have dependency injection, that is not possible. And is not very elegant.

Here comes help from ScenarioTrait. With this class, we can solve this problem,
for example, in this way:
```php
(...)
$this->registerScenario('DummyProvider', DummyProviderPaymentService::class);
$this->registerScenatio('OtherDummyProvider', OtherDummyProviderService::class);

$this->getScenatioInstance($type)->create();
(...)
```
That's it. And yes - we can pass only namespace of class (even with dependency injection) - new
instance will be created while call getScenarioInstance() method. Of course you can pass exist
instance of class (.i.e. from dependency injection) - it's up to you.

## Changelog

Go to the [Changelog](CHANGELOG.md) for a full change history of the package.

## Testing

composer test

## Security Vulnerabilities

If you discover a security vulnerability within package, please send an e-mail to Wiktor Pacer
via [[email protected]](mailto:[email protected]). All security vulnerabilities will be promptly addressed.

## License

This package is open-source software licensed under the [MIT license](https://opensource.org/licenses/MIT).