Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/nilportugues/mongodb-repository

[PHP 7] MongoDB Repository implementation
https://github.com/nilportugues/mongodb-repository

composer crud filter mongodb mongodb-orm orm pageable paginate php php7 storage

Last synced: 3 months ago
JSON representation

[PHP 7] MongoDB Repository implementation

Awesome Lists containing this project

README

        

# MongoDB Repository
![PHP7 Tested](http://php-eye.com/badge/nilportugues/mongodb-repository/php70.svg)
[![Build Status](https://travis-ci.org/PHPRepository/mongodb-repository.svg)](https://travis-ci.org/PHPRepository/mongodb-repository) [![SensioLabsInsight](https://insight.sensiolabs.com/projects/9fc69e98-13b4-4ea5-a5fb-c394b42586e3/mini.png?gold)](https://insight.sensiolabs.com/projects/9fc69e98-13b4-4ea5-a5fb-c394b42586e3) [![Latest Stable Version](https://poser.pugx.org/nilportugues/mongodb-repository/v/stable)](https://packagist.org/packages/nilportugues/mongodb-repository) [![Total Downloads](https://poser.pugx.org/nilportugues/mongodb-repository/downloads)](https://packagist.org/packages/nilportugues/mongodb-repository) [![License](https://poser.pugx.org/nilportugues/mongodb-repository/license)](https://packagist.org/packages/nilportugues/mongodb-repository)
[![Donate](https://www.paypalobjects.com/en_US/i/btn/btn_donate_SM.gif)](https://paypal.me/nilportugues)

MongoDB Repository library aims to reduce the time spent writing repositories. MongoDB Repository using *[nilportugues/repository](https://github.com/nilportugues/php-repository)* as foundation using *[mongodb/mongodb](https://github.com/mongodb/mongo-php-library)*.

Motivation for this library was the boredom of writing MongoDB to do the same thing over and over again in multiple projects.

MongoDB Repository allows you to fetch, paginate and operate with data easily without adding overhead and following good practices.

Table of Contents
=================

* [Features](#features)
* [Installation](#installation)
* [Usage](#usage)
* [Mapping](#mapping)
* [Entity class](#entity-class)
* [Mapping class](#mapping-class)
* [Mapping the Repository](#mapping-the-repository)
* [Repository](#repository)
* [Methods](#methods)
* [Available in MongoDBRepository](#available-in-mongodbrepository)
* [Available in MongoDBWriteRepository](#available-in-mongodbwriterepository)
* [Available in MongoDBReadRepository](#available-in-mongodbreadrepository)
* [Available in MongoDBPageRepository](#available-in-mongodbpagerepository)
* [Data Operations](#data-operations)
* [Fields](#fields)
* [Filtering](#filtering)
* [Pagination](#pagination)
* [Pageable](#pageable)
* [Page object](#page-object)
* [Sorting](#sorting)
* [Ordering](#ordering)

## Features

- **Repository pattern right from the start.**
- **All operations available from the beginning:**
- Search the repository using PHP objects
- No need to write MongoDB for basic operations.
- Filtering is available using the Filter object.
- Fetching certaing fields is available using the Fields Object.
- Pagination is solved available using the Page and Pageable objects.
- **Custom operations can be written using DBAL.**
- **Want to change persistence layer? Provided repository alternatives are:**
- *[InMemoryRepository](https://github.com/PHPRepository/repository)*: for testing purposes
- *[FileRepository](https://github.com/PHPRepository/filesystem-repository)*: sites without DB access or for testing purposes.
- *[SQLRepository](https://github.com/PHPRepository/sql-repository)*: migration to SQL databases is possible.
- **Mapping written in PHP.**
- Supports custom Data Types (a.k.a Value Objects).
- Mapping deep data structures made easy using dot-notation.
- **Hydratation is optional.**
- Use hydratation when needed. Use `MongoDBRepositoryHydrator` trait to enable it.
- **Both custom ids and autoincremental ids are supported**
- Want to use UUID or a custom ID strategy? No problem!
- **Caching layer required? Easily to add!**
- Require the *[Repository Cache](https://github.com/PHPRepository/repository-cache)* package from Composer to add consistent caching to all operations.

## Installation

Use [Composer](https://getcomposer.org) to install the package:

```json
$ composer require nilportugues/mongodb-repository
```

## Usage

**Show me the code**

See the [/example](https://github.com/PHPRepository/mongodb-repository/tree/master/example) directory. Examples for both `Custom ID` and `AutoIncremental ID` are provided.

**Explanation**

- You require a class implementing the `Mapping` interface provided
- You require a class implementing the `Identity` interface provided. Adds 2 methods, `id()` and `__toString`.
- You require a class to extend from `MongoDBRepository` class provided. Inject your PDO connection and the Mapping class to the `MongoDBRepository`

You're good to go.

--

# Mapping

Mapping must implement the `Mapping` interface.

Mapping classes are used to read data from entities and save them in the storage of choice. This is done by mapping the Entities fields and specifying which fields and how are actually stored in the data storage.

For complex objects, let's say an Entity that has a Value Object, it is possible to still do one single mapping on the Entity and access the Value Object properties to get them stored.

Mappings are also used to hydrate data into it's entities again if the hydrator trait is used.

## Entity class

Remember, an Entity must implement the `Identity` interface to work with MongoDBRepository. This Entity can be any class of yours.

```php
use NilPortugues\Foundation\Domain\Model\Repository\Contracts\Identity;

class User implements Identity
{
protected $userId;
protected $username;
protected $alias;
protected $email;
protected $registeredOn;

/**
* User constructor.
*
* @param $userId
* @param $username
* @param $alias
* @param $email
* @param \DateTime $registeredOn
*/
public function __construct($userId, $username, $alias, $email, \DateTime $registeredOn)
{
$this->userId = $userId;
$this->username = $username;
$this->alias = $alias;
$this->email = $email;
$this->registeredOn = $registeredOn;
}

// ... your getters/setters

public function id()
{
return $this->userId;
}

public function __toString()
{
return (string) $this->id();
}
}
```

## Mapping class

All methods from Mapping interface are mandatory.

```php
use NilPortugues\Foundation\Domain\Model\Repository\Contracts\Mapping;

class UserMapping implements Mapping
{
/**
* Name of the identity field in storage.
*/
public function identity() : string
{
return 'user_id';
}

/**
* Returns the table name.
*/
public function name() : string
{
return 'users';
}

/**
* Keys are object properties without property defined in identity().
* Values its MongoDB column equivalents.
*/
public function map() : array
{
return [
'userId' => 'user_id',
'username' => 'username',
'alias' => 'public_username',
'email' => 'email',

// Notice how we are accessing date value inside
// the \DateTime object! We use dot notation to
// access deep values.

'registeredOn.date' => 'created_at',
];
}

/**
* @param array $data
* @return User
*/
public function fromArray(array $data)
{
return new User(
$data['user_id'],
$data['username'],
$data['public_username'],
$data['email'],
new \DateTime($data['created_at'])
);
}

/**
* The automatic generated strategy used will be the data-store's if set to true.
*/
public function autoGenerateId() : bool
{
return true;
}
}
```

## Mapping the Repository

Finally, it's usage is straight-forward:

```php
use MongoDB\Client;
use NilPortugues\Foundation\Infrastructure\Model\Repository\MongoDB\MongoDBRepository;
use NilPortugues\Foundation\Infrastructure\Model\Repository\MongoDB\MongoDBRepositoryHydrator;

class UserRepository extends MongoDBRepository
{
use MongoDBRepositoryHydrator;
}

$client = new Client();
$mapping = new UserMapping();
$repository = new UserRepository($mapping, $client, 'example_db', 'users');
```

---

# Repository

The repository class implements all the methods required to interact and filter your data.

MongoDBRepository can handle all CRUD operations by default by extending the `MongoDBRepository` class.

If you're not into CRUD, you can also have read-only, write-only and pagination-only repositories:

- For read-only repositories extend the `MongoDBReadRepository` class.
- For write-only repositories extend the `MongoDBWriteRepository` class.
- For pagination-only repositories extend the `MongoDBPageRepository` class.

## Methods

### Available in MongoDBRepository

All the methods listed under MongoDBWriteRepository, MongoDBReadRepository and MongoDBPageRepository.

### Available in MongoDBWriteRepository

- `public function add($value)`
- `public function addAll(array $values)`
- `public function remove(Identity $id)`
- `public function removeAll(Filter $filter = null)`
- `public function transactional(callable $transaction)`
- `public function count(Filter $filter = null)`
- `public function exists(Identity $id)`
- `public function getDriver()`

### Available in MongoDBReadRepository

- `public function find(Identity $id, Fields $fields = null)`
- `public function findBy(Filter $filter = null, Sort $sort = null, Fields $fields = null)`
- `public function findByDistinct(Fields $distinctFields, Filter $filter = null, Sort $sort = null, Fields $fields = null)`
- `public function count(Filter $filter = null)`
- `public function exists(Identity $id)`
- `public function getDriver()`

### Available in MongoDBPageRepository

- `public function findAll(Pageable $pageable = null)`
- `public function count(Filter $filter = null)`
- `public function exists(Identity $id)`
- `public function getDriver()`

---

# Data Operations

All data can be extracted by fields name, using filters, applying ordering and pages, capable of applying fields, filters and ordering criteria.

## Fields

Selecting by field will make hydratation fail. Currently partial object hydratation is not supported.

**Class:** `NilPortugues\Foundation\Domain\Model\Repository\Fields`

**Methods:**
- `public function __construct(array $fields = [])`
- `public function add($field)`
- `public function get()`

## Filtering

**Class:** `NilPortugues\Foundation\Domain\Model\Repository\Filter`

**Methods:**
- `public function filters()`
- `public function must()`
- `public function mustNot()`
- `public function should()`
- `public function clear()`

For **must()**, **mustNot()** and **should()**, the methods available are:

- `public function notStartsWith($filterName, $value)`
- `public function notEndsWith($filterName, $value)`
- `public function notEmpty($filterName)`
- `public function empty($filterName)`
- `public function startsWith($filterName, $value)`
- `public function endsWith($filterName, $value)`
- `public function equal($filterName, $value)`
- `public function notEqual($filterName, $value)`
- `public function includeGroup($filterName, array $value)`
- `public function notIncludeGroup($filterName, array $value)`
- `public function range($filterName, $firstValue, $secondValue)`
- `public function notRange($filterName, $firstValue, $secondValue)`
- `public function notContain($filterName, $value)`
- `public function contain($filterName, $value)`
- `public function beGreaterThanOrEqual($filterName, $value)`
- `public function beGreaterThan($filterName, $value)`
- `public function beLessThanOrEqual($filterName, $value)`
- `public function beLessThan($filterName, $value)`
- `public function clear()`
- `public function get()`
- `public function hasEmpty($filterName)`

## Pagination

Pagination is handled by two objects, `Pageable` that has the requirements to paginate, and `Page` that it's actually the page with the page data, such as page number, total number, and the data.

### Pageable

**Class:** `NilPortugues\Foundation\Domain\Model\Repository\Pageable`

**Methods:**
- `public function __construct($pageNumber, $pageSize, Sort $sort = null, Filter $filter = null, Fieldse $fields = null)`
- `public function offset()`
- `public function pageNumber()`
- `public function sortings()`
- `public function next()`
- `public function pageSize()`
- `public function previousOrFirst()`
- `public function hasPrevious()`
- `public function first()`
- `public function filters()`
- `public function fields()`

### Page object

**Class:** `NilPortugues\Foundation\Domain\Model\Repository\Page`

**Methods:**
- `public function __construct(array $elements, $totalElements, $pageNumber, $totalPages, Sort $sort = null, Filter $filter = null, Fields $fields = null)`
- `public function content()`
- `public function hasPrevious()`
- `public function isFirst()`
- `public function isLast()`
- `public function hasNext()`
- `public function pageSize()`
- `public function pageNumber()`
- `public function totalPages()`
- `public function nextPageable()`
- `public function sortings()`
- `public function filters()`
- `public function fields()`
- `public function previousPageable()`
- `public function totalElements()`
- `public function map(callable $converter)`

## Sorting

**Class:** `NilPortugues\Foundation\Domain\Model\Repository\Sort`

**Methods:**
- `public function __construct(array $properties = [], Order $order = null)`
- `public function andSort(SortInterface $sort)`
- `public function orders()`
- `public function equals(SortInterface $sort)`
- `public function orderFor($propertyName)`
- `public function setOrderFor($propertyName, Order $order)`
- `public function property($propertyName)`

### Ordering

Sometimes you want to sort by multiple fields, this is where Order comes in play.

**Class**: `NilPortugues\Foundation\Domain\Model\Repository\Order`

**Methods:**
- `public function __construct($direction)`
- `public function isDescending()`
- `public function isAscending()`
- `public function __toString()`
- `public function equals($object)`
- `public function direction()`

--

# Quality

To run the PHPUnit tests at the command line, go to the tests directory and issue phpunit.

This library attempts to comply with [PSR-1](http://www.php-fig.org/psr/psr-1/), [PSR-2](http://www.php-fig.org/psr/psr-2/), [PSR-4](http://www.php-fig.org/psr/psr-4/).

If you notice compliance oversights, please send a patch via [Pull Request](https://github.com/PHPRepository/mongodb-repository/pulls).

# Contribute

Contributions to the package are always welcome!

* Report any bugs or issues you find on the [issue tracker](https://github.com/PHPRepository/mongodb-repository/issues/new).
* You can grab the source code at the package's [Git Repository](https://github.com/PHPRepository/mongodb-repository).

# Support

Get in touch with me using one of the following means:

- Emailing me at
- Opening an [Issue](https://github.com/PHPRepository/mongodb-repository/issues/new)

# Authors

* [Nil Portugués Calderó](http://nilportugues.com)
* [The Community Contributors](https://github.com/PHPRepository/mongodb-repository/graphs/contributors)

# License
The code base is licensed under the [MIT license](LICENSE).