ulmus/src/Repository.php

455 lines
12 KiB
PHP
Raw Normal View History

2019-08-21 20:13:00 +00:00
<?php
namespace Ulmus;
use Ulmus\Common\EntityResolver;
class Repository
{
const DEFAULT_ALIAS = "this";
protected ? ConnectionAdapter $adapter;
protected QueryBuilder $queryBuilder;
protected EntityResolver $entityResolver;
public string $alias;
public string $entityClass;
public function __construct(string $entity, string $alias = self::DEFAULT_ALIAS, ConnectionAdapter $adapter = null) {
$this->entityClass = $entity;
$this->alias = $alias;
$this->entityResolver = Ulmus::resolveEntity($entity);
$this->adapter = $adapter ?? $this->entityResolver->databaseAdapter() ?? Ulmus::$defaultAdapter;
$this->queryBuilder = new QueryBuilder( $this->adapter->adapter() );
2019-08-21 20:13:00 +00:00
}
2020-01-31 21:38:48 +00:00
public function loadOne() : ? object
2019-08-21 20:13:00 +00:00
{
2020-01-31 21:38:48 +00:00
return $this->limit(1)->collectionFromQuery()[0] ?? null;
2019-08-21 20:13:00 +00:00
}
public function loadOneFromField($field, $value) : ? object
{
return $this->where($field, $value)->loadOne();
}
2020-01-31 21:38:48 +00:00
public function loadFromPk($value, $primaryKey = "id") : ? object
2019-08-21 20:13:00 +00:00
{
return $this->loadOneFromField($primaryKey, $value);
2019-08-21 20:13:00 +00:00
}
2020-01-31 21:38:48 +00:00
public function loadAll() : EntityCollection
2019-08-21 20:13:00 +00:00
{
2020-01-31 21:38:48 +00:00
return $this->collectionFromQuery();
2019-08-21 20:13:00 +00:00
}
public function loadFromField($field, $value) : EntityCollection
{
return $this->where($field, $value)->collectionFromQuery();
}
public function count() : int
{
$this->select("count(*) as totalItem")->selectSqlQuery();
$this->finalizeQuery();
foreach(Ulmus::iterateQueryBuilder($this->queryBuilder, $this->adapter) as $entityData) {
return $entityData['totalItem'];
}
return 0;
}
2019-08-21 20:13:00 +00:00
public function deleteOne()
{
return $this->limit(1)->deleteSqlQuery()->runQuery();
}
public function deleteAll()
{
return $this->deleteSqlQuery()->runQuery();
}
public function deleteFromPk($value) : bool
{
if ( $value !== 0 && empty($value) ) {
throw new Exception\EntityPrimaryKeyUnknown("A primary key value has to be defined to delete an item.");
}
return (bool) $this->wherePrimaryKey($value)->deleteOne()->rowCount();
}
2020-01-31 21:38:48 +00:00
public function save(object $entity) : bool
{
2020-02-05 21:19:57 +00:00
$dataset = $entity->toArray();
$primaryKeyDefinition = Ulmus::resolveEntity($this->entityClass)->getPrimaryKeyField();
2020-02-05 21:19:57 +00:00
if ( ! $entity->isLoaded() ) {
$statement = $this->insertSqlQuery($dataset)->runQuery();
if ( ( 0 !== $statement->lastInsertId ) &&
( null !== $primaryKeyDefinition )) {
$pkField = key($primaryKeyDefinition);
$entity->$pkField = $statement->lastInsertId;
}
return true;
2020-02-05 21:19:57 +00:00
}
else {
if ( $primaryKeyDefinition === null ) {
throw new \Exception(sprintf("No primary key found for entity %s", $this->entityClass));
}
if ( [] !== $diff = $this->generateDatasetDiff($entity) ) {
$pkField = key($primaryKeyDefinition);
$pkFieldName = $primaryKeyDefinition[$pkField]->name ?? $pkField;
$this->where($pkFieldName, $dataset[$pkFieldName]);
$update = $this->updateSqlQuery($diff)->runQuery();
return $update ? (bool) $update->rowCount() : false;
}
2020-02-05 21:19:57 +00:00
}
2020-01-31 21:38:48 +00:00
2020-02-05 21:19:57 +00:00
return false;
2020-01-31 21:38:48 +00:00
}
public function saveAll(EntityCollection $collection) : void
2020-01-31 21:38:48 +00:00
{
2020-02-05 21:19:57 +00:00
foreach($collection as $entity) {
$this->save($entity);
}
2020-01-31 21:38:48 +00:00
}
public function generateDatasetDiff(object $entity) : array
{
return array_diff_assoc( array_change_key_case($entity->toArray()), array_change_key_case($entity->entityGetDataset(true)) );
}
2019-08-21 20:13:00 +00:00
public function yieldAll() : \Generator
{
}
public function select($fields) : self
{
$this->queryBuilder->select($fields);
2019-08-21 20:13:00 +00:00
return $this;
}
2020-02-05 21:19:57 +00:00
public function insert(array $fieldlist, string $table, string $alias, ? string $schema) : self
{
$this->queryBuilder->insert($fieldlist, $table, $alias, $schema);
return $this;
}
public function values(array $dataset) : self
{
$this->queryBuilder->values($dataset);
return $this;
}
public function update(array $fieldlist, string $table, string $alias, ? string $schema) : self
{
$this->queryBuilder->update($table, $alias, $schema);
return $this;
}
public function set(array $dataset) : self
{
$this->queryBuilder->set($dataset);
return $this;
}
public function delete() : self
{
$this->queryBuilder->delete($this->alias);
return $this;
}
public function from(string $table, ? string $alias, ? string $schema) : self
2019-08-21 20:13:00 +00:00
{
$this->queryBuilder->from($table, $alias, $this->adapter->adapter()->database, $schema);
2019-08-21 20:13:00 +00:00
return $this;
}
public function join(string $type, $table, $field, $value) : self
{
$this->queryBuilder->join($type, $table, $field, $value);
2019-08-21 20:13:00 +00:00
return $this;
}
public function open(string $condition = Query\Where::CONDITION_AND) : self
{
$this->queryBuilder->open($condition);
2019-08-21 20:13:00 +00:00
return $this;
}
public function orOpen() : self
{
return $this->open(Query\Where::CONDITION_OR);
}
public function close() : self
{
$this->queryBuilder->close();
2019-08-21 20:13:00 +00:00
return $this;
}
public function where($field, $value, string $operator = Query\Where::OPERATOR_EQUAL) : self
{
$this->queryBuilder->where($field, $value, $operator, Query\Where::CONDITION_AND);
2019-08-21 20:13:00 +00:00
return $this;
}
public function and($field, $value, string $operator = Query\Where::OPERATOR_EQUAL) : self
{
return $this->where($field, $value, $operator);
}
public function or($field, $value, string $operator = Query\Where::OPERATOR_EQUAL) : self
{
$this->queryBuilder->where($field, $value, $operator, Query\Where::CONDITION_OR);
2019-08-21 20:13:00 +00:00
return $this;
}
public function notWhere($field, $value, string $operator = Query\Where::OPERATOR_NOT_EQUAL) : self
2019-08-21 20:13:00 +00:00
{
$this->queryBuilder->where($field, $value, $operator, Query\Where::CONDITION_AND, true);
2019-08-21 20:13:00 +00:00
return $this;
}
public function orNot($field, $value, string $operator = Query\Where::OPERATOR_EQUAL) : self
{
$this->queryBuilder->notWhere($condition, Query\Where::CONDITION_OR, true);
2019-08-21 20:13:00 +00:00
return $this;
}
public function having() : self
{
return $this;
}
public function notHaving() : self
{
return $this;
}
public function in($field, $value, string $operator = Query\Where::OPERATOR_EQUAL) : self
{
$this->queryBuilder->where($field, $value, $operator);
2019-08-21 20:13:00 +00:00
return $this;
}
public function orIn($field, $value, string $operator = Query\Where::OPERATOR_EQUAL) : self
{
$this->queryBuilder->where($field, $value, $operator, Query\Where::CONDITION_OR);
2019-08-21 20:13:00 +00:00
return $this;
}
public function notIn($field, $value) : self
{
$this->queryBuilder->where($field, $value, Query\Where::OPERATOR_NOT_EQUAL);
2019-08-21 20:13:00 +00:00
return $this;
}
public function orNotIn($field, $value, string $operator = Query\Where::OPERATOR_EQUAL) : self
{
return $this->orNot($field, $value, Query\Where::OPERATOR_NOT_EQUAL, Query\Where::CONDITION_OR);
}
public function like($field, $value) : self
{
$this->queryBuilder->where($field, $value, Query\Where::OPERATOR_LIKE, Query\Where::CONDITION_AND);
2019-08-21 20:13:00 +00:00
return $this;
}
public function notLike($field, $value) : self
{
$this->queryBuilder->where($field, $value, Query\Where::OPERATOR_LIKE, Query\Where::CONDITION_AND, true);
2019-08-21 20:13:00 +00:00
return $this;
}
public function match() : self
{
}
public function notMatch() : self
{
}
public function between() : self
{
}
public function notBetween() : self
{
}
public function groupBy() : self
{
#$this->queryBuilder->groupBy();
return $this;
}
public function orderBy($field, ? string $direction = null) : self
2019-08-21 20:13:00 +00:00
{
$this->queryBuilder->orderBy($field, $direction);
2019-08-21 20:13:00 +00:00
return $this;
}
public function limit(int $value) : self
{
$this->queryBuilder->limit($value);
2019-08-21 20:13:00 +00:00
return $this;
}
public function offset(int $value) : self
{
$this->queryBuilder->offset($value);
2019-08-21 20:13:00 +00:00
return $this;
}
public function commit() : self
{
return $this;
}
public function rollback() : self
{
return $this;
}
2020-02-05 21:19:57 +00:00
public function wherePrimaryKey($value) : self
{
if ( null === $primaryKeyField = Ulmus::resolveEntity($this->entityClass)->getPrimaryKeyField() ) {
throw new Exception\EntityPrimaryKeyUnknown("Entity has no field containing attributes 'primary_key'");
}
$pkField = key($primaryKeyField);
return $this->where($primaryKeyField[$pkField]->name ?? $pkField, $value);
2020-02-05 21:19:57 +00:00
}
2019-08-21 20:13:00 +00:00
protected function collectionFromQuery() : EntityCollection
{
2019-08-21 20:13:00 +00:00
$class = $this->entityClass;
$entityCollection = new EntityCollection();
$this->selectSqlQuery();
$this->finalizeQuery();
2019-08-21 20:13:00 +00:00
foreach(Ulmus::iterateQueryBuilder($this->queryBuilder, $this->adapter) as $entityData) {
2019-08-21 20:13:00 +00:00
$entityCollection->append( ( new $class() )->entityFillFromDataset($entityData) );
}
return $entityCollection;
}
public function runQuery() : ? \PDOStatement
{
$this->finalizeQuery();
return Ulmus::runQuery($this->queryBuilder, $this->adapter);
}
2020-02-05 21:19:57 +00:00
protected function insertSqlQuery(array $dataset) : self
{
if ( null === $this->queryBuilder->getFragment(Query\Insert::class) ) {
$this->insert(array_keys($dataset), $this->entityResolver->tableName(), $this->alias, $this->entityResolver->schemaName());
2020-02-05 21:19:57 +00:00
}
$this->values($dataset);
2020-02-05 21:19:57 +00:00
return $this;
}
protected function updateSqlQuery(array $dataset) : self
{
if ( null === $this->queryBuilder->getFragment(Query\Update::class) ) {
$this->update($dataset, $this->entityResolver->tableName(), $this->alias, $this->entityResolver->schemaName());
2020-02-05 21:19:57 +00:00
}
$this->set($dataset);
2020-02-05 21:19:57 +00:00
return $this;
}
2019-08-21 20:13:00 +00:00
protected function selectSqlQuery() : self
{
if ( null === $this->queryBuilder->getFragment(Query\Select::class) ) {
2019-08-21 20:13:00 +00:00
$this->select("{$this->alias}.*");
}
if ( null === $this->queryBuilder->getFragment(Query\From::class) ) {
$this->from($this->entityResolver->tableName(), $this->alias, $this->entityResolver->schemaName());
2019-08-21 20:13:00 +00:00
}
return $this;
}
protected function deleteSqlQuery() : self
{
if ( null === $this->queryBuilder->getFragment(Query\Delete::class) ) {
$this->delete();
}
if ( null === $this->queryBuilder->getFragment(Query\From::class) ) {
$this->from($this->entityResolver->tableName(), null, $this->entityResolver->schemaName());
}
return $this;
}
2019-08-21 20:13:00 +00:00
protected function fromRow($row) : self
{
}
protected function fromCollection($rows) : self
{
}
public function table()
{
return "REFLECT TABLE";
}
protected function finalizeQuery() : void {}
2019-08-21 20:13:00 +00:00
}