mirror of
https://github.com/espocrm/espocrm.git
synced 2026-06-29 15:36:07 +00:00
1121 lines
27 KiB
PHP
1121 lines
27 KiB
PHP
<?php
|
||
/************************************************************************
|
||
* This file is part of EspoCRM.
|
||
*
|
||
* EspoCRM – Open Source CRM application.
|
||
* Copyright (C) 2014-2024 Yurii Kuznietsov, Taras Machyshyn, Oleksii Avramenko
|
||
* Website: https://www.espocrm.com
|
||
*
|
||
* This program is free software: you can redistribute it and/or modify
|
||
* it under the terms of the GNU Affero General Public License as published by
|
||
* the Free Software Foundation, either version 3 of the License, or
|
||
* (at your option) any later version.
|
||
*
|
||
* This program is distributed in the hope that it will be useful,
|
||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
* GNU Affero General Public License for more details.
|
||
*
|
||
* You should have received a copy of the GNU Affero General Public License
|
||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||
*
|
||
* The interactive user interfaces in modified source and object code versions
|
||
* of this program must display Appropriate Legal Notices, as required under
|
||
* Section 5 of the GNU Affero General Public License version 3.
|
||
*
|
||
* In accordance with Section 7(b) of the GNU Affero General Public License version 3,
|
||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||
************************************************************************/
|
||
|
||
namespace Espo\ORM;
|
||
|
||
use Espo\ORM\Value\ValueAccessorFactory;
|
||
use Espo\ORM\Value\ValueAccessor;
|
||
|
||
use stdClass;
|
||
use InvalidArgumentException;
|
||
use RuntimeException;
|
||
|
||
use const E_USER_DEPRECATED;
|
||
use const JSON_THROW_ON_ERROR;
|
||
|
||
class BaseEntity implements Entity
|
||
{
|
||
/** @var string */
|
||
protected $entityType;
|
||
|
||
private bool $isNotNew = false;
|
||
private bool $isSaved = false;
|
||
private bool $isFetched = false;
|
||
private bool $isBeingSaved = false;
|
||
|
||
protected ?EntityManager $entityManager;
|
||
private ?ValueAccessor $valueAccessor = null;
|
||
|
||
/** @var array<string, bool> */
|
||
private array $writtenMap = [];
|
||
/** @var array<string, array<string, mixed>> */
|
||
private array $attributes = [];
|
||
/** @var array<string, array<string, mixed>> */
|
||
private array $relations = [];
|
||
/** @var array<string, mixed> */
|
||
private array $fetchedValuesContainer = [];
|
||
/** @var array<string, mixed> */
|
||
private array $valuesContainer = [];
|
||
|
||
/**
|
||
* @deprecated As of v7.0. Use `getId`. To be changed to protected.
|
||
* @todo Change to protected in v9.0.
|
||
* @var ?string
|
||
*/
|
||
public $id = null;
|
||
|
||
/**
|
||
* @param array{
|
||
* attributes?: array<string, array<string, mixed>>,
|
||
* relations?: array<string, array<string, mixed>>,
|
||
* fields?: array<string, array<string, mixed>>
|
||
* } $defs
|
||
*/
|
||
public function __construct(
|
||
string $entityType,
|
||
array $defs,
|
||
?EntityManager $entityManager = null,
|
||
?ValueAccessorFactory $valueAccessorFactory = null
|
||
) {
|
||
$this->entityType = $entityType;
|
||
$this->entityManager = $entityManager;
|
||
|
||
$this->attributes = $defs['attributes'] ?? $this->attributes;
|
||
$this->relations = $defs['relations'] ?? $this->relations;
|
||
|
||
if ($valueAccessorFactory) {
|
||
$this->valueAccessor = $valueAccessorFactory->create($this);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Get an entity ID.
|
||
*/
|
||
public function getId(): string
|
||
{
|
||
/** @var ?string $id */
|
||
$id = $this->get('id');
|
||
|
||
if ($id === null) {
|
||
throw new RuntimeException("Entity ID is not set.");
|
||
}
|
||
|
||
if ($id === '') {
|
||
throw new RuntimeException("Entity ID is empty.");
|
||
}
|
||
|
||
return $id;
|
||
}
|
||
|
||
public function hasId(): bool
|
||
{
|
||
return $this->id !== null;
|
||
}
|
||
|
||
/**
|
||
* Clear an attribute value.
|
||
*/
|
||
public function clear(string $attribute): void
|
||
{
|
||
unset($this->valuesContainer[$attribute]);
|
||
}
|
||
|
||
/**
|
||
* Reset all attributes (empty an entity).
|
||
*/
|
||
public function reset(): void
|
||
{
|
||
$this->valuesContainer = [];
|
||
}
|
||
|
||
/**
|
||
* Set an attribute value or multiple attribute values.
|
||
*
|
||
* Two usage options:
|
||
* * `set(string $attribute, mixed $value)`
|
||
* * `set(array|object $valueMap)`
|
||
*
|
||
* @param string|stdClass|array<string, mixed> $attribute
|
||
* @param mixed $value
|
||
*/
|
||
public function set($attribute, $value = null): void
|
||
{
|
||
$p1 = $attribute;
|
||
$p2 = $value;
|
||
|
||
/**
|
||
* @var mixed $p1
|
||
* @var mixed $p2
|
||
*/
|
||
|
||
if (is_array($p1) || is_object($p1)) {
|
||
if (is_object($p1)) {
|
||
$p1 = get_object_vars($p1);
|
||
}
|
||
|
||
if ($p2 === null) {
|
||
$p2 = false;
|
||
}
|
||
|
||
if ($p2) {
|
||
// @todo Remove second parameter support in v9.0.
|
||
trigger_error(
|
||
'Second parameter is deprecated in Entity::set(array, onlyAccessible).',
|
||
E_USER_DEPRECATED
|
||
);
|
||
}
|
||
|
||
$this->populateFromArray($p1, $p2);
|
||
|
||
return;
|
||
}
|
||
|
||
if (is_string($p1)) {
|
||
$name = $p1;
|
||
|
||
if ($name == 'id') {
|
||
$this->id = $value;
|
||
}
|
||
|
||
if (!$this->hasAttribute($name)) {
|
||
return;
|
||
}
|
||
|
||
$method = '_set' . ucfirst($name);
|
||
|
||
if (method_exists($this, $method)) {
|
||
$this->$method($value);
|
||
|
||
return;
|
||
}
|
||
|
||
$this->populateFromArray([
|
||
$name => $value,
|
||
]);
|
||
|
||
return;
|
||
}
|
||
|
||
throw new InvalidArgumentException();
|
||
}
|
||
|
||
/**
|
||
* Set multiple attributes.
|
||
*
|
||
* @param array<string, mixed>|stdClass $valueMap Values.
|
||
* @since v8.1.0.
|
||
*/
|
||
public function setMultiple(array|stdClass $valueMap): void
|
||
{
|
||
$this->set($valueMap);
|
||
}
|
||
|
||
/**
|
||
* Get an attribute value.
|
||
*
|
||
* @param array<string, mixed> $params @deprecated @todo Remove in v9.0.
|
||
* @retrun mixed
|
||
*/
|
||
public function get(string $attribute, $params = [])
|
||
{
|
||
if ($attribute === 'id') {
|
||
return $this->id;
|
||
}
|
||
|
||
// Legacy.
|
||
$method = '_get' . ucfirst($attribute);
|
||
|
||
if (method_exists($this, $method)) {
|
||
return $this->$method();
|
||
}
|
||
|
||
if ($this->hasAttribute($attribute) && $this->hasInContainer($attribute)) {
|
||
return $this->getFromContainer($attribute);
|
||
}
|
||
|
||
// @todo Remove support in v9.0.
|
||
if (!empty($params)) {
|
||
trigger_error(
|
||
'Second parameter will be removed from the method Entity::get.',
|
||
E_USER_DEPRECATED
|
||
);
|
||
}
|
||
|
||
// @todo Remove support in v10.0.
|
||
if ($this->hasRelation($attribute) && $this->id && $this->entityManager) {
|
||
trigger_error(
|
||
"Accessing related records with Entity::get is deprecated. " .
|
||
"Use \$repository->getRelation(...)->find()",
|
||
E_USER_DEPRECATED
|
||
);
|
||
|
||
/** @phpstan-ignore-next-line */
|
||
return $this->entityManager
|
||
->getRepository($this->getEntityType())
|
||
->findRelated($this, $attribute, $params);
|
||
}
|
||
|
||
return null;
|
||
}
|
||
|
||
/**
|
||
* Set a value in the container.
|
||
*
|
||
* @param mixed $value
|
||
*/
|
||
protected function setInContainer(string $attribute, $value): void
|
||
{
|
||
$this->valuesContainer[$attribute] = $value;
|
||
$this->writtenMap[$attribute] = true;
|
||
}
|
||
|
||
/**
|
||
* Whether an attribute is set in the container.
|
||
*/
|
||
protected function hasInContainer(string $attribute): bool
|
||
{
|
||
return array_key_exists($attribute, $this->valuesContainer);
|
||
}
|
||
|
||
/**
|
||
* Get a value from the container.
|
||
*
|
||
* @return mixed
|
||
* @todo Add return type in v9.0.
|
||
*/
|
||
protected function getFromContainer(string $attribute)
|
||
{
|
||
if (!$this->hasInContainer($attribute)) {
|
||
return null;
|
||
}
|
||
|
||
$value = $this->valuesContainer[$attribute] ?? null;
|
||
|
||
if ($value === null) {
|
||
return null;
|
||
}
|
||
|
||
$type = $this->getAttributeType($attribute);
|
||
|
||
if ($type === self::JSON_ARRAY) {
|
||
return $this->cloneArray($value);
|
||
}
|
||
|
||
if ($type === self::JSON_OBJECT) {
|
||
return $this->cloneObject($value);
|
||
}
|
||
|
||
return $value;
|
||
}
|
||
|
||
/**
|
||
* whether an attribute is set in the fetched-container.
|
||
*/
|
||
protected function hasInFetchedContainer(string $attribute): bool
|
||
{
|
||
return array_key_exists($attribute, $this->fetchedValuesContainer);
|
||
}
|
||
|
||
/**
|
||
* Get a value from the fetched-container.
|
||
*
|
||
* @return mixed
|
||
* @todo Add return type in v9.0.
|
||
*/
|
||
protected function getFromFetchedContainer(string $attribute)
|
||
{
|
||
if (!$this->hasInFetchedContainer($attribute)) {
|
||
return null;
|
||
}
|
||
|
||
$value = $this->fetchedValuesContainer[$attribute] ?? null;
|
||
|
||
if ($value === null) {
|
||
return null;
|
||
}
|
||
|
||
$type = $this->getAttributeType($attribute);
|
||
|
||
if ($type === self::JSON_ARRAY) {
|
||
return $this->cloneArray($value);
|
||
}
|
||
|
||
if ($type === self::JSON_OBJECT) {
|
||
return $this->cloneObject($value);
|
||
}
|
||
|
||
return $value;
|
||
}
|
||
|
||
/**
|
||
* Whether an attribute value is set.
|
||
*/
|
||
public function has(string $attribute): bool
|
||
{
|
||
if ($attribute == 'id') {
|
||
return (bool) $this->id;
|
||
}
|
||
|
||
// Legacy.
|
||
$method = '_has' . ucfirst($attribute);
|
||
|
||
if (method_exists($this, $method)) {
|
||
return (bool) $this->$method();
|
||
}
|
||
|
||
if (array_key_exists($attribute, $this->valuesContainer)) {
|
||
return true;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* Whether a value object for a field can be gotten.
|
||
*/
|
||
public function isValueObjectGettable(string $field): bool
|
||
{
|
||
if (!$this->valueAccessor) {
|
||
throw new RuntimeException("No ValueAccessor.");
|
||
}
|
||
|
||
return $this->valueAccessor->isGettable($field);
|
||
}
|
||
|
||
/**
|
||
* Get a value object for a field. NULL can be returned.
|
||
*/
|
||
public function getValueObject(string $field): ?object
|
||
{
|
||
if (!$this->valueAccessor) {
|
||
throw new RuntimeException("No ValueAccessor.");
|
||
}
|
||
|
||
return $this->valueAccessor->get($field);
|
||
}
|
||
|
||
/**
|
||
* Set a value object for a field. NULL can be set.
|
||
*
|
||
* @throws RuntimeException
|
||
*/
|
||
public function setValueObject(string $field, ?object $value): void
|
||
{
|
||
if (!$this->valueAccessor) {
|
||
throw new RuntimeException("No ValueAccessor.");
|
||
}
|
||
|
||
$this->valueAccessor->set($field, $value);
|
||
}
|
||
|
||
/**
|
||
* @todo Make private in v9.0.
|
||
*/
|
||
protected function populateFromArrayItem(string $attribute, mixed $value): void
|
||
{
|
||
$preparedValue = $this->prepareAttributeValue($attribute, $value);
|
||
|
||
// Legacy.
|
||
$method = '_set' . ucfirst($attribute);
|
||
|
||
if (method_exists($this, $method)) {
|
||
$this->$method($preparedValue);
|
||
|
||
return;
|
||
}
|
||
|
||
$this->setInContainer($attribute, $preparedValue);
|
||
}
|
||
|
||
protected function prepareAttributeValue(string $attribute, mixed $value): mixed
|
||
{
|
||
if (is_null($value)) {
|
||
return null;
|
||
}
|
||
|
||
$attributeType = $this->getAttributeType($attribute);
|
||
|
||
if ($attributeType === self::FOREIGN) {
|
||
$attributeType = $this->getForeignAttributeType($attribute) ?? $attributeType;
|
||
}
|
||
|
||
switch ($attributeType) {
|
||
case self::VARCHAR:
|
||
// @todo Convert to string if not null in v9.0.
|
||
return $value;
|
||
|
||
case self::BOOL:
|
||
return ($value === 1 || $value === '1' || $value === true || $value === 'true');
|
||
|
||
case self::INT:
|
||
return intval($value);
|
||
|
||
case self::FLOAT:
|
||
return floatval($value);
|
||
|
||
case self::JSON_ARRAY:
|
||
return $this->prepareArrayAttributeValue($value);
|
||
|
||
case self::JSON_OBJECT:
|
||
return $this->prepareObjectAttributeValue($value);
|
||
|
||
default:
|
||
break;
|
||
}
|
||
|
||
return $value;
|
||
}
|
||
|
||
/**
|
||
* @param mixed $value
|
||
* @return mixed[]|null
|
||
*/
|
||
private function prepareArrayAttributeValue($value): ?array
|
||
{
|
||
if (is_string($value)) {
|
||
$preparedValue = json_decode($value);
|
||
|
||
if (!is_array($preparedValue)) {
|
||
return null;
|
||
}
|
||
|
||
return $preparedValue;
|
||
}
|
||
|
||
if (!is_array($value)) {
|
||
return null;
|
||
}
|
||
|
||
return $this->cloneArray($value);
|
||
}
|
||
|
||
/**
|
||
* @param mixed $value
|
||
*/
|
||
private function prepareObjectAttributeValue($value): ?stdClass
|
||
{
|
||
if (is_string($value)) {
|
||
$preparedValue = json_decode($value);
|
||
|
||
if (!$preparedValue instanceof stdClass) {
|
||
return null;
|
||
}
|
||
|
||
return $preparedValue;
|
||
}
|
||
|
||
$preparedValue = $value;
|
||
|
||
if (is_array($value)) {
|
||
$preparedValue = json_decode(json_encode($value, JSON_THROW_ON_ERROR));
|
||
|
||
if ($preparedValue instanceof stdClass) {
|
||
return $preparedValue;
|
||
}
|
||
}
|
||
|
||
if (!$preparedValue instanceof stdClass) {
|
||
return null;
|
||
}
|
||
|
||
return $this->cloneObject($preparedValue);
|
||
}
|
||
|
||
private function getForeignAttributeType(string $attribute): ?string
|
||
{
|
||
if (!$this->entityManager) {
|
||
return null;
|
||
}
|
||
|
||
$defs = $this->entityManager->getDefs();
|
||
|
||
$entityDefs = $defs->getEntity($this->entityType);
|
||
|
||
// This should not be removed for compatibility reasons.
|
||
if (!$entityDefs->hasAttribute($attribute)) {
|
||
return null;
|
||
}
|
||
|
||
$relation = $entityDefs->getAttribute($attribute)->getParam('relation');
|
||
$foreign = $entityDefs->getAttribute($attribute)->getParam('foreign');
|
||
|
||
if (!$relation) {
|
||
return null;
|
||
}
|
||
|
||
if (!$foreign) {
|
||
return null;
|
||
}
|
||
|
||
if (!is_string($foreign)) {
|
||
return self::VARCHAR;
|
||
}
|
||
|
||
if (!$entityDefs->getRelation($relation)->hasForeignEntityType()) {
|
||
return null;
|
||
}
|
||
|
||
$entityType = $entityDefs->getRelation($relation)->getForeignEntityType();
|
||
|
||
if (!$defs->hasEntity($entityType)) {
|
||
return null;
|
||
}
|
||
|
||
$foreignEntityDefs = $defs->getEntity($entityType);
|
||
|
||
if (!$foreignEntityDefs->hasAttribute($foreign)) {
|
||
return null;
|
||
}
|
||
|
||
return $foreignEntityDefs->getAttribute($foreign)->getType();
|
||
}
|
||
|
||
/**
|
||
* Whether an entity is new.
|
||
*/
|
||
public function isNew(): bool
|
||
{
|
||
return !$this->isNotNew;
|
||
}
|
||
|
||
/**
|
||
* Set as not new. Meaning the entity is fetched or already saved.
|
||
*/
|
||
public function setAsNotNew(): void
|
||
{
|
||
$this->isNotNew = true;
|
||
}
|
||
|
||
/**
|
||
* Whether an entity has been saved. An entity can be already saved but not yet set as not-new.
|
||
* To prevent inserting second time if save is called in an after-save hook.
|
||
*/
|
||
public function isSaved(): bool
|
||
{
|
||
return $this->isSaved;
|
||
}
|
||
|
||
/**
|
||
* Set as saved.
|
||
*/
|
||
public function setAsSaved(): void
|
||
{
|
||
$this->isSaved = true;
|
||
}
|
||
|
||
/**
|
||
* Get an entity type.
|
||
*/
|
||
public final function getEntityType(): string
|
||
{
|
||
return $this->entityType;
|
||
}
|
||
|
||
/**
|
||
* @deprecated As of v6.0. Use `hasAttribute`.
|
||
* @param string $name
|
||
* @return bool
|
||
*/
|
||
public function hasField($name)
|
||
{
|
||
return $this->hasAttribute($name);
|
||
}
|
||
|
||
/**
|
||
* Whether an entity type has an attribute defined.
|
||
*/
|
||
public function hasAttribute(string $attribute): bool
|
||
{
|
||
return isset($this->attributes[$attribute]);
|
||
}
|
||
|
||
/**
|
||
* Whether an entity type has a relation defined.
|
||
*/
|
||
public function hasRelation(string $relation): bool
|
||
{
|
||
return isset($this->relations[$relation]);
|
||
}
|
||
|
||
/**
|
||
* Get attribute list defined for an entity type.
|
||
*/
|
||
public function getAttributeList(): array
|
||
{
|
||
return array_keys($this->attributes);
|
||
}
|
||
|
||
/**
|
||
* Get relation list defined for an entity type.
|
||
*/
|
||
public function getRelationList(): array
|
||
{
|
||
return array_keys($this->relations);
|
||
}
|
||
|
||
/**
|
||
* @deprecated As of v6.0. Use `getValueMap`.
|
||
* @todo Remove in v9.0.
|
||
* @return array<string, mixed>
|
||
*/
|
||
public function toArray()
|
||
{
|
||
$arr = [];
|
||
|
||
if (isset($this->id)) {
|
||
$arr['id'] = $this->id;
|
||
}
|
||
|
||
foreach ($this->getAttributeList() as $attribute) {
|
||
if ($attribute === 'id') {
|
||
continue;
|
||
}
|
||
|
||
if ($this->has($attribute)) {
|
||
$arr[$attribute] = $this->get($attribute);
|
||
}
|
||
}
|
||
|
||
return $arr;
|
||
}
|
||
|
||
/**
|
||
* Get values.
|
||
*/
|
||
public function getValueMap(): stdClass
|
||
{
|
||
$array = $this->toArray();
|
||
|
||
return (object) $array;
|
||
}
|
||
|
||
/**
|
||
* Get an attribute type.
|
||
*/
|
||
public function getAttributeType(string $attribute): ?string
|
||
{
|
||
if (!isset($this->attributes[$attribute])) {
|
||
return null;
|
||
}
|
||
|
||
return $this->attributes[$attribute]['type'] ?? null;
|
||
}
|
||
|
||
/**
|
||
* Get a relation type.
|
||
*/
|
||
public function getRelationType(string $relation): ?string
|
||
{
|
||
if (!isset($this->relations[$relation])) {
|
||
return null;
|
||
}
|
||
|
||
return $this->relations[$relation]['type'] ?? null;
|
||
}
|
||
|
||
/**
|
||
* Get an attribute parameter.
|
||
*
|
||
* @return mixed
|
||
*/
|
||
public function getAttributeParam(string $attribute, string $name)
|
||
{
|
||
if (!isset($this->attributes[$attribute])) {
|
||
return null;
|
||
}
|
||
|
||
return $this->attributes[$attribute][$name] ?? null;
|
||
}
|
||
|
||
/**
|
||
* Get a relation parameter.
|
||
*
|
||
* @return mixed
|
||
*/
|
||
public function getRelationParam(string $relation, string $name)
|
||
{
|
||
if (!isset($this->relations[$relation])) {
|
||
return null;
|
||
}
|
||
|
||
return $this->relations[$relation][$name] ?? null;
|
||
}
|
||
|
||
/**
|
||
* Whether is fetched from DB.
|
||
*/
|
||
public function isFetched(): bool
|
||
{
|
||
return $this->isFetched;
|
||
}
|
||
|
||
/**
|
||
* @deprecated As of v6.0. Use `isAttributeChanged`.
|
||
* @param string $name
|
||
* @return bool
|
||
*/
|
||
public function isFieldChanged($name)
|
||
{
|
||
return $this->has($name) && ($this->get($name) != $this->getFetched($name));
|
||
}
|
||
|
||
/**
|
||
* Whether an attribute was changed (since syncing with DB).
|
||
*/
|
||
public function isAttributeChanged(string $name): bool
|
||
{
|
||
if (!$this->has($name)) {
|
||
return false;
|
||
}
|
||
|
||
if (!$this->hasFetched($name)) {
|
||
return true;
|
||
}
|
||
|
||
/** @var string $type */
|
||
$type = $this->getAttributeType($name);
|
||
|
||
return !self::areValuesEqual(
|
||
$type,
|
||
$this->get($name),
|
||
$this->getFetched($name),
|
||
$this->getAttributeParam($name, 'isUnordered') ?? false
|
||
);
|
||
}
|
||
|
||
/**
|
||
* Whether an attribute was written (since syncing with DB) regardless being changed.
|
||
*/
|
||
public function isAttributeWritten(string $name): bool
|
||
{
|
||
return $this->writtenMap[$name] ?? false;
|
||
}
|
||
|
||
/**
|
||
* @param mixed $v1
|
||
* @param mixed $v2
|
||
*/
|
||
protected static function areValuesEqual(string $type, $v1, $v2, bool $isUnordered = false): bool
|
||
{
|
||
if ($type === self::JSON_ARRAY) {
|
||
if (is_array($v1) && is_array($v2)) {
|
||
if ($isUnordered) {
|
||
sort($v1);
|
||
sort($v2);
|
||
}
|
||
|
||
if ($v1 != $v2) {
|
||
return false;
|
||
}
|
||
|
||
foreach ($v1 as $i => $itemValue) {
|
||
if (is_object($itemValue) && is_object($v2[$i])) {
|
||
if (!self::areValuesEqual(self::JSON_OBJECT, $itemValue, $v2[$i])) {
|
||
return false;
|
||
}
|
||
|
||
continue;
|
||
}
|
||
|
||
if ($itemValue !== $v2[$i]) {
|
||
return false;
|
||
}
|
||
}
|
||
|
||
return true;
|
||
}
|
||
}
|
||
else if ($type === self::JSON_OBJECT) {
|
||
if (is_object($v1) && is_object($v2)) {
|
||
if ($v1 != $v2) {
|
||
return false;
|
||
}
|
||
|
||
$a1 = get_object_vars($v1);
|
||
$a2 = get_object_vars($v2);
|
||
|
||
foreach (get_object_vars($v1) as $key => $itemValue) {
|
||
if (is_object($a1[$key]) && is_object($a2[$key])) {
|
||
if (!self::areValuesEqual(self::JSON_OBJECT, $a1[$key], $a2[$key])) {
|
||
return false;
|
||
}
|
||
|
||
continue;
|
||
}
|
||
|
||
if (is_array($a1[$key]) && is_array($a2[$key])) {
|
||
if (!self::areValuesEqual(self::JSON_ARRAY, $a1[$key], $a2[$key])) {
|
||
return false;
|
||
}
|
||
|
||
continue;
|
||
}
|
||
|
||
if ($a1[$key] !== $a2[$key]) {
|
||
return false;
|
||
}
|
||
}
|
||
|
||
return true;
|
||
}
|
||
}
|
||
|
||
return $v1 === $v2;
|
||
}
|
||
|
||
/**
|
||
* Set a fetched value for a specific attribute.
|
||
*/
|
||
public function setFetched(string $attribute, $value): void
|
||
{
|
||
$preparedValue = $this->prepareAttributeValue($attribute, $value);
|
||
|
||
$this->fetchedValuesContainer[$attribute] = $preparedValue;
|
||
}
|
||
|
||
/**
|
||
* Get a fetched value of a specific attribute.
|
||
*
|
||
* @return mixed
|
||
*/
|
||
public function getFetched(string $attribute)
|
||
{
|
||
if ($attribute === 'id') {
|
||
return $this->id;
|
||
}
|
||
|
||
if ($this->hasInFetchedContainer($attribute)) {
|
||
return $this->getFromFetchedContainer($attribute);
|
||
}
|
||
|
||
return null;
|
||
}
|
||
|
||
/**
|
||
* Whether a fetched value is set for a specific attribute.
|
||
*/
|
||
public function hasFetched(string $attribute): bool
|
||
{
|
||
if ($attribute === 'id') {
|
||
return !is_null($this->id);
|
||
}
|
||
|
||
return $this->hasInFetchedContainer($attribute);
|
||
}
|
||
|
||
/**
|
||
* Clear all set fetched values.
|
||
*/
|
||
public function resetFetchedValues(): void
|
||
{
|
||
$this->fetchedValuesContainer = [];
|
||
}
|
||
|
||
/**
|
||
* Copy all current values to fetched values. All current attribute values will beset as those
|
||
* that are fetched from DB.
|
||
*/
|
||
public function updateFetchedValues(): void
|
||
{
|
||
$this->fetchedValuesContainer = $this->valuesContainer;
|
||
|
||
foreach ($this->fetchedValuesContainer as $attribute => $value) {
|
||
$this->setFetched($attribute, $value);
|
||
}
|
||
|
||
$this->writtenMap = [];
|
||
}
|
||
|
||
/**
|
||
* Set an entity as fetched. All current attribute values will be set as those that are fetched
|
||
* from DB.
|
||
*/
|
||
public function setAsFetched(): void
|
||
{
|
||
$this->isFetched = true;
|
||
|
||
$this->setAsNotNew();
|
||
|
||
$this->updateFetchedValues();
|
||
}
|
||
|
||
/**
|
||
* Whether an entity is being saved.
|
||
*/
|
||
public function isBeingSaved(): bool
|
||
{
|
||
return $this->isBeingSaved;
|
||
}
|
||
|
||
public function setAsBeingSaved(): void
|
||
{
|
||
$this->isBeingSaved = true;
|
||
}
|
||
|
||
public function setAsNotBeingSaved(): void
|
||
{
|
||
$this->isBeingSaved = false;
|
||
}
|
||
|
||
/**
|
||
* Set defined default values.
|
||
*/
|
||
public function populateDefaults(): void
|
||
{
|
||
foreach ($this->attributes as $attribute => $defs) {
|
||
if (!array_key_exists('default', $defs)) {
|
||
continue;
|
||
}
|
||
|
||
$wasSet = $this->hasInContainer($attribute);
|
||
|
||
$this->setInContainer($attribute, $defs['default']);
|
||
|
||
$this->writtenMap[$attribute] = $wasSet;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Clone an array value.
|
||
*
|
||
* @param mixed[]|null $value
|
||
* @return mixed[]
|
||
*/
|
||
protected function cloneArray(?array $value): ?array
|
||
{
|
||
if ($value === null) {
|
||
return null;
|
||
}
|
||
|
||
$toClone = false;
|
||
|
||
foreach ($value as $item) {
|
||
if (is_object($item) || is_array($item)) {
|
||
$toClone = true;
|
||
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (!$toClone) {
|
||
return $value;
|
||
}
|
||
|
||
$copy = [];
|
||
|
||
/** @var array<int, stdClass|mixed[]|scalar|null> $value */
|
||
|
||
foreach ($value as $i => $item) {
|
||
if (is_object($item)) {
|
||
$copy[$i] = $this->cloneObject($item);
|
||
|
||
continue;
|
||
}
|
||
|
||
if (is_array($item)) {
|
||
if (!array_is_list($item)) {
|
||
$copy[$i] = $this->cloneObject((object) $item);
|
||
|
||
continue;
|
||
}
|
||
|
||
$copy[$i] = $this->cloneArray($item);
|
||
|
||
continue;
|
||
}
|
||
|
||
$copy[$i] = $item;
|
||
}
|
||
|
||
return $copy;
|
||
}
|
||
|
||
/**
|
||
* Clone an object value.
|
||
*/
|
||
protected function cloneObject(?stdClass $value): ?stdClass
|
||
{
|
||
if ($value === null) {
|
||
return null;
|
||
}
|
||
|
||
$copy = (object) [];
|
||
|
||
foreach (get_object_vars($value) as $k => $item) {
|
||
/** @var stdClass|mixed[]|scalar|null $item */
|
||
|
||
$key = $k;
|
||
|
||
if (!is_string($key)) {
|
||
$key = strval($key);
|
||
}
|
||
|
||
if (is_object($item)) {
|
||
$copy->$key = $this->cloneObject($item);
|
||
|
||
continue;
|
||
}
|
||
|
||
if (is_array($item)) {
|
||
$copy->$key = $this->cloneArray($item);
|
||
|
||
continue;
|
||
}
|
||
|
||
$copy->$key = $item;
|
||
}
|
||
|
||
return $copy;
|
||
}
|
||
|
||
/**
|
||
* @deprecated As of v7.0. Use `set` method instead.
|
||
* @todo Make protected in v9.0.
|
||
* @param array<string, mixed> $data
|
||
*/
|
||
public function populateFromArray(array $data, bool $onlyAccessible = true, bool $reset = false): void
|
||
{
|
||
if ($reset) {
|
||
$this->reset();
|
||
}
|
||
|
||
foreach ($this->getAttributeList() as $attribute) {
|
||
if (!array_key_exists($attribute, $data)) {
|
||
continue;
|
||
}
|
||
|
||
if ($attribute == 'id') {
|
||
$this->id = $data[$attribute];
|
||
|
||
continue;
|
||
}
|
||
|
||
if ($onlyAccessible && $this->getAttributeParam($attribute, 'notAccessible')) {
|
||
continue;
|
||
}
|
||
|
||
$value = $data[$attribute];
|
||
|
||
$this->populateFromArrayItem($attribute, $value);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @deprecated As of v7.0. Use `setInContainer` method.
|
||
* @todo Remove in v9.0.
|
||
*
|
||
* @param string $attribute
|
||
* @param mixed $value
|
||
*/
|
||
protected function setValue($attribute, $value): void
|
||
{
|
||
$this->setInContainer($attribute, $value);
|
||
}
|
||
}
|