| Current Path : /var/www/surf/TYPO3/vendor/typo3/cms-extbase/Classes/Persistence/Generic/Mapper/ |
| Current File : /var/www/surf/TYPO3/vendor/typo3/cms-extbase/Classes/Persistence/Generic/Mapper/DataMapper.php |
<?php
declare(strict_types=1);
/*
* This file is part of the TYPO3 CMS project.
*
* It is free software; you can redistribute it and/or modify it under
* the terms of the GNU General Public License, either version 2
* of the License, or any later version.
*
* For the full copyright and license information, please read the
* LICENSE.txt file that was distributed with this source code.
*
* The TYPO3 project - inspiring people to share!
*/
namespace TYPO3\CMS\Extbase\Persistence\Generic\Mapper;
use Doctrine\Instantiator\InstantiatorInterface;
use Psr\EventDispatcher\EventDispatcherInterface;
use Symfony\Component\PropertyInfo\Type;
use TYPO3\CMS\Core\Context\Context;
use TYPO3\CMS\Core\Context\LanguageAspect;
use TYPO3\CMS\Core\Database\Query\QueryHelper;
use TYPO3\CMS\Core\Database\RelationHandler;
use TYPO3\CMS\Core\Utility\GeneralUtility;
use TYPO3\CMS\Extbase\DomainObject\AbstractDomainObject;
use TYPO3\CMS\Extbase\DomainObject\DomainObjectInterface;
use TYPO3\CMS\Extbase\Event\Persistence\AfterObjectThawedEvent;
use TYPO3\CMS\Extbase\Persistence;
use TYPO3\CMS\Extbase\Persistence\Generic\Exception;
use TYPO3\CMS\Extbase\Persistence\Generic\Exception\InvalidClassException;
use TYPO3\CMS\Extbase\Persistence\Generic\Exception\UnexpectedTypeException;
use TYPO3\CMS\Extbase\Persistence\Generic\LazyLoadingProxy;
use TYPO3\CMS\Extbase\Persistence\Generic\LazyObjectStorage;
use TYPO3\CMS\Extbase\Persistence\Generic\LoadingStrategyInterface;
use TYPO3\CMS\Extbase\Persistence\Generic\Mapper\ColumnMap\Relation;
use TYPO3\CMS\Extbase\Persistence\Generic\Mapper\Exception\NonExistentPropertyException;
use TYPO3\CMS\Extbase\Persistence\Generic\Mapper\Exception\UnknownPropertyTypeException;
use TYPO3\CMS\Extbase\Persistence\Generic\Qom\QueryObjectModelFactory;
use TYPO3\CMS\Extbase\Persistence\Generic\Query;
use TYPO3\CMS\Extbase\Persistence\Generic\QueryFactoryInterface;
use TYPO3\CMS\Extbase\Persistence\Generic\Session;
use TYPO3\CMS\Extbase\Persistence\ObjectStorage;
use TYPO3\CMS\Extbase\Persistence\QueryInterface;
use TYPO3\CMS\Extbase\Persistence\QueryResultInterface;
use TYPO3\CMS\Extbase\Reflection\ClassSchema\Exception\NoPropertyTypesException;
use TYPO3\CMS\Extbase\Reflection\ClassSchema\Exception\NoSuchPropertyException;
use TYPO3\CMS\Extbase\Reflection\ReflectionService;
use TYPO3\CMS\Extbase\Utility\TypeHandlingUtility;
/**
* A mapper to map database tables configured in $TCA on domain objects.
* @internal only to be used within Extbase, not part of TYPO3 Core API.
*/
class DataMapper
{
/**
* @var QueryInterface|null
*/
protected $query;
public function __construct(
private readonly ReflectionService $reflectionService,
private readonly QueryObjectModelFactory $qomFactory,
private readonly Session $persistenceSession,
private readonly DataMapFactory $dataMapFactory,
private readonly QueryFactoryInterface $queryFactory,
private readonly EventDispatcherInterface $eventDispatcher,
private readonly InstantiatorInterface $instantiator,
) {}
public function setQuery(QueryInterface $query): void
{
$this->query = $query;
}
/**
* Maps the given rows on objects
*
* @param string $className The name of the class
* @param array $rows An array of arrays with field_name => value pairs
* @return array An array of objects of the given class
* @template T of DomainObjectInterface
* @phpstan-param class-string<T> $className
* @phpstan-return list<T>
*/
public function map($className, array $rows)
{
$objects = [];
foreach ($rows as $row) {
$objects[] = $this->mapSingleRow($this->getTargetType($className, $row), $row);
}
return $objects;
}
/**
* Returns the target type for the given row.
*
* @param string $className The name of the class
* @param array $row A single array with field_name => value pairs
* @return string The target type (a class name)
* @phpstan-param class-string $className
* @phpstan-return class-string
*/
public function getTargetType($className, array $row)
{
$dataMap = $this->getDataMap($className);
$targetType = $className;
if ($dataMap->getRecordTypeColumnName() !== null) {
foreach ($dataMap->getSubclasses() as $subclassName) {
$recordSubtype = $this->getDataMap($subclassName)->getRecordType();
if ((string)$row[$dataMap->getRecordTypeColumnName()] === (string)$recordSubtype) {
$targetType = $subclassName;
break;
}
}
}
return $targetType;
}
/**
* Maps a single row on an object of the given class
*
* @param string $className The name of the target class
* @param array $row A single array with field_name => value pairs
* @return object An object of the given class
* @template T of DomainObjectInterface
* @phpstan-param class-string<T> $className
* @phpstan-return T
*/
protected function mapSingleRow($className, array $row)
{
// @todo: this also needs to contain the query's languageAspect with its configuration
// which should be changed along with https://review.typo3.org/c/Packages/TYPO3.CMS/+/75093
$identifier = $row['uid'] . (isset($row['_LOCALIZED_UID']) ? '_' . $row['_LOCALIZED_UID'] : '');
if ($this->persistenceSession->hasIdentifier($identifier, $className)) {
$object = $this->persistenceSession->getObjectByIdentifier($identifier, $className);
} else {
$object = $this->createEmptyObject($className);
$this->persistenceSession->registerObject($object, $identifier);
$this->thawProperties($object, $row);
$event = new AfterObjectThawedEvent($object, $row);
$this->eventDispatcher->dispatch($event);
$object->_memorizeCleanState();
$this->persistenceSession->registerReconstitutedEntity($object);
}
return $object;
}
/**
* Creates a skeleton of the specified object. This is
* designed to *not* call class constructor when hydrating,
* but *do call* initializeObject() if exists and obey
* eventually registered implementation overrides ("xclass").
*
* @param class-string $className Name of the class to create a skeleton for
* @throws InvalidClassException
* @template T of DomainObjectInterface
* @phpstan-param class-string<T> $className
* @phpstan-return T
*/
protected function createEmptyObject(string $className): DomainObjectInterface
{
// Note: The class_implements() function also invokes autoload to assure that the interfaces
// and the class are loaded. Would end up with __PHP_Incomplete_Class without it.
if (!in_array(DomainObjectInterface::class, class_implements($className) ?: [])) {
throw new InvalidClassException('Cannot create empty instance of the class "' . $className
. '" because it does not implement the TYPO3\\CMS\\Extbase\\DomainObject\\DomainObjectInterface.', 1234386924);
}
// Use GU::getClassName() to obey class implementation overrides.
$object = $this->instantiator->instantiate(GeneralUtility::getClassName($className));
if (is_callable($callable = [$object, 'initializeObject'])) {
$callable();
}
return $object;
}
/**
* Sets the given properties on the object.
*
* @param DomainObjectInterface $object The object to set properties on
* @throws NonExistentPropertyException
* @throws UnknownPropertyTypeException
*/
protected function thawProperties(DomainObjectInterface $object, array $row)
{
$className = get_class($object);
$classSchema = $this->reflectionService->getClassSchema($className);
$dataMap = $this->getDataMap($className);
$object->_setProperty(AbstractDomainObject::PROPERTY_UID, (int)$row['uid']);
$object->_setProperty(AbstractDomainObject::PROPERTY_PID, (int)($row['pid'] ?? 0));
$object->_setProperty(AbstractDomainObject::PROPERTY_LOCALIZED_UID, (int)$row['uid']);
$object->_setProperty(AbstractDomainObject::PROPERTY_VERSIONED_UID, (int)$row['uid']);
if ($dataMap->getLanguageIdColumnName() !== null) {
$object->_setProperty(AbstractDomainObject::PROPERTY_LANGUAGE_UID, (int)($row[$dataMap->getLanguageIdColumnName()] ?? 0));
if (isset($row['_LOCALIZED_UID'])) {
$object->_setProperty(AbstractDomainObject::PROPERTY_LOCALIZED_UID, (int)$row['_LOCALIZED_UID']);
}
}
if (!empty($row['_ORIG_uid']) && !empty($GLOBALS['TCA'][$dataMap->getTableName()]['ctrl']['versioningWS'])) {
$object->_setProperty(AbstractDomainObject::PROPERTY_VERSIONED_UID, (int)$row['_ORIG_uid']);
}
foreach ($classSchema->getDomainObjectProperties() as $property) {
$propertyName = $property->getName();
if (!$dataMap->isPersistableProperty($propertyName)) {
continue;
}
$columnMap = $dataMap->getColumnMap($propertyName);
$columnName = $columnMap->getColumnName();
if (!isset($row[$columnName])) {
continue;
}
$nonProxyPropertyTypes = $property->getFilteredTypes([$property, 'filterLazyLoadingProxyAndLazyObjectStorage']);
if ($nonProxyPropertyTypes === []) {
throw new UnknownPropertyTypeException(
'The type of property ' . $className . '::' . $propertyName . ' could not be identified, therefore the desired value (' .
var_export($row[$columnName], true) . ') cannot be mapped onto it. The type of a class property is usually defined via property types or php doc blocks. ' .
'Make sure the property has a property type or valid @var tag set which defines the type.',
1579965021
);
}
if (count($nonProxyPropertyTypes) > 1) {
throw new UnknownPropertyTypeException(
'The type of property ' . $className . '::' . $propertyName . ' could not be identified because the property is defined as union or intersection type, therefore the desired value (' .
var_export($row[$columnName], true) . ') cannot be mapped onto it. Make sure to use only a single type.',
1660215701
);
}
$propertyType = $nonProxyPropertyTypes[0]->getClassName() ?? $nonProxyPropertyTypes[0]->getBuiltinType();
$propertyValue = null;
switch ($propertyType) {
case 'int':
case 'integer':
$propertyValue = (int)$row[$columnName];
break;
case 'float':
$propertyValue = (float)$row[$columnName];
break;
case 'bool':
case 'boolean':
$propertyValue = (bool)$row[$columnName];
break;
case 'string':
$propertyValue = (string)$row[$columnName];
break;
case 'array':
// $propertyValue = $this->mapArray($row[$columnName]); // Not supported, yet!
break;
case \SplObjectStorage::class:
case ObjectStorage::class:
$propertyValue = $this->mapResultToPropertyValue(
$object,
$propertyName,
$this->fetchRelated($object, $propertyName, $row[$columnName])
);
break;
default:
if (is_subclass_of($propertyType, \DateTimeInterface::class)) {
$propertyValue = $this->mapDateTime(
$row[$columnName],
$columnMap->getDateTimeStorageFormat(),
$propertyType
);
} elseif (TypeHandlingUtility::isCoreType($propertyType)) {
$propertyValue = $this->mapCoreType($propertyType, $row[$columnName]);
} else {
$propertyValue = $this->mapObjectToClassProperty(
$object,
$propertyName,
$row[$columnName]
);
}
}
if ($propertyValue !== null || $property->isNullable()) {
$object->_setProperty($propertyName, $propertyValue);
}
}
}
/**
* Map value to a core type
*
* @param string $type
* @param mixed $value
* @return \TYPO3\CMS\Core\Type\TypeInterface
*/
protected function mapCoreType($type, $value)
{
return new $type($value);
}
/**
* Creates a DateTime from a unix timestamp or date/datetime/time value.
* If the input is empty, NULL is returned.
*
* @param int|string $value Unix timestamp or date/datetime/time value
* @param string|null $storageFormat Storage format for native date/datetime/time fields
* @param string $targetType The object class name to be created
* @return \DateTimeInterface
*/
protected function mapDateTime($value, $storageFormat = null, $targetType = \DateTime::class)
{
$dateTimeTypes = QueryHelper::getDateTimeTypes();
// Invalid values are converted to NULL
if (empty($value) || $value === '0000-00-00' || $value === '0000-00-00 00:00:00' || $value === '00:00:00') {
return null;
}
if (!in_array($storageFormat, $dateTimeTypes, true)) {
// Integer timestamps are also stored "as is" in the database, but are UTC by definition,
// so we convert the timestamp to an ISO representation.
$value = date('c', (int)$value);
}
// All date/datetime/time values are stored in the database "as is", independent of any time zone information.
// It is therefore only important to use the same time zone in PHP when storing and retrieving the values.
return GeneralUtility::makeInstance($targetType, $value);
}
/**
* Fetches a collection of objects related to a property of a parent object
*
* @param DomainObjectInterface $parentObject The object instance this proxy is part of
* @param string $propertyName The name of the proxied property in it's parent
* @param mixed $fieldValue The raw field value.
* @param bool $enableLazyLoading A flag indication if the related objects should be lazy loaded
* @return \TYPO3\CMS\Extbase\Persistence\Generic\LazyObjectStorage|Persistence\QueryResultInterface The result
*/
public function fetchRelated(DomainObjectInterface $parentObject, $propertyName, $fieldValue = '', $enableLazyLoading = true)
{
$property = $this->reflectionService->getClassSchema(get_class($parentObject))->getProperty($propertyName);
if ($enableLazyLoading && $property->isLazy()) {
if ($property->isObjectStorageType()) {
$result = GeneralUtility::makeInstance(LazyObjectStorage::class, $parentObject, $propertyName, $fieldValue, $this);
} elseif (empty($fieldValue)) {
$result = null;
} else {
$result = GeneralUtility::makeInstance(LazyLoadingProxy::class, $parentObject, $propertyName, $fieldValue, $this);
}
} else {
$result = $this->fetchRelatedEager($parentObject, $propertyName, $fieldValue);
}
return $result;
}
/**
* Fetches the related objects from the storage backend.
*
* @param DomainObjectInterface $parentObject The object instance this proxy is part of
* @param string $propertyName The name of the proxied property in it's parent
* @param mixed $fieldValue The raw field value.
* @return mixed
*/
protected function fetchRelatedEager(DomainObjectInterface $parentObject, $propertyName, $fieldValue = '')
{
return $fieldValue === '' ? $this->getEmptyRelationValue($parentObject, $propertyName) : $this->getNonEmptyRelationValue($parentObject, $propertyName, $fieldValue);
}
/**
* @param string $propertyName
* @return array|null
*/
protected function getEmptyRelationValue(DomainObjectInterface $parentObject, $propertyName)
{
$columnMap = $this->getDataMap(get_class($parentObject))->getColumnMap($propertyName);
$relatesToOne = $columnMap->getTypeOfRelation() == Relation::HAS_ONE;
return $relatesToOne ? null : [];
}
/**
* @param string $propertyName
* @param string $fieldValue
* @return Persistence\QueryResultInterface
*/
protected function getNonEmptyRelationValue(DomainObjectInterface $parentObject, $propertyName, $fieldValue)
{
$query = $this->getPreparedQuery($parentObject, $propertyName, $fieldValue);
return $query->execute();
}
/**
* Builds and returns the prepared query, ready to be executed.
*
* @param string $propertyName
* @param string $fieldValue
* @return Persistence\QueryInterface
*/
protected function getPreparedQuery(DomainObjectInterface $parentObject, $propertyName, $fieldValue = '')
{
$dataMap = $this->getDataMap(get_class($parentObject));
$columnMap = $dataMap->getColumnMap($propertyName);
$type = $this->getType(get_class($parentObject), $propertyName);
$query = $this->queryFactory->create($type);
if ($this->query && $query instanceof Query) {
$query->setParentQuery($this->query);
}
$query->getQuerySettings()->setRespectStoragePage(false);
$query->getQuerySettings()->setRespectSysLanguage(false);
$languageAspect = $query->getQuerySettings()->getLanguageAspect();
$languageUid = $languageAspect->getContentId();
if ($this->query) {
$languageAspect = $this->query->getQuerySettings()->getLanguageAspect();
$languageUid = $languageAspect->getContentId();
if ($dataMap->getLanguageIdColumnName() !== null && !$this->query->getQuerySettings()->getRespectSysLanguage()) {
//pass language of parent record to child objects, so they can be overlaid correctly in case
//e.g. findByUid is used.
//the languageUid is used for getRecordOverlay later on, despite RespectSysLanguage being false
$parentLanguageUid = (int)$parentObject->_getProperty(AbstractDomainObject::PROPERTY_LANGUAGE_UID);
// do not override the language when the parent language uid is set to all languages (-1)
if ($parentLanguageUid !== -1) {
$languageUid = $parentLanguageUid;
}
}
}
// we always want to overlay relations as most of the time they are stored in db using default language uids
$languageAspect = new LanguageAspect(
$languageUid,
$languageUid,
$languageAspect->getOverlayType() === LanguageAspect::OVERLAYS_OFF ? LanguageAspect::OVERLAYS_MIXED : $languageAspect->getOverlayType()
);
$query->getQuerySettings()->setLanguageAspect($languageAspect);
if ($columnMap->getTypeOfRelation() === Relation::HAS_MANY) {
if (null !== $orderings = $this->getOrderingsForColumnMap($columnMap)) {
$query->setOrderings($orderings);
}
} elseif ($columnMap->getTypeOfRelation() === Relation::HAS_AND_BELONGS_TO_MANY) {
$query->setSource($this->getSource($parentObject, $propertyName));
if ($columnMap->getChildSortByFieldName() !== null) {
$query->setOrderings([$columnMap->getChildSortByFieldName() => QueryInterface::ORDER_ASCENDING]);
}
}
$query->matching($this->getConstraint($query, $parentObject, $propertyName, $fieldValue, (array)$columnMap->getRelationTableMatchFields()));
return $query;
}
/**
* Get orderings array for extbase query by columnMap
*
* @phpstan-return array<non-empty-string, QueryInterface::ORDER_*>|null
* @return array<string, string>|null
*/
public function getOrderingsForColumnMap(ColumnMap $columnMap): array|null
{
if ($columnMap->getChildSortByFieldName() !== null) {
return [$columnMap->getChildSortByFieldName() => QueryInterface::ORDER_ASCENDING];
}
if ($columnMap->getChildTableDefaultSortings() === null) {
return null;
}
$orderings = [];
$fields = QueryHelper::parseOrderBy($columnMap->getChildTableDefaultSortings());
foreach ($fields as $field) {
$fieldName = $field[0] ?? null;
if ($fieldName === null) {
continue;
}
if (($fieldOrdering = $field[1] ?? null) === null) {
$orderings[$fieldName] = QueryInterface::ORDER_ASCENDING;
continue;
}
$fieldOrdering = strtoupper($fieldOrdering);
if (!in_array($fieldOrdering, [QueryInterface::ORDER_ASCENDING, QueryInterface::ORDER_DESCENDING], true)) {
$orderings[$fieldName] = QueryInterface::ORDER_ASCENDING;
continue;
}
$orderings[$fieldName] = $fieldOrdering;
}
return $orderings !== [] ? $orderings : null;
}
/**
* Builds and returns the constraint for multi value properties.
*
* @param string $propertyName
* @param string $fieldValue
* @param array $relationTableMatchFields
* @return \TYPO3\CMS\Extbase\Persistence\Generic\Qom\ConstraintInterface $constraint
*/
protected function getConstraint(QueryInterface $query, DomainObjectInterface $parentObject, $propertyName, $fieldValue = '', $relationTableMatchFields = [])
{
$dataMap = $this->getDataMap(get_class($parentObject));
$columnMap = $dataMap->getColumnMap($propertyName);
$workspaceId = GeneralUtility::makeInstance(Context::class)->getPropertyFromAspect('workspace', 'id');
if ($columnMap && $workspaceId > 0) {
$resolvedRelationIds = $this->resolveRelationValuesOfField($dataMap, $columnMap, $parentObject, $fieldValue, $workspaceId);
} else {
$resolvedRelationIds = [];
}
// Work with the UIDs directly in a workspace
if (!empty($resolvedRelationIds)) {
if ($query->getSource() instanceof Persistence\Generic\Qom\JoinInterface) {
$constraint = $query->in($query->getSource()->getJoinCondition()->getProperty1Name(), $resolvedRelationIds);
// When querying MM relations directly, Typo3DbQueryParser uses enableFields and thus, filters
// out versioned records by default. However, we directly query versioned UIDs here, so we want
// to include the versioned records explicitly.
if ($columnMap->getTypeOfRelation() === Relation::HAS_AND_BELONGS_TO_MANY) {
$query->getQuerySettings()->setEnableFieldsToBeIgnored(['pid']);
$query->getQuerySettings()->setIgnoreEnableFields(true);
}
} else {
$constraint = $query->in('uid', $resolvedRelationIds);
}
if ($columnMap->getParentTableFieldName() !== null) {
$constraint = $query->logicalAnd(
$constraint,
$query->equals($columnMap->getParentTableFieldName(), $dataMap->getTableName())
);
}
} elseif ($columnMap->getParentKeyFieldName() !== null) {
$value = $parentObject;
// If this a MM relation, and MM relations do not know about workspaces, the MM relations always point to the
// versioned record, so this must be taken into account here and the versioned record's UID must be used.
if ($columnMap->getTypeOfRelation() === Relation::HAS_AND_BELONGS_TO_MANY) {
// The versioned UID is used ideally the version ID of a translated record, so this takes precedence over the localized UID
if ($value->_hasProperty(AbstractDomainObject::PROPERTY_VERSIONED_UID) && $value->_getProperty(AbstractDomainObject::PROPERTY_VERSIONED_UID) > 0 && $value->_getProperty(AbstractDomainObject::PROPERTY_VERSIONED_UID) !== $value->getUid()) {
$value = (int)$value->_getProperty(AbstractDomainObject::PROPERTY_VERSIONED_UID);
}
}
$constraint = $query->equals($columnMap->getParentKeyFieldName(), $value);
if ($columnMap->getParentTableFieldName() !== null) {
$constraint = $query->logicalAnd(
$constraint,
$query->equals($columnMap->getParentTableFieldName(), $dataMap->getTableName())
);
}
} else {
// Note: $fieldValue is annotated as a string, but this cannot be trusted as the callers do not ensure this.
$constraint = $query->in('uid', GeneralUtility::intExplode(',', (string)$fieldValue));
}
if (!empty($relationTableMatchFields)) {
foreach ($relationTableMatchFields as $relationTableMatchFieldName => $relationTableMatchFieldValue) {
$constraint = $query->logicalAnd($constraint, $query->equals($relationTableMatchFieldName, $relationTableMatchFieldValue));
}
}
return $constraint;
}
/**
* This resolves relations via RelationHandler and returns their UIDs respectively, and works for MM/ForeignField/CSV in IRRE + Select + Group.
*
* Note: This only happens for resolving properties for models. When limiting a parentQuery, the Typo3DbQueryParser is taking care of it.
*
* By using the RelationHandler, the localized, deleted and moved records turn out to be properly resolved
* without having to build intermediate queries.
*
* This is currently only used in workspaces' context, as it is 1 additional DB query needed.
*
* @param DataMap $dataMap
* @param ColumnMap $columnMap
* @param DomainObjectInterface $parentObject
* @param string $fieldValue
* @param int $workspaceId
* @return array|false|mixed
*/
protected function resolveRelationValuesOfField(DataMap $dataMap, ColumnMap $columnMap, DomainObjectInterface $parentObject, $fieldValue, int $workspaceId)
{
$parentId = $parentObject->getUid();
// versionedUid in a multi-language setup is the overlaid versioned AND translated ID
if ($parentObject->_hasProperty(AbstractDomainObject::PROPERTY_VERSIONED_UID) && $parentObject->_getProperty(AbstractDomainObject::PROPERTY_VERSIONED_UID) > 0 && $parentObject->_getProperty(AbstractDomainObject::PROPERTY_VERSIONED_UID) !== $parentId) {
$parentId = $parentObject->_getProperty(AbstractDomainObject::PROPERTY_VERSIONED_UID);
} elseif ($parentObject->_hasProperty(AbstractDomainObject::PROPERTY_LANGUAGE_UID) && $parentObject->_getProperty(AbstractDomainObject::PROPERTY_LANGUAGE_UID) > 0) {
$parentId = $parentObject->_getProperty(AbstractDomainObject::PROPERTY_LOCALIZED_UID);
}
$relationHandler = GeneralUtility::makeInstance(RelationHandler::class);
$relationHandler->setWorkspaceId($workspaceId);
$relationHandler->setUseLiveReferenceIds(true);
$relationHandler->setUseLiveParentIds(true);
$tableName = $dataMap->getTableName();
$fieldName = $columnMap->getColumnName();
$fieldConfiguration = $GLOBALS['TCA'][$tableName]['columns'][$fieldName]['config'] ?? null;
if (!is_array($fieldConfiguration)) {
return [];
}
$relationHandler->start(
$fieldValue,
$fieldConfiguration['allowed'] ?? $fieldConfiguration['foreign_table'] ?? '',
$fieldConfiguration['MM'] ?? '',
$parentId,
$tableName,
$fieldConfiguration
);
$relationHandler->processDeletePlaceholder();
$relatedUids = [];
if (!empty($relationHandler->tableArray)) {
$relatedUids = reset($relationHandler->tableArray);
}
return $relatedUids;
}
/**
* Builds and returns the source to build a join for a m:n relation.
*
* @param string $propertyName
* @return \TYPO3\CMS\Extbase\Persistence\Generic\Qom\SourceInterface $source
*/
protected function getSource(DomainObjectInterface $parentObject, $propertyName)
{
$columnMap = $this->getDataMap(get_class($parentObject))->getColumnMap($propertyName);
$left = $this->qomFactory->selector(null, $columnMap->getRelationTableName());
$childClassName = $this->getType(get_class($parentObject), $propertyName);
$right = $this->qomFactory->selector($childClassName, $columnMap->getChildTableName());
$joinCondition = $this->qomFactory->equiJoinCondition($columnMap->getRelationTableName(), $columnMap->getChildKeyFieldName(), $columnMap->getChildTableName(), 'uid');
$source = $this->qomFactory->join($left, $right, Query::JCR_JOIN_TYPE_INNER, $joinCondition);
return $source;
}
/**
* Returns the mapped classProperty from the identityMap or
* mapResultToPropertyValue()
*
* If the field value is empty and the column map has no parent key field name,
* the relation will be empty. If the persistence session has a registered object of
* the correct type and identity (fieldValue), this function returns that object.
* Otherwise, it proceeds with mapResultToPropertyValue().
*
* @param DomainObjectInterface $parentObject
* @param string $propertyName
* @param mixed $fieldValue the raw field value
* @return mixed
* @see mapResultToPropertyValue()
*/
protected function mapObjectToClassProperty(DomainObjectInterface $parentObject, $propertyName, $fieldValue)
{
if ($this->propertyMapsByForeignKey($parentObject, $propertyName)) {
$result = $this->fetchRelated($parentObject, $propertyName, $fieldValue);
$propertyValue = $this->mapResultToPropertyValue($parentObject, $propertyName, $result);
} elseif (empty($fieldValue)) {
$propertyValue = $this->getEmptyRelationValue($parentObject, $propertyName);
} else {
$property = $this->reflectionService->getClassSchema(get_class($parentObject))->getProperty($propertyName);
if ($this->persistenceSession->hasIdentifier((string)$fieldValue, $property->getType())) {
$propertyValue = $this->persistenceSession->getObjectByIdentifier((string)$fieldValue, $property->getType());
} else {
$primaryType = $this->reflectionService
->getClassSchema(get_class($parentObject))
->getProperty($propertyName)
->getPrimaryType();
if ($primaryType === null) {
throw NoPropertyTypesException::create($parentObject::class, $propertyName);
}
$className = $primaryType->getClassName();
if (!is_string($className)) {
throw new \LogicException(
sprintf('Evaluated type of class property %s::%s is not a class name. Check the type declaration of the property to use a valid class name.', $parentObject::class, $propertyName),
1660217846
);
}
if ($this->persistenceSession->hasIdentifier((string)$fieldValue, $className)) {
$propertyValue = $this->persistenceSession->getObjectByIdentifier((string)$fieldValue, $className);
} else {
$result = $this->fetchRelated($parentObject, $propertyName, $fieldValue);
$propertyValue = $this->mapResultToPropertyValue($parentObject, $propertyName, $result);
}
}
}
return $propertyValue;
}
/**
* Checks if the relation is based on a foreign key.
*
* @param string $propertyName
* @return bool TRUE if the property is mapped
*/
protected function propertyMapsByForeignKey(DomainObjectInterface $parentObject, $propertyName)
{
$columnMap = $this->getDataMap(get_class($parentObject))->getColumnMap($propertyName);
return $columnMap->getParentKeyFieldName() !== null;
}
/**
* Returns the given result as property value of the specified property type.
*
* @param string $propertyName
* @param mixed $result The result
* @return mixed
*/
public function mapResultToPropertyValue(DomainObjectInterface $parentObject, $propertyName, $result)
{
$propertyValue = null;
if ($result instanceof LoadingStrategyInterface) {
$propertyValue = $result;
} else {
$property = $this->reflectionService->getClassSchema(get_class($parentObject))->getProperty($propertyName);
$primaryType = $property->getPrimaryType();
if ($primaryType === null) {
throw NoPropertyTypesException::create($parentObject::class, $propertyName);
}
if ($primaryType->getBuiltinType() === 'array' || in_array($primaryType->getClassName(), [\ArrayObject::class, \SplObjectStorage::class, ObjectStorage::class], true)) {
$objects = [];
foreach ($result as $value) {
$objects[] = $value;
}
if ($primaryType->getClassName() === \ArrayObject::class) {
$propertyValue = new \ArrayObject($objects);
} elseif ($primaryType->getClassName() === ObjectStorage::class) {
$propertyValue = new ObjectStorage();
foreach ($objects as $object) {
$propertyValue->attach($object);
}
$propertyValue->_memorizeCleanState();
} else {
$propertyValue = $objects;
}
} elseif (strpbrk((string)$primaryType->getClassName(), '_\\') !== false) {
// @todo: check the strpbrk function call. Seems to be a check for Tx_Foo_Bar style class names
if ($result instanceof QueryResultInterface) {
$propertyValue = $result->getFirst();
} else {
$propertyValue = $result;
}
}
}
return $propertyValue;
}
/**
* Counts the number of related objects assigned to a property of a parent object
*
* @param DomainObjectInterface $parentObject The object instance this proxy is part of
* @param string $propertyName The name of the proxied property in it's parent
* @param mixed $fieldValue The raw field value.
* @return int
*/
public function countRelated(DomainObjectInterface $parentObject, $propertyName, $fieldValue = '')
{
$query = $this->getPreparedQuery($parentObject, $propertyName, $fieldValue);
return $query->execute()->count();
}
/**
* Returns a data map for a given class name
*
* @param string $className The class name you want to fetch the Data Map for
* @throws Persistence\Generic\Exception
* @return DataMap The data map
*/
public function getDataMap($className)
{
if (!is_string($className) || $className === '') {
throw new Exception('No class name was given to retrieve the Data Map for.', 1251315965);
}
return $this->dataMapFactory->buildDataMap($className);
}
/**
* Returns the selector (table) name for a given class name.
*
* @param string $className
* @return string The selector name
*/
public function convertClassNameToTableName($className)
{
return $this->getDataMap($className)->getTableName();
}
/**
* Returns the column name for a given property name of the specified class.
*
* @param string $propertyName
* @param string $className
* @return string The column name
*/
public function convertPropertyNameToColumnName($propertyName, $className = null)
{
if (!empty($className)) {
$dataMap = $this->getDataMap($className);
if ($dataMap !== null) {
$columnMap = $dataMap->getColumnMap($propertyName);
if ($columnMap !== null) {
return $columnMap->getColumnName();
}
}
}
return GeneralUtility::camelCaseToLowerCaseUnderscored($propertyName);
}
/**
* Returns the type of a child object.
*
* @param string $parentClassName The class name of the object this proxy is part of
* @param string $propertyName The name of the proxied property in it's parent
* @throws UnexpectedTypeException
* @return string The class name of the child object
*/
public function getType($parentClassName, $propertyName)
{
try {
$primaryType = $this->reflectionService
->getClassSchema($parentClassName)
->getProperty($propertyName)
->getPrimaryType();
if ($primaryType === null) {
throw NoPropertyTypesException::create($parentClassName, $propertyName);
}
if ($primaryType->isCollection() && $primaryType->getCollectionValueTypes() !== []) {
$primaryCollectionValueType = $primaryType->getCollectionValueTypes()[0];
return $primaryCollectionValueType->getClassName()
?? $primaryCollectionValueType->getBuiltinType();
}
return $primaryType->getClassName()
?? $primaryType->getBuiltinType();
} catch (NoSuchPropertyException|NoPropertyTypesException $e) {
}
throw new UnexpectedTypeException('Could not determine the child object type.', 1251315967);
}
/**
* Returns a plain value, i.e. objects are flattened out if possible.
* Multi value objects or arrays will be converted to a comma-separated list for use in IN SQL queries.
*
* @param mixed $input The value that will be converted.
* @param ColumnMap $columnMap Optional column map for retrieving the date storage format.
* @throws \InvalidArgumentException
* @throws UnexpectedTypeException
* @return int|string
*/
public function getPlainValue($input, $columnMap = null)
{
if ($input === null) {
return 'NULL';
}
if ($input instanceof LazyLoadingProxy) {
$input = $input->_loadRealInstance();
}
if (is_bool($input)) {
$parameter = (int)$input;
} elseif (is_int($input)) {
$parameter = $input;
} elseif ($input instanceof \DateTimeInterface) {
if ($columnMap !== null && $columnMap->getDateTimeStorageFormat() !== null) {
$storageFormat = $columnMap->getDateTimeStorageFormat();
switch ($storageFormat) {
case 'datetime':
$parameter = $input->format('Y-m-d H:i:s');
break;
case 'date':
$parameter = $input->format('Y-m-d');
break;
case 'time':
$parameter = $input->format('H:i');
break;
default:
throw new \InvalidArgumentException('Column map DateTime format "' . $storageFormat . '" is unknown. Allowed values are date, datetime or time.', 1395353470);
}
} else {
$parameter = $input->format('U');
}
} elseif ($input instanceof DomainObjectInterface) {
$parameter = (int)$input->getUid();
} elseif (TypeHandlingUtility::isValidTypeForMultiValueComparison($input)) {
$plainValueArray = [];
foreach ($input as $inputElement) {
$plainValueArray[] = $this->getPlainValue($inputElement, $columnMap);
}
$parameter = implode(',', $plainValueArray);
} elseif (is_object($input)) {
if (TypeHandlingUtility::isCoreType($input) || $input instanceof \Stringable) {
$parameter = (string)$input;
} else {
throw new UnexpectedTypeException('An object of class "' . get_class($input) . '" could not be converted to a plain value.', 1274799934);
}
} else {
$parameter = (string)$input;
}
return $parameter;
}
}