34 Commits

Author SHA1 Message Date
81ad81f7bf v1.1.0 2022-09-09 10:33:54 -04:00
7fc7de7390 Added Factory implementation 2022-09-09 10:33:19 -04:00
55275d2d75 Factory Implementation 2022-09-09 10:31:38 -04:00
12065b376b v2.0.0 2022-09-08 21:43:02 -04:00
3bc54fb9d1 Second version 2022-09-08 21:42:32 -04:00
230faaf3cc Abstracts 2022-09-08 21:35:33 -04:00
7f33475e35 Interfaces 2022-09-08 21:35:25 -04:00
8beb07819c Dependencies 2022-09-08 21:24:50 -04:00
91886eca0f Ignores 2022-09-08 21:24:41 -04:00
2d44c5d057 Cleanup 2022-09-08 21:24:29 -04:00
123d46d33c Fixed Model joins 2021-12-20 23:34:52 -03:00
fbd5b350bd FIX: sym -> symb, cast object to array, and use alias when alias IS NOT empty 2021-12-20 23:33:02 -03:00
65c224c636 Merge branch 'develop' into master 2021-08-01 20:51:44 -04:00
2bf938a9b7 FIX: array_walk returns bool and tertiary operator 2021-08-01 20:50:42 -04:00
6cd26a88ea Merge branch 'develop' into master 2021-08-01 20:34:04 -04:00
293b5af3ff FIX: self -> static 2021-08-01 20:31:49 -04:00
022ba575b7 Merge branch 'develop' into master 2021-06-04 21:15:26 -04:00
643c3e714f FIX: Find where in add 2021-05-23 21:27:13 -04:00
db2864395c FIX: Factory for find 2021-05-23 21:07:48 -04:00
1cd06d5fe6 Add and Edit Model 2021-05-23 21:03:10 -04:00
c913f65b91 Considera leer el dato y correccion en los nombres de los metodos 2021-03-16 00:22:13 -03:00
244d8cc414 FIX: null cases & method name 2021-03-16 00:19:51 -03:00
c8a7781c88 Merge branch 'develop' 2021-03-15 11:17:11 -03:00
630c971b45 Date traits 2021-03-15 11:14:10 -03:00
1505539e61 Merge branch 'develop' into master 2021-03-09 00:19:55 -03:00
c441d41a02 FIX: Not through enough 2021-03-09 00:19:32 -03:00
3087a48c43 Merge branch 'develop' into master 2021-03-09 00:15:12 -03:00
3aed13e6a0 FIX: usar correcta descripcion 2021-03-09 00:14:31 -03:00
43f545516d Merge branch 'develop' 2020-08-04 01:33:09 -04:00
4d64143dbc FIX: No se configuraba la creacion de nuevo modelo 2020-08-04 01:32:59 -04:00
b757ed19b2 Merge branch 'develop' 2020-08-03 23:54:16 -04:00
1dc21d8fb7 FIX: Interface differences 2020-08-03 23:54:07 -04:00
9dc71e4d77 Merge branch 'develop' 2020-08-03 23:51:05 -04:00
65bec43b45 FIX: Herencia de cambios anteriores 2020-08-03 23:50:58 -04:00
12 changed files with 320 additions and 543 deletions

7
.gitignore vendored
View File

@ -1,3 +1,6 @@
# Composer
/vendor/
composer.lock
**/vendor/
**/*.lock
# PHPStorm
**/.idea/

View File

@ -1,14 +0,0 @@
<?php
namespace ProVM\Common\Alias;
use ProVM\Common\Factory\Model as ModelFactory;
interface Model {
public function getTable(): string;
public function setFactory(ModelFactory $factory): Model;
public function parentOf(string $child_model_class, array $relation_definitions): array;
public function childOf(string $parent_model_class, array $relation_definitions): Model;
public function siblingOf(string $sibling_model_class, string $connecting_table, array $relation_definitions): array;
public function toArray(): array;
}

View File

@ -1,125 +0,0 @@
<?php
namespace ProVM\Common\Define;
use \Model as BaseModel;
use ProVM\Common\Alias\Model as ModelInterface;
use ProVM\Common\Factory\Model as ModelFactory;
abstract class Model extends BaseModel implements ModelInterface {
const SELF_KEY = 'self_key';
const PARENT_KEY = 'parent_key';
const CHILD_KEY = 'child_key';
const SIBLING_KEY = 'sibling_key';
const SELF_CONNECT_KEY = 'self_connect_key';
const SIBLING_CONNECT_KEY = 'sibling_connect_key';
public function getTable(): string {
return static::$_table;
}
protected $factory;
public function setFactory(ModelFactory $factory): ModelInterface {
$this->factory = $factory;
return $this;
}
protected function checkDefinitions(array $definitions, array $required, array $default) {
foreach ($default as $key => $value) {
if (!isset($definitions[$key])) {
$definitions[$key] = $value;
}
}
foreach ($required as $definition) {
if (!isset($definitions[$definition])) {
$trace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 1);
throw new \InvalidArgumentException($definition . ' is required for ' . $trace[1]['function'] . ' in ' . get_called_class());
}
}
return $definitions;
}
public function parentOf(string $child_model_class, array $relation_definitions): array {
$relation_definitions = $this->checkDefinitions($relation_definitions, [
Model::SELF_KEY,
Model::CHILD_KEY
], [
Model::SELF_KEY => 'id'
]);
return $this->factory
->find($child_model_class)
->where([
[
$relation_definitions[Model::CHILD_KEY],
$this->{$relation_definitions[Model::SELF_KEY]}
]
])
->many();
}
public function childOf(string $parent_model_class, array $relation_definitions): ModelInterface {
$relation_definitions = $this->checkDefinitions($relation_definitions, [
Model::SELF_KEY,
Model::PARENT_KEY
], [
Model::PARENT_KEY => 'id'
]);
$parent_table = (new $parent_model_class())->getTable();
return $this->factory
->find($parent_model_class)
->where([
[
$relation_definitions[Model::PARENT_KEY],
$this->{$relation_definitions[Model::SELF_KEY]}
]
])
->one();
}
public function siblingOf(string $sibling_model_class, string $connecting_table, array $relation_definitions): array {
$relation_definitions = $this->checkDefinitions($relation_definitions, [
Model::SELF_KEY,
Model::SIBLING_KEY,
Model::SELF_CONNECT_KEY,
Model::SIBLING_CONNECT_KEY
], [
Model::SELF_KEY => 'id',
Model::SIBLING_KEY => 'id'
]);
$sibling_table = (new $sibling_model_class())->getTable();
return $this->find($sibling_model_class)
->select([
[
$sibling_table,
'*'
],
[
$connecting_table,
'*'
]
])
->join([
[
$connecting_table,
implode('.', [
$connecting_table,
$relation_definitions[Model::SIBLING_CONNECT_KEY]
]),
implode('.', [
$sibling_table,
$relation_definitions[Model::SIBLING_KEY]
])
]
])
->where([
[
implode('.', [
$connecting_table,
$relation_definitions[Model::SELF_CONNECT_KEY]
]),
$this->{$relation_definitions[Model::SELF_KEY]}
]
])
->many();
}
public function toArray(): array {
return $this->asArray();
}
}

View File

@ -1,388 +0,0 @@
<?php
namespace ProVM\Common\Factory;
use ORM;
use Model as BaseModel;
use ProVM\Common\Alias\Model as ModelInterface;
class Model {
public function reset(): Model {
$reset = [
'class',
'columns',
'joins',
'conditions',
'grouping',
'ordering',
'limit',
'offset'
];
foreach ($reset as $p) {
$this->{$p} = null;
}
return $this;
}
public function create(string $model_class, array $data = null): ModelInterface {
if ($data !== null) {
$model = $this->find($model_class);
foreach ($data as $f => $v) {
$model = $model->where([[$f, $v]]);
}
$model = $model->one();
if ($model !== null) {
return $model;
}
}
return BaseModel::factory($model_class)->create($data);
}
protected $class;
public function find(string $model_class): Model {
if (!class_exists($model_class)) {
throw new \InvalidArgumentException('El modelo ' . $model_class . ' no existe.');
}
$this->reset();
$this->class = $model_class;
return $this;
}
protected $columns;
public function select($columns): Model {
if ($this->columns === null) {
$this->columns = [];
}
if (!is_array($columns)) {
$columns = [$columns];
}
foreach ($columns as $c) {
$col = (object) [
'table' => '',
'column' => $c
];
if (is_array($c)) {
$col = (object) [
'table' => $c['table'] ?? $c[0],
'column' => $c['column'] ?? $c[1]
];
}
$this->columns []= $col;
}
return $this;
}
protected $joins;
public function join(array $joins): Model {
if ($this->joins === null) {
$this->joins = [];
}
foreach ($joins as $j) {
$join = (object) [
'table' => $j['table'] ?? $j[0],
'from' => $j['from'] ?? $j[1],
'to' => $j['to'] ?? $j[2],
'sym' => $j['sym'] ?? ($j[3] ?? '='),
'alias' => $j['alias'] ?? '',
'type' => strtolower($j['type']) ?? '',
'params' => $j['params'] ?? ''
];
$this->joins []= $join;
}
return $this;
}
protected $conditions;
public function where(array $conditions): Model {
if ($this->conditions === null) {
$this->conditions = [];
}
foreach ($conditions as $c) {
$cond = (object) [
'column' => $c['column'] ?? $c[0],
'value' => $c['value'] ?? $c[1],
'sym' => strtolower($c['sym'] ?? ($c[2] ?? '=')),
'type' => strtolower($c['type']) ?? ''
];
$this->conditions []= $cond;
}
return $this;
}
protected $grouping;
public function group($groups): Model {
if ($this->grouping === null) {
$this->grouping = [];
}
if (!is_array($groups)) {
$groups = [$groups];
}
foreach ($groups as $g) {
$this->grouping []= $g;
}
return $this;
}
protected $ordering;
public function order($orders): Model {
if ($this->ordering === null) {
$this->ordering = [];
}
if (!is_array($orders)) {
$orders = [$orders];
}
foreach ($orders as $o) {
$order = (object) [
'column' => $o,
'direction' => 'asc'
];
if (is_array($o)) {
$order = (object) [
'column' => $o['column'] ?? $o[0],
'direction' => strtolower($o['direction']) ?? $o[1]
];
}
$this->ordering []= $order;
}
return $this;
}
protected $limit;
public function limit(int $limit): Model {
$this->limit = $limit;
}
protected $offset;
public function offset(int $offset): Model {
$this->offset = $offset;
}
protected function parseSelect(ORM $orm): ORM {
if ($this->columns === null) {
return $orm;
}
foreach ($this->columns as $col) {
$orm = $orm->select(trim(implode('.', $col), '.'));
}
return $orm;
}
protected function parseJoin(ORM $orm): ORM {
if ($this->joins === null) {
return $orm;
}
foreach ($this->joins as $join) {
$method = 'join';
switch ($join->type) {
case 'raw':
$method = 'rawJoin';
break;
case 'inner':
$method = 'innerJoin';
break;
case 'left':
case 'left outer':
case 'left_outer':
case 'leftouter':
$method = 'leftOuterJoin';
break;
case 'right':
case 'right outer':
case 'right_outer':
case 'rightouter':
$method = 'rightOuterJoin';
break;
case 'full':
case 'full outer':
case 'full_outer':
case 'fullouter':
$method = 'fullOuterJoin';
break;
}
if ($join->type == 'raw') {
$orm = $orm->{$method}($join->table, [$join->from, $join->symb, $join->to], $join->alias, $join->params);
} elseif ($join->alias === '') {
$orm = $orm->{$method}($join->table, [$join->from, $join->symb, $join->to], $join->alias);
} else {
$orm = $orm->{$method}($join->table, [$join->from, $join->symb, $join->to]);
}
}
return $orm;
}
protected function parseWhere(ORM $orm): ORM {
if ($this->conditions === null) {
return $orm;
}
foreach ($this->conditions as $cond) {
$method = 'where';
switch ($cond->sym) {
case '<':
$method = 'whereLt';
break;
case '<=':
$method = 'whereLte';
break;
case '>':
$method = 'whereGt';
break;
case '>=':
$method = 'whereGte';
break;
case '!=':
$method = 'whereNotEqual';
break;
case 'like':
$method = 'whereLike';
break;
case 'not like':
case 'not_like':
case 'notlike':
$method = 'whereNotLike';
break;
}
switch ($cond->type) {
case 'equal':
$method = 'whereEqual';
break;
case 'not equal':
case 'not_equal':
case 'notequal':
$method = 'whereNotEqual';
break;
case 'less':
case 'less than':
case 'less_than':
case 'lessthan':
case 'lt':
$method = 'whereLt';
break;
case 'less equal':
case 'less_equal':
case 'lessequal':
case 'less than equal':
case 'less_than_equal':
case 'lessthanequal':
case 'lte':
$method = 'whereLte';
break;
case 'greater':
case 'greater than':
case 'greater_than':
case 'greaterthan':
case 'gt':
$method = 'whereGt';
break;
case 'greater equal':
case 'greater_equal':
case 'greaterequal':
case 'greater than equal':
case 'greater_than_equal':
case 'greaterthanequal':
case 'gte':
$method = 'whereGte';
break;
case 'like':
$method = 'whereLike';
break;
case 'not like':
case 'not_like':
case 'notlike':
$method = 'whereNotLike';
break;
case 'in':
$method = 'whereIn';
break;
case 'not in':
case 'not_in':
case 'notin':
$method = 'whereNotIn';
break;
case 'raw':
$method = 'rawWhere';
break;
}
$orm = $orm->{$method}($cond->column, $cond->value);
}
return $orm;
}
protected function parseGroup(ORM $orm): ORM {
if ($this->grouping === null) {
return $orm;
}
foreach ($this->grouping as $group) {
if (strpos($group, '(') !== false) {
$orm = $orm->groupByExpr($group);
} else {
$orm = $orm->groupBy($group);
}
}
return $orm;
}
protected function parseOrder(ORM $orm): ORM {
if ($this->ordering === null) {
return $orm;
}
foreach ($this->ordering as $order) {
if (strpos($order->column, '(') !== false) {
$orm = $orm->orderByExpr($order->column);
continue;
}
switch ($order->direction) {
case 'asc':
case 'ascending':
$orm = $orm->orderByAsc($order->column);
break;
case 'desc':
case 'descending':
$orm = $orm->orderByDesc($order->column);
break;
}
}
return $orm;
}
protected function parseLimit(ORM $orm): ORM {
if ($this->limit === null) {
return $orm;
}
return $orm->limit($this->limit);
}
protected function parseOffset(ORM $orm): ORM {
if ($this->offset === null) {
return $orm;
}
return $orm->offset($this->offset);
}
public function one($id = null): ?ModelInterface {
$result = $this->build()->findOne($id);
if (!$result) {
return null;
}
$result->setFactory($this);
return $result;
}
public function many(): ?array {
$results = $this->build()->findMany();
if (!$results) {
return null;
}
foreach ($results as &$r) {
$r->setFactory($this);
}
return $results;
}
public function array(): ?array {
$results = $this->build()->findArray();
if (!$results) {
return null;
}
return $results;
}
protected function build(): ORM {
$orm = BaseModel::factory($this->class);
$methods = [
'select',
'join',
'where',
'group',
'order',
'limit',
'offset'
];
foreach ($methods as $m) {
$method = 'parse' . ucfirst($m);
$orm = $this->{$method}($orm);
}
return $orm;
}
}

View File

@ -1,6 +0,0 @@
<?php
namespace ProVM\Common\Form;
use ProVM\Common\Define\Model as BaseModel;
abstract class Model extends BaseModel {}

View File

@ -1,13 +1,14 @@
{
"name": "provm/models",
"description": "Model handling using j4mie/paris",
"type": "library",
"description": "Model with Repository for Database Mapping",
"type": "project",
"require": {
"j4mie/paris": "^1.5"
"provm/database": "^2",
"provm/query_builder": "^1"
},
"require-dev": {
"phpunit/phpunit": "^9.2",
"kint-php/kint": "^3.3"
"phpunit/phpunit": "^9",
"kint-php/kint": "^4"
},
"license": "prioprietary",
"authors": [
@ -17,8 +18,26 @@
}
],
"autoload": {
"psr-4": {
"ProVM\\Common\\": "common"
}
"psr-4": {
"ProVM\\": "src/"
}
},
"repositories": [
{
"type": "git",
"url": "https://git.provm.cl/ProVM/database.git"
},
{
"type": "git",
"url": "https://git.provm.cl/ProVM/query_builder.git"
}
],
"config": {
"http-basic": {
"git.provm.cl": {
"username": "aldarien",
"password": "$&At'GQmd@Ul;=j@'2d#@N&H"
}
}
}
}

64
src/Alias/Model.php Normal file
View File

@ -0,0 +1,64 @@
<?php
namespace ProVM\Alias;
use ProVM\Concept\Model\Factory;
use ProVM\Concept\Model as ModelInterface;
abstract class Model implements ModelInterface
{
protected Factory $factory;
public function setFactory(Factory $factory): ModelInterface
{
$this->factory = $factory;
return $this;
}
public function getFactory(): Factory
{
return $this->factory;
}
protected int $id;
public function setId(int $id): ModelInterface
{
$this->id = $id;
return $this;
}
public function getId(): int
{
return $this->id;
}
protected bool $new;
public function setNew(): ModelInterface
{
$this->new = true;
return $this;
}
public function isNew(): bool
{
return $this->new ?? false;
}
protected bool $dirty;
public function setDirty(): ModelInterface
{
$this->dirty = true;
return $this;
}
public function isDirty(): bool
{
return $this->dirty ?? $this->isNew();
}
public function save(): void
{
if ($this->isDirty()) {
$this->getFactory()->get(get_class($this))->save($this);
}
}
public function edit(array $data): void
{
foreach ($data as $key => $val) {
$m = 'set' . ucwords($key);
$this->{$m}($val);
}
$this->isDirty();
}
}

View File

@ -0,0 +1,119 @@
<?php
namespace ProVM\Alias\Model;
use ProVM\Concept\Database\Connection;
use ProVM\Concept\Database\QueryBuilder;
use ProVM\Concept\Model;
use ProVM\Concept\Model\Factory;
use ProVM\Concept\Model\Repository as RepositoryInterface;
abstract class Repository implements RepositoryInterface
{
public function __construct(Connection $connection, QueryBuilder $builder, Factory $factory)
{
$this->setConnection($connection)
->setQueryBuilder($builder)
->setFactory($factory)
->setup();
}
protected Connection $connection;
public function setConnection(Connection $connection): Repository
{
$this->connection = $connection;
return $this;
}
public function getConnection(): Connection
{
return $this->connection;
}
protected QueryBuilder $builder;
public function setQueryBuilder(QueryBuilder $builder): RepositoryInterface
{
$this->builder = $builder;
return $this;
}
public function getQueryBuilder(): QueryBuilder
{
return $this->builder;
}
protected Factory $factory;
public function setFactory(Factory $factory): RepositoryInterface
{
$this->factory = $factory;
return $this;
}
public function getFactory(): Factory
{
return $this->factory;
}
protected string $table;
public function setTable(string $table): RepositoryInterface
{
$this->table = $table;
return $this;
}
public function getTable(): string
{
return $this->table;
}
protected array $columns;
public function setColumns(array $columns): RepositoryInterface
{
foreach ($columns as $column) {
$this->addColumn($column);
}
return $this;
}
public function addColumn(string $column): RepositoryInterface
{
$this->columns []= $column;
return $this;
}
public function getColumns(): array
{
return $this->columns;
}
public function save(Model $model): void
{
if (!$model->isDirty() and !$model->isNew()) {
return;
}
$cols = [];
$values = [];
foreach ($this->getColumns() as $column) {
$m = 'get' . ucwords($column);
$cols []= '?';
$values []= $model->{$m}();
}
$query = $this->getQueryBuilder()->insert($this->getTable())->columns($this->getColumns())->values($cols);
$this->getConnection()->execute($query, $values);
}
public function edit(Model $model, array $data): Model
{
foreach ($this->getColumns() as $col) {
if (isset($data[$col])) {
$m = 'set' . ucwords($col);
$model->{$m}($data[$col]);
}
}
return $model;
}
public function fetchById(int $id): Model
{
$query = $this->getQueryBuilder()
->select()
->from($this->getTable())
->where([['id', '?']])
->limit(1);
return $this->load($this->getConnection()->execute($query, [$id])->getAsArray()[0]);
}
public function fetchAll(): array
{
$query = $this->getQueryBuilder()
->select()
->from($this->getTable());
return array_map([$this, 'load'], $this->getConnection()->query($query)->getAsArray());
}
}

18
src/Concept/Model.php Normal file
View File

@ -0,0 +1,18 @@
<?php
namespace ProVM\Concept;
use ProVM\Concept\Model\Factory;
interface Model
{
public function setFactory(Factory $factory): Model;
public function getFactory(): Factory;
public function setId(int $id): Model;
public function getId(): int;
public function setNew(): Model;
public function isNew(): bool;
public function setDirty(): Model;
public function isDirty(): bool;
public function save(): void;
public function edit(array $data): void;
}

View File

@ -0,0 +1,13 @@
<?php
namespace ProVM\Concept\Model;
use Psr\Container\ContainerInterface;
interface Factory
{
public function setContainer(ContainerInterface $container): Factory;
public function getContainer(): ContainerInterface;
public function setNamespace(string $namespace): Factory;
public function getNamespace(): string;
public function get(string $repository_name): Repository;
}

View File

@ -0,0 +1,28 @@
<?php
namespace ProVM\Concept\Model;
use ProVM\Concept\Database\Connection;
use ProVM\Concept\Database\QueryBuilder;
use ProVM\Concept\Model;
interface Repository
{
public function setConnection(Connection $connection): Repository;
public function getConnection(): Connection;
public function setQueryBuilder(QueryBuilder $builder): Repository;
public function getQueryBuilder(): QueryBuilder;
public function setFactory(Factory $factory): Repository;
public function getFactory(): Factory;
public function setup(): Repository;
public function setTable(string $table): Repository;
public function getTable(): string;
public function setColumns(array $columns): Repository;
public function addColumn(string $column): Repository;
public function getColumns(): array;
public function load(array $data): Model;
public function save(Model $model): void;
public function edit(Model $model, array $data): Model;
public function create(array $data): Model;
public function fetchById(int $id): Model;
public function fetchAll(): array;
}

View File

@ -0,0 +1,46 @@
<?php
namespace ProVM\Implement\Model;
use Psr\Container\ContainerInterface;
use ProVM\Concept\Model\Factory as FactoryInterface;
use ProVM\Concept\Model\Repository;
class Factory implements FactoryInterface
{
public function __construct(ContainerInterface $container)
{
$this->setContainer($container);
}
protected ContainerInterface $container;
public function setContainer(ContainerInterface $container): FactoryInterface
{
$this->container = $container;
return $this;
}
public function getContainer(): ContainerInterface
{
return $this->container;
}
protected string $namespace;
public function setNamespace(string $namespace): FactoryInterface
{
$this->namespace = $namespace;
return $this;
}
public function getNamespace(): string
{
return $this->namespace;
}
protected function buildRepository(string $repository_name): string
{
return implode("\\", [
$this->getNamespace(),
$repository_name
]);
}
public function get(string $repository_name): Repository
{
return $this->getContainer()->get($this->buildRepository($repository_name));
}
}