From e7c77e6532d893356c326ae39f9480bb32d4f8a9 Mon Sep 17 00:00:00 2001 From: Daniel Siepmann Date: Mon, 19 Aug 2024 14:40:18 +0200 Subject: [PATCH 1/3] Add TYPO3 13 LTS Support --- .github/workflows/ci.yaml | 23 +- .php-cs-fixer.dist.php | 3 + .../Command/ImportConfigurationCommand.php | 2 +- .../Controller/Backend/AbstractController.php | 7 +- .../CustomAnnotationExtractor.php | 912 +++++++++++++----- Classes/Domain/Import/Importer/SaveData.php | 8 +- .../Typo3Converter/GeneralConverter.php | 6 +- .../Model/Backend/ImportConfiguration.php | 9 +- .../Backend/OrganisationRepository.php | 4 - Classes/Extension.php | 53 +- .../DataProcessing/ResolveEntities.php | 6 + .../ImportConfiguration/ContainsPlace.xml | 31 +- .../FlexForm/ImportConfiguration/Static.xml | 29 +- .../ImportConfiguration/SyncScope.xml | 29 +- .../FlexForm/Pages/tourist_attraction.xml | 24 +- Configuration/TCA/Overrides/pages.php | 2 +- .../tt_content_tourist_attraction.php | 2 +- .../TCA/tx_thuecat_tourist_attraction.php | 14 + .../TypoScript/Default/Setup.typoscript | 5 - .../User/All.tsconfig => user.tsconfig} | 0 Documentation/Changelog/3.1.0.rst | 28 + Documentation/Index.rst | 1 + Documentation/Settings.cfg | 52 - Documentation/guides.xml | 35 + .../Backend/Configuration/Index.html | 4 +- .../Templates/Backend/Import/Index.html | 2 +- Resources/Public/Icons/ModuleGroup.svg | 2 +- Tests/Acceptance/BackendConfigurationCest.php | 70 -- Tests/Acceptance/Data/.gitkeep | 0 Tests/Acceptance/Data/BasicDatabase.php | 87 -- .../Acceptance/Data/Sites/default/config.yaml | 47 - Tests/Acceptance/Support/AcceptanceTester.php | 54 -- Tests/Acceptance/Support/Environment.php | 60 -- Tests/Acceptance/_output/.gitignore | 2 - Tests/Functional/AbstractImportTestCase.php | 20 + .../ImportConfigurationCommandTest.php | 4 + Tests/Functional/ImportTest.php | 13 + codeception.dist.yml | 44 - composer.json | 25 +- ext_emconf.php | 2 +- ext_localconf.php | 2 +- ext_tables.php | 5 + phpstan-baseline.neon | 15 - shell.nix | 21 - 44 files changed, 926 insertions(+), 838 deletions(-) rename Configuration/{TSconfig/User/All.tsconfig => user.tsconfig} (100%) create mode 100644 Documentation/Changelog/3.1.0.rst delete mode 100644 Documentation/Settings.cfg create mode 100644 Documentation/guides.xml delete mode 100644 Tests/Acceptance/BackendConfigurationCest.php delete mode 100644 Tests/Acceptance/Data/.gitkeep delete mode 100644 Tests/Acceptance/Data/BasicDatabase.php delete mode 100644 Tests/Acceptance/Data/Sites/default/config.yaml delete mode 100644 Tests/Acceptance/Support/AcceptanceTester.php delete mode 100644 Tests/Acceptance/Support/Environment.php delete mode 100644 Tests/Acceptance/_output/.gitignore delete mode 100644 codeception.dist.yml create mode 100644 ext_tables.php diff --git a/.github/workflows/ci.yaml b/.github/workflows/ci.yaml index 2ade17ba..73fc0084 100644 --- a/.github/workflows/ci.yaml +++ b/.github/workflows/ci.yaml @@ -91,6 +91,10 @@ jobs: typo3-version: '^12.4' - php-version: '8.3' typo3-version: '^12.4' + - php-version: '8.2' + typo3-version: '^13.4' + - php-version: '8.3' + typo3-version: '^13.4' steps: - uses: actions/checkout@v3 @@ -103,7 +107,6 @@ jobs: - name: Install dependencies with expected TYPO3 version run: |- composer require --no-interaction --prefer-dist --no-progress "typo3/cms-backend:${{ matrix.typo3-version }}" "typo3/cms-core:${{ matrix.typo3-version }}" "typo3/cms-extbase:${{ matrix.typo3-version }}" "typo3/cms-frontend:${{ matrix.typo3-version }}" "typo3/cms-fluid-styled-content:${{ matrix.typo3-version }}" - ./vendor/bin/codecept build - name: Code Quality (by PHPStan) run: ./vendor/bin/phpstan analyse @@ -125,6 +128,12 @@ jobs: - php-version: '8.3' typo3-version: '^12.4' db-version: '8' + - php-version: '8.2' + typo3-version: '^13.4' + db-version: '8' + - php-version: '8.3' + typo3-version: '^13.4' + db-version: '8' steps: - uses: actions/checkout@v3 @@ -158,15 +167,3 @@ jobs: export typo3DatabaseUsername="root" export typo3DatabasePassword="root" ./vendor/bin/phpunit - - tests-acceptance: - runs-on: ubuntu-latest - steps: - - uses: actions/checkout@v3 - - - uses: cachix/install-nix-action@v24 - with: - nix_path: nixpkgs=channel:nixos-unstable - - - name: Run Acceptance Tests - run: nix-shell --run project-test-acceptance diff --git a/.php-cs-fixer.dist.php b/.php-cs-fixer.dist.php index 5bd6dc42..a847a6cc 100644 --- a/.php-cs-fixer.dist.php +++ b/.php-cs-fixer.dist.php @@ -2,6 +2,9 @@ $finder = (new PhpCsFixer\Finder()) ->ignoreVCSIgnored(true) ->in(realpath(__DIR__)) + ->notPath([ + 'Classes/Domain/Import/EntityMapper/CustomAnnotationExtractor.php', + ]); ; return (new \PhpCsFixer\Config()) diff --git a/Classes/Command/ImportConfigurationCommand.php b/Classes/Command/ImportConfigurationCommand.php index c8b1630f..0d2f2168 100644 --- a/Classes/Command/ImportConfigurationCommand.php +++ b/Classes/Command/ImportConfigurationCommand.php @@ -52,7 +52,7 @@ protected function configure(): void ); } - protected function execute(InputInterface $input, OutputInterface $output) + protected function execute(InputInterface $input, OutputInterface $output): int { Bootstrap::initializeBackendAuthentication(); diff --git a/Classes/Controller/Backend/AbstractController.php b/Classes/Controller/Backend/AbstractController.php index fe4ed3cd..7d6361a7 100644 --- a/Classes/Controller/Backend/AbstractController.php +++ b/Classes/Controller/Backend/AbstractController.php @@ -47,6 +47,11 @@ protected function initializeView(): void protected function htmlResponse(?string $html = null): ResponseInterface { - return parent::htmlResponse($html ?? $this->moduleTemplate->render()); + $action = sprintf( + '%s/%s', + str_replace('\\', '/', $this->request->getControllerName()), + ucfirst($this->request->getControllerActionName()), + ); + return parent::htmlResponse($html ?? $this->moduleTemplate->render($action)); } } diff --git a/Classes/Domain/Import/EntityMapper/CustomAnnotationExtractor.php b/Classes/Domain/Import/EntityMapper/CustomAnnotationExtractor.php index 936b6fa5..ac6b0a46 100644 --- a/Classes/Domain/Import/EntityMapper/CustomAnnotationExtractor.php +++ b/Classes/Domain/Import/EntityMapper/CustomAnnotationExtractor.php @@ -23,12 +23,13 @@ namespace WerkraumMedia\ThueCat\Domain\Import\EntityMapper; +use Symfony\Component\TypeInfo\Type; +use Symfony\Component\TypeInfo\TypeContext\TypeContextFactory; use function in_array; use InvalidArgumentException; use LogicException; use phpDocumentor\Reflection\DocBlock; use phpDocumentor\Reflection\DocBlock\Tags\InvalidTag; -use phpDocumentor\Reflection\DocBlock\Tags\Param; use phpDocumentor\Reflection\DocBlockFactory; use phpDocumentor\Reflection\DocBlockFactoryInterface; use phpDocumentor\Reflection\Types\Context; @@ -42,7 +43,7 @@ use Symfony\Component\PropertyInfo\Extractor\ReflectionExtractor; use Symfony\Component\PropertyInfo\PropertyDescriptionExtractorInterface; use Symfony\Component\PropertyInfo\PropertyTypeExtractorInterface; -use Symfony\Component\PropertyInfo\Type; +use Symfony\Component\PropertyInfo\Type as LegacyType; use Symfony\Component\PropertyInfo\Util\PhpDocTypeHelper; /** @@ -52,339 +53,754 @@ * * Make updating the file contents easier by keeping the original file contents as close as possible. */ -class CustomAnnotationExtractor implements PropertyDescriptionExtractorInterface, PropertyTypeExtractorInterface, ConstructorArgumentTypeExtractorInterface -{ - final public const PROPERTY = 0; - final public const ACCESSOR = 1; - final public const MUTATOR = 2; - - /** - * @var array - */ - private array $docBlocks = []; - - /** - * @var Context[] - */ - private array $contexts = []; - - private readonly \phpDocumentor\Reflection\DocBlockFactoryInterface $docBlockFactory; - private readonly \phpDocumentor\Reflection\Types\ContextFactory $contextFactory; - private readonly \Symfony\Component\PropertyInfo\Util\PhpDocTypeHelper $phpDocTypeHelper; - private readonly array $mutatorPrefixes; - private readonly array $accessorPrefixes; - private readonly array $arrayMutatorPrefixes; - - /** - * @param string[]|null $mutatorPrefixes - * @param string[]|null $accessorPrefixes - * @param string[]|null $arrayMutatorPrefixes - */ - public function __construct(DocBlockFactoryInterface $docBlockFactory = null, array $mutatorPrefixes = null, array $accessorPrefixes = null, array $arrayMutatorPrefixes = null) + +if (class_exists(TypeContextFactory::class)) { + class CustomAnnotationExtractor implements PropertyDescriptionExtractorInterface, PropertyTypeExtractorInterface, ConstructorArgumentTypeExtractorInterface { - if (!class_exists(DocBlockFactory::class)) { - throw new LogicException(sprintf('Unable to use the "%s" class as the "phpdocumentor/reflection-docblock" package is not installed. Try running composer require "phpdocumentor/reflection-docblock".', self::class)); + public const PROPERTY = 0; + public const ACCESSOR = 1; + public const MUTATOR = 2; + + /** + * @var array + */ + private array $docBlocks = []; + + /** + * @var Context[] + */ + private array $contexts = []; + + private DocBlockFactoryInterface $docBlockFactory; + private ContextFactory $contextFactory; + private TypeContextFactory $typeContextFactory; + private PhpDocTypeHelper $phpDocTypeHelper; + private array $mutatorPrefixes; + private array $accessorPrefixes; + private array $arrayMutatorPrefixes; + + /** + * @param string[]|null $mutatorPrefixes + * @param string[]|null $accessorPrefixes + * @param string[]|null $arrayMutatorPrefixes + */ + public function __construct(?DocBlockFactoryInterface $docBlockFactory = null, ?array $mutatorPrefixes = null, ?array $accessorPrefixes = null, ?array $arrayMutatorPrefixes = null) + { + if (!class_exists(DocBlockFactory::class)) { + throw new \LogicException(sprintf('Unable to use the "%s" class as the "phpdocumentor/reflection-docblock" package is not installed. Try running composer require "phpdocumentor/reflection-docblock".', __CLASS__)); + } + + $this->docBlockFactory = $docBlockFactory ?: DocBlockFactory::createInstance(); + $this->contextFactory = new ContextFactory(); + $this->typeContextFactory = new TypeContextFactory(); + $this->phpDocTypeHelper = new PhpDocTypeHelper(); + $this->mutatorPrefixes = $mutatorPrefixes ?? ReflectionExtractor::$defaultMutatorPrefixes; + $this->accessorPrefixes = $accessorPrefixes ?? ReflectionExtractor::$defaultAccessorPrefixes; + $this->arrayMutatorPrefixes = $arrayMutatorPrefixes ?? ReflectionExtractor::$defaultArrayMutatorPrefixes; } - $this->docBlockFactory = $docBlockFactory ?: DocBlockFactory::createInstance(); - $this->contextFactory = new ContextFactory(); - $this->phpDocTypeHelper = new PhpDocTypeHelper(); - $this->mutatorPrefixes = $mutatorPrefixes ?? ReflectionExtractor::$defaultMutatorPrefixes; - $this->accessorPrefixes = $accessorPrefixes ?? ReflectionExtractor::$defaultAccessorPrefixes; - $this->arrayMutatorPrefixes = $arrayMutatorPrefixes ?? ReflectionExtractor::$defaultArrayMutatorPrefixes; - } + public function getShortDescription(string $class, string $property, array $context = []): ?string + { + /** @var $docBlock DocBlock */ + [$docBlock] = $this->findDocBlock($class, $property); + if (!$docBlock) { + return null; + } + + $shortDescription = $docBlock->getSummary(); + + if ($shortDescription) { + return $shortDescription; + } + + foreach ($docBlock->getTagsByName('var') as $var) { + if ($var && !$var instanceof InvalidTag) { + $varDescription = $var->getDescription()->render(); + + if ($varDescription) { + return $varDescription; + } + } + } - /** - * {@inheritdoc} - */ - public function getShortDescription(string $class, string $property, array $context = []): ?string - { - /** @var $docBlock DocBlock */ - [$docBlock] = $this->getDocBlock($class, $property); - if (!$docBlock) { return null; } - $shortDescription = $docBlock->getSummary(); + public function getLongDescription(string $class, string $property, array $context = []): ?string + { + /** @var $docBlock DocBlock */ + [$docBlock] = $this->findDocBlock($class, $property); + if (!$docBlock) { + return null; + } + + $contents = $docBlock->getDescription()->render(); - if (!empty($shortDescription)) { - return $shortDescription; + return '' === $contents ? null : $contents; } - foreach ($docBlock->getTagsByName('var') as $var) { - if ($var && !$var instanceof InvalidTag) { - $varDescription = $var->getDescription()->render(); + public function getTypes(string $class, string $property, array $context = []): ?array + { + /** @var $docBlock DocBlock */ + [$docBlock, $source, $prefix] = $this->findDocBlock($class, $property); + if (!$docBlock) { + return null; + } + + $tag = match ($source) { + self::PROPERTY => 'var', + self::ACCESSOR => 'return', + self::MUTATOR => 'param', + }; + + $parentClass = null; + $types = []; + /** @var DocBlock\Tags\Var_|DocBlock\Tags\Return_|DocBlock\Tags\Param $tag */ + foreach ($docBlock->getTagsByName($tag) as $tag) { + if ($tag && !$tag instanceof InvalidTag && null !== $tag->getType()) { + foreach ($this->phpDocTypeHelper->getTypes($tag->getType()) as $type) { + switch ($type->getClassName()) { + case 'self': + case 'static': + $resolvedClass = $class; + break; + + case 'parent': + if (false !== $resolvedClass = $parentClass ??= get_parent_class($class)) { + break; + } + // no break + + default: + $types[] = $type; + continue 2; + } - if (!empty($varDescription)) { - return $varDescription; + $types[] = new LegacyType(LegacyType::BUILTIN_TYPE_OBJECT, $type->isNullable(), $resolvedClass, $type->isCollection(), $type->getCollectionKeyTypes(), $type->getCollectionValueTypes()); + } } } - } - return null; - } + if (!isset($types[0])) { + return null; + } - /** - * {@inheritdoc} - */ - public function getLongDescription(string $class, string $property, array $context = []): ?string - { - /** @var $docBlock DocBlock */ - [$docBlock] = $this->getDocBlock($class, $property); - if (!$docBlock) { - return null; + if (!\in_array($prefix, $this->arrayMutatorPrefixes, true)) { + return $types; + } + + return [new LegacyType(LegacyType::BUILTIN_TYPE_ARRAY, false, null, true, new LegacyType(LegacyType::BUILTIN_TYPE_INT), $types[0])]; } - $contents = $docBlock->getDescription()->render(); + public function getTypesFromConstructor(string $class, string $property): ?array + { + $docBlock = $this->getDocBlockFromConstructor($class, $property); - return $contents === '' ? null : $contents; - } + if (!$docBlock) { + return null; + } - /** - * {@inheritdoc} - */ - public function getTypes(string $class, string $property, array $context = []): ?array - { - /** @var $docBlock DocBlock */ - [$docBlock, $source, $prefix] = $this->getDocBlock($class, $property); - if (!$docBlock) { - return null; + $types = []; + /** @var DocBlock\Tags\Var_|DocBlock\Tags\Return_|DocBlock\Tags\Param $tag */ + foreach ($docBlock->getTagsByName('param') as $tag) { + if ($tag && null !== $tag->getType()) { + $types[] = $this->phpDocTypeHelper->getTypes($tag->getType()); + } + } + + if (!isset($types[0]) || [] === $types[0]) { + return null; + } + + return array_merge([], ...$types); } - switch ($source) { - case self::PROPERTY: - $tag = 'var'; - break; + /** + * @experimental + */ + public function getType(string $class, string $property, array $context = []): ?Type + { + /** @var $docBlock DocBlock */ + [$docBlock, $source, $prefix] = $this->findDocBlock($class, $property); + if (!$docBlock) { + return null; + } - case self::ACCESSOR: - $tag = 'return'; - break; + $tag = match ($source) { + self::PROPERTY => 'var', + self::ACCESSOR => 'return', + self::MUTATOR => 'param', + }; - case self::MUTATOR: - $tag = 'param'; - break; - } + $types = []; + $typeContext = $this->typeContextFactory->createFromClassName($class); - $parentClass = null; - $types = []; - /** @var DocBlock\Tags\Var_|DocBlock\Tags\Return_|DocBlock\Tags\Param $tag */ - foreach ($docBlock->getTagsByName($tag) as $tag) { - if ($tag && !$tag instanceof InvalidTag && $tag->getType() !== null) { - foreach ($this->phpDocTypeHelper->getTypes($tag->getType()) as $type) { - switch ($type->getClassName()) { - case 'self': - case 'static': - $resolvedClass = $class; - break; - - case 'parent': - if (false !== $resolvedClass = $parentClass ?? $parentClass = get_parent_class($class)) { - break; - } - // no break + /** @var DocBlock\Tags\Var_|DocBlock\Tags\Return_|DocBlock\Tags\Param $tag */ + foreach ($docBlock->getTagsByName($tag) as $tag) { + if ($tag instanceof InvalidTag || !$tagType = $tag->getType()) { + continue; + } + + $type = $this->phpDocTypeHelper->getType($tagType); - default: - $types[] = $type; - continue 2; + if (!$type instanceof ObjectType) { + $types[] = $type; + + continue; + } + + $normalizedClassName = match ($type->getClassName()) { + 'self' => $typeContext->getDeclaringClass(), + 'static' => $typeContext->getCalledClass(), + default => $type->getClassName(), + }; + + if ('parent' === $normalizedClassName) { + try { + $normalizedClassName = $typeContext->getParentClass(); + } catch (LogicException) { + // if there is no parent for the current class, we keep the "parent" raw string } + } + + $types[] = $type->isNullable() ? Type::nullable(Type::object($normalizedClassName)) : Type::object($normalizedClassName); + } + + if (null === $type = $types[0] ?? null) { + return null; + } + + if (!\in_array($prefix, $this->arrayMutatorPrefixes, true)) { + return $type; + } - $types[] = new Type(Type::BUILTIN_TYPE_OBJECT, $type->isNullable(), $resolvedClass, $type->isCollection(), $type->getCollectionKeyTypes(), $type->getCollectionValueTypes()); + return Type::list($type); + } + + /** + * @experimental + */ + public function getTypeFromConstructor(string $class, string $property): ?Type + { + if (!$docBlock = $this->getDocBlockFromConstructor($class, $property)) { + return null; + } + + $types = []; + /** @var DocBlock\Tags\Var_|DocBlock\Tags\Return_|DocBlock\Tags\Param $tag */ + foreach ($docBlock->getTagsByName('param') as $tag) { + if ($tag instanceof InvalidTag || !$tagType = $tag->getType()) { + continue; } + + $types[] = $this->phpDocTypeHelper->getType($tagType); } + + return $types[0] ?? null; } - if (!isset($types[0])) { - return null; + public function getDocBlock(string $class, string $property): ?DocBlock + { + $output = $this->findDocBlock($class, $property); + + return $output[0]; } - if (!in_array($prefix, $this->arrayMutatorPrefixes)) { - return $types; + private function getDocBlockFromConstructor(string $class, string $property): ?DocBlock + { + try { + $reflectionClass = new \ReflectionClass($class); + } catch (\ReflectionException) { + return null; + } + $reflectionConstructor = $reflectionClass->getConstructor(); + if (!$reflectionConstructor) { + return null; + } + + try { + $docBlock = $this->docBlockFactory->create($reflectionConstructor, $this->contextFactory->createFromReflector($reflectionConstructor)); + + return $this->filterDocBlockParams($docBlock, $property); + } catch (\InvalidArgumentException) { + return null; + } } - return [new Type(Type::BUILTIN_TYPE_ARRAY, false, null, true, new Type(Type::BUILTIN_TYPE_INT), $types[0])]; - } + private function filterDocBlockParams(DocBlock $docBlock, string $allowedParam): DocBlock + { + $tags = array_values(array_filter($docBlock->getTagsByName('param'), fn ($tag) => $tag instanceof DocBlock\Tags\Param && $allowedParam === $tag->getVariableName())); - /** - * {@inheritdoc} - */ - public function getTypesFromConstructor(string $class, string $property): ?array - { - $docBlock = $this->getDocBlockFromConstructor($class, $property); + return new DocBlock($docBlock->getSummary(), $docBlock->getDescription(), $tags, $docBlock->getContext(), + $docBlock->getLocation(), $docBlock->isTemplateStart(), $docBlock->isTemplateEnd()); + } - if (!$docBlock) { - return null; + /** + * @return array{DocBlock|null, int|null, string|null} + */ + private function findDocBlock(string $class, string $property): array + { + $propertyHash = sprintf('%s::%s', $class, $property); + + if (isset($this->docBlocks[$propertyHash])) { + return $this->docBlocks[$propertyHash]; + } + + try { + $reflectionProperty = new \ReflectionProperty($class, $property); + } catch (\ReflectionException) { + $reflectionProperty = null; + } + + $ucFirstProperty = ucfirst($property); + + switch (true) { + // We re order the different cases + case [$docBlock, $prefix] = $this->getDocBlockFromMethod($class, $ucFirstProperty, self::MUTATOR): + $data = [$docBlock, self::MUTATOR, $prefix]; + break; + + case [$docBlock] = $this->getDocBlockFromMethod($class, $ucFirstProperty, self::ACCESSOR): + $data = [$docBlock, self::ACCESSOR, null]; + break; + + case $reflectionProperty?->isPromoted() && $docBlock = $this->getDocBlockFromConstructor($class, $property): + $data = [$docBlock, self::MUTATOR, null]; + break; + + case $docBlock = $this->getDocBlockFromProperty($class, $property): + $data = [$docBlock, self::PROPERTY, null]; + break; + + default: + $data = [null, null, null]; + } + + return $this->docBlocks[$propertyHash] = $data; } - $types = []; - /** @var DocBlock\Tags\Var_|DocBlock\Tags\Return_|DocBlock\Tags\Param $tag */ - foreach ($docBlock->getTagsByName('param') as $tag) { - if ($tag && $tag->getType() !== null) { - $types[] = $this->phpDocTypeHelper->getTypes($tag->getType()); + private function getDocBlockFromProperty(string $class, string $property): ?DocBlock + { + // Use a ReflectionProperty instead of $class to get the parent class if applicable + try { + $reflectionProperty = new \ReflectionProperty($class, $property); + } catch (\ReflectionException) { + return null; + } + + $reflector = $reflectionProperty->getDeclaringClass(); + + foreach ($reflector->getTraits() as $trait) { + if ($trait->hasProperty($property)) { + return $this->getDocBlockFromProperty($trait->getName(), $property); + } + } + + try { + return $this->docBlockFactory->create($reflectionProperty, $this->createFromReflector($reflector)); + } catch (\InvalidArgumentException|\RuntimeException) { + return null; } } - if (!isset($types[0]) || $types[0] === []) { - return null; + /** + * @return array{DocBlock, string}|null + */ + private function getDocBlockFromMethod(string $class, string $ucFirstProperty, int $type): ?array + { + $prefixes = self::ACCESSOR === $type ? $this->accessorPrefixes : $this->mutatorPrefixes; + $prefix = null; + + foreach ($prefixes as $prefix) { + $methodName = $prefix.$ucFirstProperty; + + try { + $reflectionMethod = new \ReflectionMethod($class, $methodName); + if ($reflectionMethod->isStatic()) { + continue; + } + + if ( + (self::ACCESSOR === $type && 0 === $reflectionMethod->getNumberOfRequiredParameters()) + || (self::MUTATOR === $type && $reflectionMethod->getNumberOfParameters() >= 1) + ) { + break; + } + } catch (\ReflectionException) { + // Try the next prefix if the method doesn't exist + } + } + + if (!isset($reflectionMethod)) { + return null; + } + + $reflector = $reflectionMethod->getDeclaringClass(); + + foreach ($reflector->getTraits() as $trait) { + if ($trait->hasMethod($methodName)) { + return $this->getDocBlockFromMethod($trait->getName(), $ucFirstProperty, $type); + } + } + + try { + return [$this->docBlockFactory->create($reflectionMethod, $this->createFromReflector($reflector)), $prefix]; + } catch (\InvalidArgumentException|\RuntimeException) { + return null; + } } - return array_merge([], ...$types); - } + /** + * Prevents a lot of redundant calls to ContextFactory::createForNamespace(). + */ + private function createFromReflector(\ReflectionClass $reflector): Context + { + $cacheKey = $reflector->getNamespaceName().':'.$reflector->getFileName(); - private function getDocBlockFromConstructor(string $class, string $property): ?DocBlock - { - try { - $reflectionClass = new ReflectionClass($class); - } catch (ReflectionException $e) { - return null; + if (isset($this->contexts[$cacheKey])) { + return $this->contexts[$cacheKey]; + } + + $this->contexts[$cacheKey] = $this->contextFactory->createFromReflector($reflector); + + return $this->contexts[$cacheKey]; } - $reflectionConstructor = $reflectionClass->getConstructor(); - if (!$reflectionConstructor) { - return null; + } +} else { + class CustomAnnotationExtractor implements PropertyDescriptionExtractorInterface, PropertyTypeExtractorInterface, ConstructorArgumentTypeExtractorInterface + { + final public const PROPERTY = 0; + final public const ACCESSOR = 1; + final public const MUTATOR = 2; + + /** + * @var array + */ + private array $docBlocks = []; + + /** + * @var Context[] + */ + private array $contexts = []; + + private readonly \phpDocumentor\Reflection\DocBlockFactoryInterface $docBlockFactory; + private readonly \phpDocumentor\Reflection\Types\ContextFactory $contextFactory; + private readonly \Symfony\Component\PropertyInfo\Util\PhpDocTypeHelper $phpDocTypeHelper; + private readonly array $mutatorPrefixes; + private readonly array $accessorPrefixes; + private readonly array $arrayMutatorPrefixes; + + /** + * @param string[]|null $mutatorPrefixes + * @param string[]|null $accessorPrefixes + * @param string[]|null $arrayMutatorPrefixes + */ + public function __construct(DocBlockFactoryInterface $docBlockFactory = null, array $mutatorPrefixes = null, array $accessorPrefixes = null, array $arrayMutatorPrefixes = null) + { + if (!class_exists(DocBlockFactory::class)) { + throw new LogicException(sprintf('Unable to use the "%s" class as the "phpdocumentor/reflection-docblock" package is not installed. Try running composer require "phpdocumentor/reflection-docblock".', self::class)); + } + + $this->docBlockFactory = $docBlockFactory ?: DocBlockFactory::createInstance(); + $this->contextFactory = new ContextFactory(); + $this->phpDocTypeHelper = new PhpDocTypeHelper(); + $this->mutatorPrefixes = $mutatorPrefixes ?? ReflectionExtractor::$defaultMutatorPrefixes; + $this->accessorPrefixes = $accessorPrefixes ?? ReflectionExtractor::$defaultAccessorPrefixes; + $this->arrayMutatorPrefixes = $arrayMutatorPrefixes ?? ReflectionExtractor::$defaultArrayMutatorPrefixes; } - try { - $docBlock = $this->docBlockFactory->create($reflectionConstructor, $this->contextFactory->createFromReflector($reflectionConstructor)); + /** + * {@inheritdoc} + */ + public function getShortDescription(string $class, string $property, array $context = []): ?string + { + /** @var $docBlock DocBlock */ + [$docBlock] = $this->getDocBlock($class, $property); + if (!$docBlock) { + return null; + } + + $shortDescription = $docBlock->getSummary(); + + if (!empty($shortDescription)) { + return $shortDescription; + } + + foreach ($docBlock->getTagsByName('var') as $var) { + if ($var && !$var instanceof InvalidTag) { + $varDescription = $var->getDescription()->render(); + + if (!empty($varDescription)) { + return $varDescription; + } + } + } - return $this->filterDocBlockParams($docBlock, $property); - } catch (InvalidArgumentException) { return null; } - } - private function filterDocBlockParams(DocBlock $docBlock, string $allowedParam): DocBlock - { - $tags = array_values(array_filter($docBlock->getTagsByName('param'), function ($tag) use ($allowedParam) { - return $tag instanceof Param && $allowedParam === $tag->getVariableName(); - })); - - return new DocBlock( - $docBlock->getSummary(), - $docBlock->getDescription(), - $tags, - $docBlock->getContext(), - $docBlock->getLocation(), - $docBlock->isTemplateStart(), - $docBlock->isTemplateEnd() - ); - } + /** + * {@inheritdoc} + */ + public function getLongDescription(string $class, string $property, array $context = []): ?string + { + /** @var $docBlock DocBlock */ + [$docBlock] = $this->getDocBlock($class, $property); + if (!$docBlock) { + return null; + } - /** - * @return array{DocBlock|null, int|null, string|null} - */ - private function getDocBlock(string $class, string $property): array - { - $propertyHash = sprintf('%s::%s', $class, $property); + $contents = $docBlock->getDescription()->render(); - if (isset($this->docBlocks[$propertyHash])) { - return $this->docBlocks[$propertyHash]; + return $contents === '' ? null : $contents; } - $ucFirstProperty = ucfirst($property); + /** + * {@inheritdoc} + */ + public function getTypes(string $class, string $property, array $context = []): ?array + { + /** @var $docBlock DocBlock */ + [$docBlock, $source, $prefix] = $this->getDocBlock($class, $property); + if (!$docBlock) { + return null; + } + + switch ($source) { + case self::PROPERTY: + $tag = 'var'; + break; + + case self::ACCESSOR: + $tag = 'return'; + break; - switch (true) { - case [$docBlock, $prefix] = $this->getDocBlockFromMethod($class, $ucFirstProperty, self::MUTATOR): - $data = [$docBlock, self::MUTATOR, $prefix]; - break; + case self::MUTATOR: + $tag = 'param'; + break; + } + + $parentClass = null; + $types = []; + /** @var DocBlock\Tags\Var_|DocBlock\Tags\Return_|DocBlock\Tags\Param $tag */ + foreach ($docBlock->getTagsByName($tag) as $tag) { + if ($tag && !$tag instanceof InvalidTag && $tag->getType() !== null) { + foreach ($this->phpDocTypeHelper->getTypes($tag->getType()) as $type) { + switch ($type->getClassName()) { + case 'self': + case 'static': + $resolvedClass = $class; + break; - case [$docBlock] = $this->getDocBlockFromMethod($class, $ucFirstProperty, self::ACCESSOR): - $data = [$docBlock, self::ACCESSOR, null]; - break; + case 'parent': + if (false !== $resolvedClass = $parentClass ?? $parentClass = get_parent_class($class)) { + break; + } + // no break - case $docBlock = $this->getDocBlockFromProperty($class, $property): - $data = [$docBlock, self::PROPERTY, null]; - break; + default: + $types[] = $type; + continue 2; + } - default: - $data = [null, null, null]; + $types[] = new Type(Type::BUILTIN_TYPE_OBJECT, $type->isNullable(), $resolvedClass, $type->isCollection(), $type->getCollectionKeyTypes(), $type->getCollectionValueTypes()); + } + } + } + + if (!isset($types[0])) { + return null; + } + + if (!in_array($prefix, $this->arrayMutatorPrefixes)) { + return $types; + } + + return [new Type(Type::BUILTIN_TYPE_ARRAY, false, null, true, new Type(Type::BUILTIN_TYPE_INT), $types[0])]; } - return $this->docBlocks[$propertyHash] = $data; - } + /** + * {@inheritdoc} + */ + public function getTypesFromConstructor(string $class, string $property): ?array + { + $docBlock = $this->getDocBlockFromConstructor($class, $property); - private function getDocBlockFromProperty(string $class, string $property): ?DocBlock - { - // Use a ReflectionProperty instead of $class to get the parent class if applicable - try { - $reflectionProperty = new ReflectionProperty($class, $property); - } catch (ReflectionException $e) { - return null; + if (!$docBlock) { + return null; + } + + $types = []; + /** @var DocBlock\Tags\Var_|DocBlock\Tags\Return_|DocBlock\Tags\Param $tag */ + foreach ($docBlock->getTagsByName('param') as $tag) { + if ($tag && $tag->getType() !== null) { + $types[] = $this->phpDocTypeHelper->getTypes($tag->getType()); + } + } + + if (!isset($types[0]) || $types[0] === []) { + return null; + } + + return array_merge([], ...$types); } - $reflector = $reflectionProperty->getDeclaringClass(); + private function getDocBlockFromConstructor(string $class, string $property): ?DocBlock + { + try { + $reflectionClass = new ReflectionClass($class); + } catch (ReflectionException $e) { + return null; + } + $reflectionConstructor = $reflectionClass->getConstructor(); + if (!$reflectionConstructor) { + return null; + } + + try { + $docBlock = $this->docBlockFactory->create($reflectionConstructor, $this->contextFactory->createFromReflector($reflectionConstructor)); - foreach ($reflector->getTraits() as $trait) { - if ($trait->hasProperty($property)) { - return $this->getDocBlockFromProperty($trait->getName(), $property); + return $this->filterDocBlockParams($docBlock, $property); + } catch (InvalidArgumentException) { + return null; } } - try { - return $this->docBlockFactory->create($reflectionProperty, $this->createFromReflector($reflector)); - } catch (InvalidArgumentException|RuntimeException) { - return null; + private function filterDocBlockParams(DocBlock $docBlock, string $allowedParam): DocBlock + { + $tags = array_values(array_filter($docBlock->getTagsByName('param'), function ($tag) use ($allowedParam) { + return $tag instanceof Param && $allowedParam === $tag->getVariableName(); + })); + + return new DocBlock( + $docBlock->getSummary(), + $docBlock->getDescription(), + $tags, + $docBlock->getContext(), + $docBlock->getLocation(), + $docBlock->isTemplateStart(), + $docBlock->isTemplateEnd() + ); } - } - /** - * @return array{DocBlock, string}|null - */ - private function getDocBlockFromMethod(string $class, string $ucFirstProperty, int $type): ?array - { - $prefixes = $type === self::ACCESSOR ? $this->accessorPrefixes : $this->mutatorPrefixes; - $prefix = null; + /** + * @return array{DocBlock|null, int|null, string|null} + */ + private function getDocBlock(string $class, string $property): array + { + $propertyHash = sprintf('%s::%s', $class, $property); - foreach ($prefixes as $prefix) { - $methodName = $prefix . $ucFirstProperty; + if (isset($this->docBlocks[$propertyHash])) { + return $this->docBlocks[$propertyHash]; + } - try { - $reflectionMethod = new ReflectionMethod($class, $methodName); - if ($reflectionMethod->isStatic()) { - continue; - } + $ucFirstProperty = ucfirst($property); - if ( - ($type === self::ACCESSOR && $reflectionMethod->getNumberOfRequiredParameters() === 0) || - ($type === self::MUTATOR && $reflectionMethod->getNumberOfParameters() >= 1) - ) { + switch (true) { + case [$docBlock, $prefix] = $this->getDocBlockFromMethod($class, $ucFirstProperty, self::MUTATOR): + $data = [$docBlock, self::MUTATOR, $prefix]; break; - } - } catch (ReflectionException) { - // Try the next prefix if the method doesn't exist + + case [$docBlock] = $this->getDocBlockFromMethod($class, $ucFirstProperty, self::ACCESSOR): + $data = [$docBlock, self::ACCESSOR, null]; + break; + + case $docBlock = $this->getDocBlockFromProperty($class, $property): + $data = [$docBlock, self::PROPERTY, null]; + break; + + default: + $data = [null, null, null]; } - } - if (!isset($reflectionMethod)) { - return null; + return $this->docBlocks[$propertyHash] = $data; } - $reflector = $reflectionMethod->getDeclaringClass(); + private function getDocBlockFromProperty(string $class, string $property): ?DocBlock + { + // Use a ReflectionProperty instead of $class to get the parent class if applicable + try { + $reflectionProperty = new ReflectionProperty($class, $property); + } catch (ReflectionException $e) { + return null; + } + + $reflector = $reflectionProperty->getDeclaringClass(); - foreach ($reflector->getTraits() as $trait) { - if ($trait->hasMethod($methodName)) { - return $this->getDocBlockFromMethod($trait->getName(), $ucFirstProperty, $type); + foreach ($reflector->getTraits() as $trait) { + if ($trait->hasProperty($property)) { + return $this->getDocBlockFromProperty($trait->getName(), $property); + } } - } - try { - return [$this->docBlockFactory->create($reflectionMethod, $this->createFromReflector($reflector)), $prefix]; - } catch (InvalidArgumentException|RuntimeException) { - return null; + try { + return $this->docBlockFactory->create($reflectionProperty, $this->createFromReflector($reflector)); + } catch (InvalidArgumentException|RuntimeException) { + return null; + } } - } - /** - * Prevents a lot of redundant calls to ContextFactory::createForNamespace(). - */ - private function createFromReflector(ReflectionClass $reflector): Context - { - $cacheKey = $reflector->getNamespaceName() . ':' . $reflector->getFileName(); + /** + * @return array{DocBlock, string}|null + */ + private function getDocBlockFromMethod(string $class, string $ucFirstProperty, int $type): ?array + { + $prefixes = $type === self::ACCESSOR ? $this->accessorPrefixes : $this->mutatorPrefixes; + $prefix = null; + + foreach ($prefixes as $prefix) { + $methodName = $prefix . $ucFirstProperty; + + try { + $reflectionMethod = new ReflectionMethod($class, $methodName); + if ($reflectionMethod->isStatic()) { + continue; + } - if (isset($this->contexts[$cacheKey])) { - return $this->contexts[$cacheKey]; + if ( + ($type === self::ACCESSOR && $reflectionMethod->getNumberOfRequiredParameters() === 0) || + ($type === self::MUTATOR && $reflectionMethod->getNumberOfParameters() >= 1) + ) { + break; + } + } catch (ReflectionException) { + // Try the next prefix if the method doesn't exist + } + } + + if (!isset($reflectionMethod)) { + return null; + } + + $reflector = $reflectionMethod->getDeclaringClass(); + + foreach ($reflector->getTraits() as $trait) { + if ($trait->hasMethod($methodName)) { + return $this->getDocBlockFromMethod($trait->getName(), $ucFirstProperty, $type); + } + } + + try { + return [$this->docBlockFactory->create($reflectionMethod, $this->createFromReflector($reflector)), $prefix]; + } catch (InvalidArgumentException|RuntimeException) { + return null; + } } - $this->contexts[$cacheKey] = $this->contextFactory->createFromReflector($reflector); + /** + * Prevents a lot of redundant calls to ContextFactory::createForNamespace(). + */ + private function createFromReflector(ReflectionClass $reflector): Context + { + $cacheKey = $reflector->getNamespaceName() . ':' . $reflector->getFileName(); + + if (isset($this->contexts[$cacheKey])) { + return $this->contexts[$cacheKey]; + } + + $this->contexts[$cacheKey] = $this->contextFactory->createFromReflector($reflector); - return $this->contexts[$cacheKey]; + return $this->contexts[$cacheKey]; + } } } diff --git a/Classes/Domain/Import/Importer/SaveData.php b/Classes/Domain/Import/Importer/SaveData.php index 10a7a640..7cf8b215 100644 --- a/Classes/Domain/Import/Importer/SaveData.php +++ b/Classes/Domain/Import/Importer/SaveData.php @@ -161,10 +161,10 @@ private function getEntityData(Entity $entity): array private function getExistingUid(Entity $entity): int { - $tableColumns = $this->connectionPool + $table = $this->connectionPool ->getConnectionForTable($entity->getTypo3DatabaseTableName()) - ->getSchemaManager() - ->listTableColumns($entity->getTypo3DatabaseTableName()) + ->getSchemaInformation() + ->introspectTable($entity->getTypo3DatabaseTableName()) ; $queryBuilder = $this->connectionPool->getQueryBuilderForTable($entity->getTypo3DatabaseTableName()); @@ -175,7 +175,7 @@ private function getExistingUid(Entity $entity): int 'remote_id', $queryBuilder->createNamedParameter($entity->getRemoteId()) )); - if (isset($tableColumns['sys_language_uid'])) { + if ($table->hasColumn('sys_language_uid')) { $queryBuilder->andWhere($queryBuilder->expr()->eq( 'sys_language_uid', $queryBuilder->createNamedParameter($entity->getTypo3SystemLanguageUid()) diff --git a/Classes/Domain/Import/Typo3Converter/GeneralConverter.php b/Classes/Domain/Import/Typo3Converter/GeneralConverter.php index d96cb39b..3973fa85 100644 --- a/Classes/Domain/Import/Typo3Converter/GeneralConverter.php +++ b/Classes/Domain/Import/Typo3Converter/GeneralConverter.php @@ -228,9 +228,9 @@ private function getManagerUid(object $entity): string [$entity->getManagedBy()] ) ); - $manager = $this->organisationRepository->findOneByRemoteId( - $entity->getManagedBy()->getId() - ); + $manager = $this->organisationRepository->findOneBy([ + 'remoteId' => $entity->getManagedBy()->getId(), + ]); return $manager ? (string)$manager->getUid() : ''; } diff --git a/Classes/Domain/Model/Backend/ImportConfiguration.php b/Classes/Domain/Model/Backend/ImportConfiguration.php index 895a4724..6083413e 100644 --- a/Classes/Domain/Model/Backend/ImportConfiguration.php +++ b/Classes/Domain/Model/Backend/ImportConfiguration.php @@ -25,6 +25,7 @@ use DateTimeImmutable; use Exception; +use RuntimeException; use TYPO3\CMS\Core\Utility\ArrayUtility; use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Extbase\DomainObject\AbstractEntity; @@ -162,7 +163,13 @@ private function getEntries(): array private function getConfigurationAsArray(): array { - return GeneralUtility::xml2array($this->configuration); + $asArray = GeneralUtility::xml2array($this->configuration); + + if (is_array($asArray) === false) { + throw new RuntimeException('Could not parse the configuration: ' . $asArray, 1729148214); + } + + return $asArray; } /** diff --git a/Classes/Domain/Repository/Backend/OrganisationRepository.php b/Classes/Domain/Repository/Backend/OrganisationRepository.php index 446cb6c5..bd80b388 100644 --- a/Classes/Domain/Repository/Backend/OrganisationRepository.php +++ b/Classes/Domain/Repository/Backend/OrganisationRepository.php @@ -25,11 +25,7 @@ use TYPO3\CMS\Extbase\Persistence\Generic\Typo3QuerySettings; use TYPO3\CMS\Extbase\Persistence\Repository; -use WerkraumMedia\ThueCat\Domain\Model\Backend\Organisation; -/** - * @method Organisation|null findOneByRemoteId(string $remoteId) - */ class OrganisationRepository extends Repository { public function __construct( diff --git a/Classes/Extension.php b/Classes/Extension.php index 7e1edd8c..42f2c885 100644 --- a/Classes/Extension.php +++ b/Classes/Extension.php @@ -25,6 +25,7 @@ use TYPO3\CMS\Core\Cache\Backend\TransientMemoryBackend; use TYPO3\CMS\Core\DataHandling\PageDoktypeRegistry; +use TYPO3\CMS\Core\Information\Typo3Version; use TYPO3\CMS\Core\Utility\ExtensionManagementUtility; use TYPO3\CMS\Core\Utility\GeneralUtility; @@ -43,11 +44,16 @@ public static function getLanguagePath(): string return 'LLL:EXT:' . self::EXTENSION_KEY . '/Resources/Private/Language/'; } - public static function registerConfig(): void + public static function registerExtLocalconfConfigConfig(): void { self::addCaching(); self::addContentElements(); - self::addPageTypes(); + self::addPageTypesExtLocalconf(); + } + + public static function registerExtTablesConfig(): void + { + self::addPageTypesExtTables(); } public static function getIconPath(): string @@ -59,25 +65,28 @@ private static function addContentElements(): void { $languagePath = self::getLanguagePath() . 'locallang_tca.xlf:tt_content'; - ExtensionManagementUtility::addPageTSConfig(' - mod.wizards.newContentElement.wizardItems.thuecat { - header = ' . $languagePath . '.group - show = * - elements { - thuecat_tourist_attraction{ - title = ' . $languagePath . '.thuecat_tourist_attraction - description = ' . $languagePath . '.thuecat_tourist_attraction.description - iconIdentifier = tt_content_thuecat_tourist_attraction - tt_content_defValues { - CType = thuecat_tourist_attraction + // TODO: typo3/cms-core:14.0 Remove this code block as CEs are auto registered. + if (version_compare(GeneralUtility::makeInstance(Typo3Version::class)->__toString(), '13.0', '<')) { + ExtensionManagementUtility::addPageTSConfig(' + mod.wizards.newContentElement.wizardItems.thuecat { + header = ' . $languagePath . '.group + show = * + elements { + thuecat_tourist_attraction{ + title = ' . $languagePath . '.thuecat_tourist_attraction + description = ' . $languagePath . '.thuecat_tourist_attraction.description + iconIdentifier = tt_content_thuecat_tourist_attraction + tt_content_defValues { + CType = thuecat_tourist_attraction + } } } } - } - '); + '); + } } - private static function addPageTypes(): void + private static function addPageTypesExtTables(): void { $registry = GeneralUtility::makeInstance(PageDoktypeRegistry::class); $registry->add( @@ -87,10 +96,16 @@ private static function addPageTypes(): void 'allowedTables' => '*', ] ); + } - ExtensionManagementUtility::addUserTSConfig( - "@import 'EXT:" . self::EXTENSION_KEY . "/Configuration/TSconfig/User/All.tsconfig'" - ); + private static function addPageTypesExtLocalconf(): void + { + // TODO: typo3/cms-core:14.0 Remove this code block as Configuration/user.tsconfig will be loaded since 13.x + if (version_compare(GeneralUtility::makeInstance(Typo3Version::class)->__toString(), '13.0', '<')) { + ExtensionManagementUtility::addUserTSConfig( + "@import 'EXT:" . self::EXTENSION_KEY . '/Configuration/user.tsconfig' + ); + } } private static function addCaching(): void diff --git a/Classes/Frontend/DataProcessing/ResolveEntities.php b/Classes/Frontend/DataProcessing/ResolveEntities.php index 3d7797af..252929ca 100644 --- a/Classes/Frontend/DataProcessing/ResolveEntities.php +++ b/Classes/Frontend/DataProcessing/ResolveEntities.php @@ -25,6 +25,7 @@ use TYPO3\CMS\Core\Database\Connection; use TYPO3\CMS\Core\Database\ConnectionPool; +use TYPO3\CMS\Core\Domain\Repository\PageRepository; use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Extbase\Persistence\Generic\Mapper\DataMapper; use TYPO3\CMS\Frontend\ContentObject\ContentObjectRenderer; @@ -79,6 +80,11 @@ private function resolveEntities(string $tableName, array $uids): array $rows = []; foreach ($queryBuilder->executeQuery()->iterateAssociative() as $row) { + // TODO: typo3/cms-core:14.0 Remove this condition, should always be an instance now. + if (!$this->tsfe->sys_page instanceof PageRepository) { + continue; + } + $row = $this->tsfe->sys_page->getLanguageOverlay($tableName, $row); if (is_array($row)) { $rows[] = $row; diff --git a/Configuration/FlexForm/ImportConfiguration/ContainsPlace.xml b/Configuration/FlexForm/ImportConfiguration/ContainsPlace.xml index 0218b0eb..fe23de10 100644 --- a/Configuration/FlexForm/ImportConfiguration/ContainsPlace.xml +++ b/Configuration/FlexForm/ImportConfiguration/ContainsPlace.xml @@ -5,29 +5,24 @@ - - LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:importConfiguration.containsPlace.sheetTitle - + LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:importConfiguration.containsPlace.sheetTitle array - - - - input - int,required - - + + + number + 1 + - - - LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:importConfiguration.containsPlace.containsPlaceId.description - - input - trim,required - - + + LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:importConfiguration.containsPlace.containsPlaceId.description + + input + trim + 1 + diff --git a/Configuration/FlexForm/ImportConfiguration/Static.xml b/Configuration/FlexForm/ImportConfiguration/Static.xml index 02c7732f..87403b06 100644 --- a/Configuration/FlexForm/ImportConfiguration/Static.xml +++ b/Configuration/FlexForm/ImportConfiguration/Static.xml @@ -5,19 +5,15 @@ - - LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:importConfiguration.static.sheetTitle - + LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:importConfiguration.static.sheetTitle array - - - - input - int,required - - + + + number + 1 + LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:importConfiguration.static.urls @@ -29,13 +25,12 @@ array - - - - input - required,trim - - + + + input + trim + 1 + diff --git a/Configuration/FlexForm/ImportConfiguration/SyncScope.xml b/Configuration/FlexForm/ImportConfiguration/SyncScope.xml index 59e715cd..689204f3 100644 --- a/Configuration/FlexForm/ImportConfiguration/SyncScope.xml +++ b/Configuration/FlexForm/ImportConfiguration/SyncScope.xml @@ -5,28 +5,23 @@ - - LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:importConfiguration.syncScope.sheetTitle - + LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:importConfiguration.syncScope.sheetTitle array - - - - input - int,required - - + + + number + 1 + - - - - input - trim,required - - + + + input + trim + 1 + diff --git a/Configuration/FlexForm/Pages/tourist_attraction.xml b/Configuration/FlexForm/Pages/tourist_attraction.xml index 317d6763..6ce5260f 100644 --- a/Configuration/FlexForm/Pages/tourist_attraction.xml +++ b/Configuration/FlexForm/Pages/tourist_attraction.xml @@ -5,23 +5,19 @@ - - LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:pages.tourist_attraction.sheetTitle - + LLL:EXT:thuecat/Resources/Private/Language/locallang_flexform.xlf:pages.tourist_attraction.sheetTitle array - - - - select - selectMultipleSideBySide - tx_thuecat_tourist_attraction - AND {#tx_thuecat_tourist_attraction}.{#sys_language_uid} IN (0,-1) - 1 - 1 - - + + + select + selectMultipleSideBySide + tx_thuecat_tourist_attraction + AND {#tx_thuecat_tourist_attraction}.{#sys_language_uid} IN (0,-1) + 1 + 1 + diff --git a/Configuration/TCA/Overrides/pages.php b/Configuration/TCA/Overrides/pages.php index 87b10268..dde2772b 100644 --- a/Configuration/TCA/Overrides/pages.php +++ b/Configuration/TCA/Overrides/pages.php @@ -25,7 +25,7 @@ 'type' => 'flex', 'ds_pointerField' => 'doktype', 'ds' => [ - 'default' => ' array input 48 ', + 'default' => ' array input 48 ', ], ], ], diff --git a/Configuration/TCA/Overrides/tt_content_tourist_attraction.php b/Configuration/TCA/Overrides/tt_content_tourist_attraction.php index 85b9c526..a385b4b7 100644 --- a/Configuration/TCA/Overrides/tt_content_tourist_attraction.php +++ b/Configuration/TCA/Overrides/tt_content_tourist_attraction.php @@ -60,7 +60,7 @@ [ 'label' => $languagePath, 'value' => $cType, - 'icon' => Extension::getIconPath() . 'tt_content_' . $cType . '.svg', + 'icon' => 'tt_content_' . $cType, 'group' => Extension::TCA_SELECT_GROUP_IDENTIFIER, ] ); diff --git a/Configuration/TCA/tx_thuecat_tourist_attraction.php b/Configuration/TCA/tx_thuecat_tourist_attraction.php index 8a93e9f5..3050f2dd 100644 --- a/Configuration/TCA/tx_thuecat_tourist_attraction.php +++ b/Configuration/TCA/tx_thuecat_tourist_attraction.php @@ -56,6 +56,20 @@ 'type' => 'passthrough', ], ], + 'disable' => [ + 'label' => 'LLL:EXT:core/Resources/Private/Language/locallang_general.xlf:LGL.enabled', + 'config' => [ + 'type' => 'check', + 'renderType' => 'checkboxToggle', + 'default' => 0, + 'items' => [ + [ + 'label' => '', + 'invertStateDisplay' => true, + ], + ], + ], + ], 'title' => [ 'label' => $languagePath . '.title', diff --git a/Configuration/TypoScript/Default/Setup.typoscript b/Configuration/TypoScript/Default/Setup.typoscript index b05171bb..28807d3a 100644 --- a/Configuration/TypoScript/Default/Setup.typoscript +++ b/Configuration/TypoScript/Default/Setup.typoscript @@ -5,10 +5,5 @@ module { tx_thuecat_import_configuration = 0 } } - view { - templateRootPaths { - 0 = EXT:thuecat/Resources/Private/Templates/ - } - } } } diff --git a/Configuration/TSconfig/User/All.tsconfig b/Configuration/user.tsconfig similarity index 100% rename from Configuration/TSconfig/User/All.tsconfig rename to Configuration/user.tsconfig diff --git a/Documentation/Changelog/3.1.0.rst b/Documentation/Changelog/3.1.0.rst new file mode 100644 index 00000000..b8ae2afd --- /dev/null +++ b/Documentation/Changelog/3.1.0.rst @@ -0,0 +1,28 @@ +3.1.0 +===== + +Breaking +-------- + +Nothing + +Features +-------- + +* Add TYPO3 v13 LTS Support. + +Fixes +----- + +Nothing + +Tasks +----- + +Nothing + +Deprecation +----------- + +Nothing + diff --git a/Documentation/Index.rst b/Documentation/Index.rst index 59bc1fa2..4a5ef2a2 100644 --- a/Documentation/Index.rst +++ b/Documentation/Index.rst @@ -20,4 +20,5 @@ Table of Contents Configuration Integration Changelog + Maintenance Sitemap diff --git a/Documentation/Settings.cfg b/Documentation/Settings.cfg deleted file mode 100644 index a392c73b..00000000 --- a/Documentation/Settings.cfg +++ /dev/null @@ -1,52 +0,0 @@ -[general] - -# ................................................................................. -# ... (required) title (displayed in left sidebar (desktop) or top panel (mobile) -# ................................................................................. -project = TYPO3 EXT:thuecat - -# ................................................................................. -# ... (recommended) version, displayed next to title (desktop) and in + + + + + + + diff --git a/Resources/Private/Templates/Backend/Configuration/Index.html b/Resources/Private/Templates/Backend/Configuration/Index.html index 160692e5..fd190220 100644 --- a/Resources/Private/Templates/Backend/Configuration/Index.html +++ b/Resources/Private/Templates/Backend/Configuration/Index.html @@ -54,7 +54,7 @@

{f:translate(id: 'module.organisations.headline')}

-
+
@@ -104,7 +104,7 @@

{f:translate(id: 'module.organisations.headline')}

-
+
diff --git a/Resources/Private/Templates/Backend/Import/Index.html b/Resources/Private/Templates/Backend/Import/Index.html index 14e12146..612e3c8f 100644 --- a/Resources/Private/Templates/Backend/Import/Index.html +++ b/Resources/Private/Templates/Backend/Import/Index.html @@ -24,7 +24,7 @@

{f:translate(id: 'module.imports.headline')}

-
+
diff --git a/Resources/Public/Icons/ModuleGroup.svg b/Resources/Public/Icons/ModuleGroup.svg index a8811731..3e8f40b8 100644 --- a/Resources/Public/Icons/ModuleGroup.svg +++ b/Resources/Public/Icons/ModuleGroup.svg @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/Tests/Acceptance/BackendConfigurationCest.php b/Tests/Acceptance/BackendConfigurationCest.php deleted file mode 100644 index 2b302a4a..00000000 --- a/Tests/Acceptance/BackendConfigurationCest.php +++ /dev/null @@ -1,70 +0,0 @@ - - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * as published by the Free Software Foundation; either version 2 - * 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 General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA - * 02110-1301, USA. - */ - -namespace WerkraumMedia\ThueCat\Tests\Acceptance; - -use WerkraumMedia\ThueCat\Tests\Acceptance\Support\AcceptanceTester; - -class BackendConfigurationCest -{ - public function _before(AcceptanceTester $I): void - { - $I->amOnPage('/typo3'); - $I->submitForm('#typo3-login-form', [ - 'username' => 'admin', - 'p_field' => 'password', - ]); - $I->waitForText('New TYPO3 site'); - } - - public function showsIndex(AcceptanceTester $I): void - { - $I->click('Configurations'); - $I->switchToContentFrame(); - $I->see('ThüCAT - Configurations'); - $I->see('Example Configuration'); - $I->see('Please provide an import configuration and trigger import to create an organisation.'); - } - - public function allowsToImportExistingConfiguration(AcceptanceTester $I): void - { - $I->click('Configurations'); - $I->switchToContentFrame(); - $I->see('Example Configuration'); - $I->see('Never'); - $I->click('Import based on import configuration'); - - $I->see('Import finished'); - $I->see('Imported configuration "Example Configuration".'); - $I->see('Tourist-Information Schmalkalden'); - } - - public function showsExecutedImport(AcceptanceTester $I): void - { - $I->click('Imports'); - $I->switchToContentFrame(); - $I->see('ThüCAT - Imports'); - - $I->see('Example Configuration'); - } -} diff --git a/Tests/Acceptance/Data/.gitkeep b/Tests/Acceptance/Data/.gitkeep deleted file mode 100644 index e69de29b..00000000 diff --git a/Tests/Acceptance/Data/BasicDatabase.php b/Tests/Acceptance/Data/BasicDatabase.php deleted file mode 100644 index 88e9cae8..00000000 --- a/Tests/Acceptance/Data/BasicDatabase.php +++ /dev/null @@ -1,87 +0,0 @@ - [ - 0 => [ - 'uid' => '1', - 'pid' => '0', - 'tstamp' => '1366642540', - 'username' => 'admin', - 'password' => '$1$tCrlLajZ$C0sikFQQ3SWaFAZ1Me0Z/1', - 'admin' => '1', - 'disable' => '0', - 'starttime' => '0', - 'endtime' => '0', - 'options' => '0', - 'crdate' => '1366642540', - 'workspace_perms' => '1', - 'deleted' => '0', - 'TSconfig' => null, - 'lastlogin' => '1371033743', - 'workspace_id' => '0', - ], - ], - 'pages' => [ - 0 => [ - 'uid' => '1', - 'pid' => '0', - 'doktype' => PageRepository::DOKTYPE_DEFAULT, - 'slug' => '/', - 'title' => 'Rootpage', - ], - 1 => [ - 'uid' => '2', - 'pid' => '1', - 'doktype' => PageRepository::DOKTYPE_SYSFOLDER, - 'slug' => '/storage', - 'title' => 'Storage', - ], - ], - 'tx_thuecat_import_configuration' => [ - 0 => [ - 'uid' => '1', - 'pid' => '2', - 'title' => 'Example Configuration', - 'type' => 'static', - 'configuration' => ' - - - - - 2 - - - - - - - - https://thuecat.org/resources/644315157726-jmww - - - - 0 - - - - - - https://thuecat.org/resources/072778761562-kwah - - - - 0 - - - - - - - ', - ], - ], -]; diff --git a/Tests/Acceptance/Data/Sites/default/config.yaml b/Tests/Acceptance/Data/Sites/default/config.yaml deleted file mode 100644 index f33946fc..00000000 --- a/Tests/Acceptance/Data/Sites/default/config.yaml +++ /dev/null @@ -1,47 +0,0 @@ -base: 'http://localhost:8080' -languages: - - - title: Deutsch - enabled: true - base: / - typo3Language: de - locale: de_DE.UTF-8 - iso-639-1: de - navigationTitle: Deutsch - hreflang: de-DE - direction: '' - flag: de - websiteTitle: '' - languageId: 0 - - - title: English - enabled: true - base: /en - typo3Language: default - locale: en_GB.UTF-8 - iso-639-1: en - websiteTitle: '' - navigationTitle: English - hreflang: en-GB - direction: '' - flag: gb - languageId: 1 - fallbackType: strict - fallbacks: '0' - - - title: French - enabled: true - base: /fr/ - typo3Language: fr - locale: fr_FR.ytf8 - iso-639-1: fr - websiteTitle: '' - navigationTitle: '' - hreflang: fr-FR - direction: '' - fallbackType: strict - fallbacks: '1,0' - flag: fr - languageId: 2 -rootPageId: 1 -websiteTitle: 'Example Website' diff --git a/Tests/Acceptance/Support/AcceptanceTester.php b/Tests/Acceptance/Support/AcceptanceTester.php deleted file mode 100644 index 5ebb41e9..00000000 --- a/Tests/Acceptance/Support/AcceptanceTester.php +++ /dev/null @@ -1,54 +0,0 @@ - - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * as published by the Free Software Foundation; either version 2 - * 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 General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA - * 02110-1301, USA. - */ - -namespace WerkraumMedia\ThueCat\Tests\Acceptance\Support; - -use Codeception\Actor; -use TYPO3\TestingFramework\Core\Acceptance\Step\FrameSteps; -use WerkraumMedia\ThueCat\Tests\Acceptance\Support\_generated\AcceptanceTesterActions; - -/** - * Inherited Methods - * - * @method void wantToTest($text) - * @method void wantTo($text) - * @method void execute($callable) - * @method void expectTo($prediction) - * @method void expect($prediction) - * @method void amGoingTo($argumentation) - * @method void am($role) - * @method void lookForwardTo($achieveValue) - * @method void comment($description) - * @method void pause() - * - * @SuppressWarnings(PHPMD) -*/ -class AcceptanceTester extends Actor -{ - use FrameSteps; - use AcceptanceTesterActions; - - /** - * Define custom actions here - */ -} diff --git a/Tests/Acceptance/Support/Environment.php b/Tests/Acceptance/Support/Environment.php deleted file mode 100644 index 64877100..00000000 --- a/Tests/Acceptance/Support/Environment.php +++ /dev/null @@ -1,60 +0,0 @@ - - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * as published by the Free Software Foundation; either version 2 - * 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 General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA - * 02110-1301, USA. - */ - -namespace WerkraumMedia\ThueCat\Tests\Acceptance\Support; - -use Codappix\Typo3PhpDatasets\TestingFramework; -use Codeception\Event\SuiteEvent; -use TYPO3\TestingFramework\Core\Acceptance\Extension\BackendEnvironment; - -/** - * Load various core extensions and extension under test. - */ -class Environment extends BackendEnvironment -{ - use TestingFramework; - - protected $localConfig = [ - 'coreExtensionsToLoad' => [ - 'install', - 'core', - 'backend', - 'extbase', - 'frontend', - 'fluid', - ], - 'testExtensionsToLoad' => [ - 'werkraummedia/thuecat', - ], - 'pathsToLinkInTestInstance' => [ - '/../../../../../../Tests/Acceptance/Data/Sites/' => 'typo3conf/sites', - ], - ]; - - public function bootstrapTypo3Environment(SuiteEvent $suiteEvent) - { - parent::bootstrapTypo3Environment($suiteEvent); - - $this->importPHPDataSet(__DIR__ . '/../Data/BasicDatabase.php'); - } -} diff --git a/Tests/Acceptance/_output/.gitignore b/Tests/Acceptance/_output/.gitignore deleted file mode 100644 index c96a04f0..00000000 --- a/Tests/Acceptance/_output/.gitignore +++ /dev/null @@ -1,2 +0,0 @@ -* -!.gitignore \ No newline at end of file diff --git a/Tests/Functional/AbstractImportTestCase.php b/Tests/Functional/AbstractImportTestCase.php index a52d829c..9138ded0 100644 --- a/Tests/Functional/AbstractImportTestCase.php +++ b/Tests/Functional/AbstractImportTestCase.php @@ -28,7 +28,10 @@ use TypeError; use TYPO3\CMS\Core\Context\Context; use TYPO3\CMS\Core\Context\DateTimeAspect; +use TYPO3\CMS\Core\Core\SystemEnvironmentBuilder; +use TYPO3\CMS\Core\Http\ServerRequest; use TYPO3\CMS\Core\Localization\LanguageServiceFactory; +use TYPO3\CMS\Extbase\Configuration\ConfigurationManagerInterface; abstract class AbstractImportTestCase extends \TYPO3\TestingFramework\Core\Functional\FunctionalTestCase { @@ -135,4 +138,21 @@ protected function setDateAspect(DateTimeImmutable $dateTime): void $aspect = new DateTimeAspect($dateTime); $context->setAspect('date', $aspect); } + + /** + * Workaround ConfigurationManager requiring request + */ + protected function workaroundExtbaseConfiguration(): void + { + $fakeRequest = new ServerRequest(); + $fakeRequest = $fakeRequest->withAttribute('applicationType', SystemEnvironmentBuilder::REQUESTTYPE_BE); + $configurationManager = $this->get(ConfigurationManagerInterface::class); + + // TODO: typo3/cms-core:14.0 Remove condition, the method should always be available + if (method_exists($configurationManager, 'setRequest') === false) { + return; + } + + $configurationManager->setRequest($fakeRequest); + } } diff --git a/Tests/Functional/ImportConfigurationCommandTest.php b/Tests/Functional/ImportConfigurationCommandTest.php index 5289f2b4..004219e1 100644 --- a/Tests/Functional/ImportConfigurationCommandTest.php +++ b/Tests/Functional/ImportConfigurationCommandTest.php @@ -35,6 +35,8 @@ final class ImportConfigurationCommandTest extends AbstractImportTestCase #[Test] public function canImport(): void { + $this->workaroundExtbaseConfiguration(); + $subject = $this->getContainer()->get(ImportConfigurationCommand::class); self::assertInstanceOf(Command::class, $subject); @@ -50,6 +52,8 @@ public function canImport(): void #[Test] public function throwsExceptionOnNoneExistingConfiguration(): void { + $this->workaroundExtbaseConfiguration(); + $subject = $this->getContainer()->get(ImportConfigurationCommand::class); self::assertInstanceOf(Command::class, $subject); diff --git a/Tests/Functional/ImportTest.php b/Tests/Functional/ImportTest.php index 5b5a24a2..2224ca2a 100644 --- a/Tests/Functional/ImportTest.php +++ b/Tests/Functional/ImportTest.php @@ -23,7 +23,11 @@ * 02110-1301, USA. */ +use DateTimeImmutable; +use DateTimeZone; use PHPUnit\Framework\Attributes\Test; +use TYPO3\CMS\Core\Context\Context; +use TYPO3\CMS\Core\Context\DateTimeAspect; use WerkraumMedia\ThueCat\Domain\Import\ImportConfiguration; use WerkraumMedia\ThueCat\Domain\Import\Importer; use WerkraumMedia\ThueCat\Domain\Repository\Backend\ImportConfigurationRepository; @@ -387,6 +391,15 @@ public function importsWithBrokenOpeningHour(): void private function importConfiguration(): void { + $this->workaroundExtbaseConfiguration(); + + $this->get(Context::class)->setAspect( + 'date', + new DateTimeAspect( + new DateTimeImmutable('2024-03-03 00:00:00', new DateTimeZone('UTC')) + ) + ); + $configuration = $this->get(ImportConfigurationRepository::class)->findByUid(1); self::assertInstanceOf(ImportConfiguration::class, $configuration); $this->get(Importer::class)->importConfiguration($configuration); diff --git a/codeception.dist.yml b/codeception.dist.yml deleted file mode 100644 index f9f7b80e..00000000 --- a/codeception.dist.yml +++ /dev/null @@ -1,44 +0,0 @@ -namespace: 'WerkraumMedia\ThueCat\Tests\Acceptance\Support' - -paths: - tests: 'Tests/Acceptance' - data: 'Tests/Acceptance/Data' - output: '.Build/web/typo3temp/var/tests/AcceptanceReports' - support: 'Tests/Acceptance/Support' - -settings: - debug: true - -extensions: - enabled: - - 'Codeception\Extension\RunFailed' - -suites: - acceptance: - actor: 'AcceptanceTester' - path: . - extensions: - enabled: - - 'Codeception\Extension\RunProcess': - - 'geckodriver > .Build/web/typo3temp/var/tests/AcceptanceReports/geckodriver.log 2>&1' - - 'TYPO3_PATH_APP="$INSTANCE_PATH" TYPO3_PATH_ROOT="$INSTANCE_PATH" php -S 127.0.0.1:8080 -t "$INSTANCE_PATH" > .Build/web/typo3temp/var/tests/AcceptanceReports/php.log 2>&1' - - 'WerkraumMedia\ThueCat\Tests\Acceptance\Support\Environment': - 'typo3DatabaseUsername': 'testing' - 'typo3DatabasePassword': 'testing' - - modules: - enabled: - - WebDriver: - url: 'http://localhost:8080' - browser: 'firefox' - restart: true - path: '' - wait: 5 - capabilities: - moz:firefoxOptions: - args: - - '-headless' - - '\TYPO3\TestingFramework\Core\Acceptance\Helper\Acceptance' - step_decorators: - - 'Codeception\Step\Retry' - diff --git a/composer.json b/composer.json index 41ea9ae4..0185bfce 100644 --- a/composer.json +++ b/composer.json @@ -42,28 +42,27 @@ "psr/http-factory": "^1.0", "psr/http-message": "^2.0", "psr/log": "^2.0 || ^3.0", - "symfony/console": "^6.4", - "symfony/dependency-injection": "^6.4", - "symfony/property-access": "^6.4", - "symfony/property-info": "^6.4", + "symfony/console": "^6.4 || ^7.1", + "symfony/dependency-injection": "^6.4 || ^7.1", + "symfony/property-access": "^6.4 || ^7.1", + "symfony/property-info": "^6.4 || ^7.1", "symfony/serializer": "^6.4", - "typo3/cms-backend": "^12.4", - "typo3/cms-core": "^12.4", - "typo3/cms-extbase": "^12.4", - "typo3/cms-frontend": "^12.4", - "typo3/cms-install": "^12.4" + "typo3/cms-backend": "^12.4 || ^13.4", + "typo3/cms-core": "^12.4 || ^13.4", + "typo3/cms-extbase": "^12.4 || ^13.4", + "typo3/cms-frontend": "^12.4 || ^13.4", + "typo3/cms-install": "^12.4 || ^13.4" }, "require-dev": { "codappix/typo3-php-datasets": "^1.4", - "codeception/codeception": "^5.0", - "codeception/module-webdriver": "^4.0", "friendsofphp/php-cs-fixer": "^3.40", "phpstan/extension-installer": "^1.1", "phpstan/phpstan": "1.10.46", "phpstan/phpstan-phpunit": "^1.3", - "phpunit/phpunit": "^10.4", + "phpunit/phpunit": "^10.4 || ^11.4", "saschaegerer/phpstan-typo3": "^1.9", - "typo3/cms-fluid-styled-content": "^12.4", + "staabm/phpstan-todo-by": "^0.1.32", + "typo3/cms-fluid-styled-content": "^12.4 || ^13.4", "typo3/testing-framework": "^8.0" }, "config": { diff --git a/ext_emconf.php b/ext_emconf.php index bc50bb69..fd4bc2f6 100644 --- a/ext_emconf.php +++ b/ext_emconf.php @@ -10,7 +10,7 @@ 'author' => 'Daniel Siepmann', 'author_email' => 'coding@daniel-siepmann.de', 'author_company' => '', - 'version' => '3.0.1', + 'version' => '3.1.0', 'constraints' => [ 'depends' => [ 'core' => '', diff --git a/ext_localconf.php b/ext_localconf.php index 537e04fd..3e5ed651 100644 --- a/ext_localconf.php +++ b/ext_localconf.php @@ -8,7 +8,7 @@ defined('TYPO3') or die(); -Extension::registerConfig(); +Extension::registerExtLocalconfConfigConfig(); (static function (string $extensionKey) { ExtensionManagementUtility::addTypoScriptSetup( diff --git a/ext_tables.php b/ext_tables.php new file mode 100644 index 00000000..3808ffd2 --- /dev/null +++ b/ext_tables.php @@ -0,0 +1,5 @@ +, string given\\.$#" count: 1 @@ -234,8 +224,3 @@ parameters: message: "#^Parameter \\#1 \\$mods of method WerkraumMedia\\\\ThueCat\\\\Updates\\\\BackendModuleUserPermission\\:\\:updateMods\\(\\) expects string, mixed given\\.$#" count: 1 path: Classes/Updates/BackendModuleUserPermission.php - - - - message: "#^Method WerkraumMedia\\\\ThueCat\\\\Tests\\\\Acceptance\\\\Support\\\\Environment\\:\\:bootstrapTypo3Environment\\(\\) has no return type specified\\.$#" - count: 1 - path: Tests/Acceptance/Support/Environment.php diff --git a/shell.nix b/shell.nix index 9ca6aee8..d6a860cd 100644 --- a/shell.nix +++ b/shell.nix @@ -49,26 +49,6 @@ let ''; }; - projectTestAcceptance = pkgs.writeShellApplication { - name = "project-test-acceptance"; - runtimeInputs = [ - projectInstall - pkgs.sqlite - pkgs.firefox - pkgs.geckodriver - php - ]; - text = '' - project-install - - export INSTANCE_PATH="$PROJECT_ROOT/.Build/web/typo3temp/var/tests/acceptance" - export typo3DatabaseDriver=pdo_sqlite - - mkdir -p "$INSTANCE_PATH" - ./vendor/bin/codecept run - ''; - }; - in pkgs.mkShell { name = "TYPO3 Extension ThüCAT"; buildInputs = [ @@ -77,7 +57,6 @@ in pkgs.mkShell { projectInstall projectCgl projectCglFix - projectTestAcceptance ]; shellHook = '' From bf22aa0e2415f42127b12376efe1ab503fb30aed Mon Sep 17 00:00:00 2001 From: Daniel Siepmann Date: Wed, 6 Nov 2024 08:25:00 +0100 Subject: [PATCH 2/3] Remove content elements --- Classes/Extension.php | 26 --- Configuration/TCA/Overrides/sys_template.php | 5 - Configuration/TCA/Overrides/tt_content.php | 21 +-- .../tt_content_tourist_attraction.php | 71 -------- .../Rendering/TouristAttraction.typoscript | 13 -- .../Rendering/_base.typoscript | 9 - .../ContentElements/setup.typoscript | 1 - .../Changelog/{3.1.0.rst => 4.0.0.rst} | 7 +- Resources/Private/Language/locallang_tca.xlf | 6 - .../ContentElement/TouristAttraction.html | 165 ------------------ .../ContentElement/Accessibility.html | 0 .../Partials}/ContentElement/Address.html | 0 .../Partials}/ContentElement/Digital.html | 0 .../Partials}/ContentElement/Languages.html | 0 .../Partials}/ContentElement/Museum.html | 0 .../Partials}/ContentElement/Offers.html | 0 .../Partials}/ContentElement/Opening.html | 0 .../Partials}/ContentElement/Parking.html | 0 .../Partials}/ContentElement/Payment.html | 0 .../Partials}/ContentElement/Photography.html | 0 .../Partials}/ContentElement/Sanitation.html | 0 .../Partials}/ContentElement/Service.html | 0 .../Partials}/ContentElement/Traffic.html | 0 .../ContentElement/TouristAttraction.html | 0 .../Frontend/Extensions/example/composer.json | 28 +++ .../Fixtures/Frontend/Rendering.typoscript | 18 +- .../ContentElement/Accessibility.html | 136 --------------- .../Partials/ContentElement/Address.html | 24 --- .../Partials/ContentElement/Digital.html | 9 - .../Partials/ContentElement/Languages.html | 9 - .../Partials/ContentElement/Museum.html | 9 - .../Partials/ContentElement/Offers.html | 80 --------- .../Partials/ContentElement/Opening.html | 27 --- .../Partials/ContentElement/Parking.html | 22 --- .../Partials/ContentElement/Payment.html | 9 - .../Partials/ContentElement/Photography.html | 9 - .../Partials/ContentElement/Sanitation.html | 9 - .../Partials/ContentElement/Service.html | 5 - .../Partials/ContentElement/Traffic.html | 9 - Tests/Functional/FrontendTest.php | 1 + composer.json | 6 + ext_emconf.php | 2 +- ext_tables.php | 2 + shell.nix | 6 +- 44 files changed, 67 insertions(+), 677 deletions(-) delete mode 100644 Configuration/TCA/Overrides/tt_content_tourist_attraction.php delete mode 100644 Configuration/TypoScript/ContentElements/Rendering/TouristAttraction.typoscript delete mode 100644 Configuration/TypoScript/ContentElements/Rendering/_base.typoscript delete mode 100644 Configuration/TypoScript/ContentElements/setup.typoscript rename Documentation/Changelog/{3.1.0.rst => 4.0.0.rst} (51%) delete mode 100644 Resources/Private/Templates/Frontend/ContentElement/TouristAttraction.html rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Accessibility.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Address.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Digital.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Languages.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Museum.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Offers.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Opening.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Parking.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Payment.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Photography.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Sanitation.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Service.html (100%) rename {Resources/Private/Partials/Frontend => Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials}/ContentElement/Traffic.html (100%) rename Tests/Functional/Fixtures/Frontend/{ => Extensions/example}/Resources/Private/Templates/ContentElement/TouristAttraction.html (100%) create mode 100644 Tests/Functional/Fixtures/Frontend/Extensions/example/composer.json delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Accessibility.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Address.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Digital.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Languages.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Museum.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Offers.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Opening.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Parking.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Payment.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Photography.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Sanitation.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Service.html delete mode 100644 Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Traffic.html diff --git a/Classes/Extension.php b/Classes/Extension.php index 42f2c885..426b12f7 100644 --- a/Classes/Extension.php +++ b/Classes/Extension.php @@ -47,7 +47,6 @@ public static function getLanguagePath(): string public static function registerExtLocalconfConfigConfig(): void { self::addCaching(); - self::addContentElements(); self::addPageTypesExtLocalconf(); } @@ -61,31 +60,6 @@ public static function getIconPath(): string return 'EXT:' . self::EXTENSION_KEY . '/Resources/Public/Icons/'; } - private static function addContentElements(): void - { - $languagePath = self::getLanguagePath() . 'locallang_tca.xlf:tt_content'; - - // TODO: typo3/cms-core:14.0 Remove this code block as CEs are auto registered. - if (version_compare(GeneralUtility::makeInstance(Typo3Version::class)->__toString(), '13.0', '<')) { - ExtensionManagementUtility::addPageTSConfig(' - mod.wizards.newContentElement.wizardItems.thuecat { - header = ' . $languagePath . '.group - show = * - elements { - thuecat_tourist_attraction{ - title = ' . $languagePath . '.thuecat_tourist_attraction - description = ' . $languagePath . '.thuecat_tourist_attraction.description - iconIdentifier = tt_content_thuecat_tourist_attraction - tt_content_defValues { - CType = thuecat_tourist_attraction - } - } - } - } - '); - } - } - private static function addPageTypesExtTables(): void { $registry = GeneralUtility::makeInstance(PageDoktypeRegistry::class); diff --git a/Configuration/TCA/Overrides/sys_template.php b/Configuration/TCA/Overrides/sys_template.php index 9f6c0c77..4f9cfb25 100644 --- a/Configuration/TCA/Overrides/sys_template.php +++ b/Configuration/TCA/Overrides/sys_template.php @@ -8,11 +8,6 @@ defined('TYPO3') or die(); (static function (string $extensionKey, string $tableName) { - ExtensionManagementUtility::addStaticFile( - $extensionKey, - 'Configuration/TypoScript/ContentElements', - 'ThüCAT - Content Elements' - ); ExtensionManagementUtility::addStaticFile( $extensionKey, 'Configuration/TypoScript/PageTypes', diff --git a/Configuration/TCA/Overrides/tt_content.php b/Configuration/TCA/Overrides/tt_content.php index 19374c65..f1a0d57d 100644 --- a/Configuration/TCA/Overrides/tt_content.php +++ b/Configuration/TCA/Overrides/tt_content.php @@ -5,19 +5,10 @@ use TYPO3\CMS\Core\Utility\ExtensionManagementUtility; use WerkraumMedia\ThueCat\Extension; -defined('TYPO3') or die(); - -(static function (string $extensionKey, string $tableName) { - $languagePath = Extension::getLanguagePath() - . 'locallang_tca.xlf:' . $tableName; - - ExtensionManagementUtility::addTcaSelectItemGroup( - $tableName, - 'CType', - Extension::TCA_SELECT_GROUP_IDENTIFIER, - $languagePath . '.group' - ); -})( - Extension::EXTENSION_KEY, - 'tt_content' +ExtensionManagementUtility::addTcaSelectItemGroup( + 'tt_content', + 'CType', + Extension::TCA_SELECT_GROUP_IDENTIFIER, + Extension::getLanguagePath() . 'locallang_tca.xlf:tt_content.group', + 'bottom', ); diff --git a/Configuration/TCA/Overrides/tt_content_tourist_attraction.php b/Configuration/TCA/Overrides/tt_content_tourist_attraction.php deleted file mode 100644 index a385b4b7..00000000 --- a/Configuration/TCA/Overrides/tt_content_tourist_attraction.php +++ /dev/null @@ -1,71 +0,0 @@ - [ - 'typeicon_classes' => [ - $cType => 'tt_content_' . $cType, - ], - ], - 'types' => [ - $cType => [ - 'showitem' => '--div--;LLL:EXT:core/Resources/Private/Language/Form/locallang_tabs.xlf:general,' - . '--palette--;;general,' - . '--palette--;;headers,' - . 'records,' - . '--div--;LLL:EXT:frontend/Resources/Private/Language/locallang_ttc.xlf:tabs.appearance,' - . '--palette--;;frames,' - . '--palette--;;appearanceLinks,' - . '--div--;LLL:EXT:core/Resources/Private/Language/Form/locallang_tabs.xlf:language,' - . '--palette--;;language,' - . '--div--;LLL:EXT:core/Resources/Private/Language/Form/locallang_tabs.xlf:access,' - . '--palette--;;hidden,' - . '--palette--;;access,' - . '--div--;LLL:EXT:core/Resources/Private/Language/Form/locallang_tabs.xlf:categories,' - . '--div--;LLL:EXT:core/Resources/Private/Language/locallang_tca.xlf:sys_category.tabs.category,' - . 'categories,' - . '--div--;LLL:EXT:core/Resources/Private/Language/Form/locallang_tabs.xlf:notes,' - . 'rowDescription,' - . '--div--;LLL:EXT:core/Resources/Private/Language/Form/locallang_tabs.xlf:extended', - 'columnsOverrides' => [ - 'records' => [ - 'config' => [ - 'allowed' => 'tx_thuecat_tourist_attraction', - 'suggestOptions' => [ - 'tx_thuecat_tourist_attraction' => [ - 'addWhere' => 'sys_language_uid in (0,-1)', - ], - ], - ], - ], - ], - ], - ], - ]); - - ExtensionManagementUtility::addTcaSelectItem( - $tableName, - 'CType', - [ - 'label' => $languagePath, - 'value' => $cType, - 'icon' => 'tt_content_' . $cType, - 'group' => Extension::TCA_SELECT_GROUP_IDENTIFIER, - ] - ); -})( - Extension::EXTENSION_KEY, - 'tt_content', - 'thuecat_tourist_attraction' -); diff --git a/Configuration/TypoScript/ContentElements/Rendering/TouristAttraction.typoscript b/Configuration/TypoScript/ContentElements/Rendering/TouristAttraction.typoscript deleted file mode 100644 index 67be9f92..00000000 --- a/Configuration/TypoScript/ContentElements/Rendering/TouristAttraction.typoscript +++ /dev/null @@ -1,13 +0,0 @@ -tt_content { - thuecat_tourist_attraction =< lib.thuecatContentElement - thuecat_tourist_attraction { - templateName = TouristAttraction - dataProcessing { - 10 = WerkraumMedia\ThueCat\Frontend\DataProcessing\ResolveEntities - 10 { - table = tx_thuecat_tourist_attraction - uids.data = field:records - } - } - } -} diff --git a/Configuration/TypoScript/ContentElements/Rendering/_base.typoscript b/Configuration/TypoScript/ContentElements/Rendering/_base.typoscript deleted file mode 100644 index 97108d65..00000000 --- a/Configuration/TypoScript/ContentElements/Rendering/_base.typoscript +++ /dev/null @@ -1,9 +0,0 @@ -lib.thuecatContentElement =< lib.contentElement -lib.thuecatContentElement { - partialRootPaths { - 0 = EXT:thuecat/Resources/Private/Partials/Frontend/ContentElement/ - } - templateRootPaths { - 0 = EXT:thuecat/Resources/Private/Templates/Frontend/ContentElement/ - } -} diff --git a/Configuration/TypoScript/ContentElements/setup.typoscript b/Configuration/TypoScript/ContentElements/setup.typoscript deleted file mode 100644 index 3c8b4e50..00000000 --- a/Configuration/TypoScript/ContentElements/setup.typoscript +++ /dev/null @@ -1 +0,0 @@ -@import 'EXT:thuecat/Configuration/TypoScript/ContentElements/Rendering/*.typoscript' diff --git a/Documentation/Changelog/3.1.0.rst b/Documentation/Changelog/4.0.0.rst similarity index 51% rename from Documentation/Changelog/3.1.0.rst rename to Documentation/Changelog/4.0.0.rst index b8ae2afd..1e9449aa 100644 --- a/Documentation/Changelog/3.1.0.rst +++ b/Documentation/Changelog/4.0.0.rst @@ -1,10 +1,13 @@ -3.1.0 +4.0.0 ===== Breaking -------- -Nothing +* Removed content element. + + No Content element is provided any longer. + We recommend to build your own tailored content elements instead. Features -------- diff --git a/Resources/Private/Language/locallang_tca.xlf b/Resources/Private/Language/locallang_tca.xlf index 8a9bca37..b01711aa 100644 --- a/Resources/Private/Language/locallang_tca.xlf +++ b/Resources/Private/Language/locallang_tca.xlf @@ -311,12 +311,6 @@ ThüCAT - - Tourist Attraction - - - Renders selected tourist attractions - ThüCAT diff --git a/Resources/Private/Templates/Frontend/ContentElement/TouristAttraction.html b/Resources/Private/Templates/Frontend/ContentElement/TouristAttraction.html deleted file mode 100644 index b75f2feb..00000000 --- a/Resources/Private/Templates/Frontend/ContentElement/TouristAttraction.html +++ /dev/null @@ -1,165 +0,0 @@ - -
- -
-
- {entity.slogan} -

{entity.title} ({entity.town.title})

- {entity.description -> f:format.html()} -
-
- - - -

ⓒ {entity.media.mainImage.author}

-
-
-
-
-
- - -
- - Address -

{f:translate(id: 'content.address', extensionName: 'Thuecat')}

- {f:render(partial: 'Address', arguments: {address: entity.address, url: entity.url})} -
- -
- Address -

{f:translate(id: 'content.distanceToPublicTransport', extensionName: 'Thuecat')}

-

- {entity.distanceToPublicTransport.value} {f:translate(id: 'content.unit.{entity.distanceToPublicTransport.unit}', default: entity.distanceToPublicTransport.unit, extensionName: 'Thuecat')} -

-
-
-
-
- - Address -

{f:translate(id: 'content.parkingFacilitiesNearBy', extensionName: 'Thuecat')}

- {f:render(partial: 'Parking', arguments: {parkingFacilitiesNearBy: entity.parkingFacilitiesNearBy})} -
-
-
- -
- - Address -

{f:translate(id: 'content.address', extensionName: 'Thuecat')}

- {f:render(partial: 'Address', arguments: {address: entity.address, url: entity.url})} -
-
-
- - Address -

{f:translate(id: 'content.distanceToPublicTransport', extensionName: 'Thuecat')}

-

- {entity.distanceToPublicTransport.value} {f:translate(id: 'content.unit.{entity.distanceToPublicTransport.unit}', default: entity.distanceToPublicTransport.unit, extensionName: 'Thuecat')} -

-
-
-
- - Address -

{f:translate(id: 'content.parkingFacilitiesNearBy', extensionName: 'Thuecat')}

- {f:render(partial: 'Parking', arguments: {parkingFacilitiesNearBy: entity.parkingFacilitiesNearBy})} -
-
-
-
-
-
- -
-

{f:translate(id: 'content.generalInformation', extensionName: 'Thuecat')}

-

- - {f:render(partial: 'Service', arguments: {otherServices: entity.otherServices})} - - {f:translate(id: 'content.petsAllowed.{entity.petsAllowed}', default: entity.petsAllowed, extensionName: 'Thuecat')} - {f:translate(id: 'content.isAccessibleForFree.{entity.isAccessibleForFree}', default: entity.isAccessibleForFree, extensionName: 'Thuecat')} - {f:translate(id: 'content.publicAccess.{entity.publicAccess}', default: entity.publicAccess, extensionName: 'Thuecat')} - {f:translate(id: 'content.accessibilitySpecification.certificationStatus.{entity.accessibilitySpecification.certificationStatus}', default: entity.accessibilitySpecification.certificationStatus, extensionName: 'Thuecat')} -

- -

{f:translate(id: 'content.museum', extensionName: 'Thuecat')}

- {f:render(partial: 'Museum', arguments: {museumServices: entity.museumServices})} -
- -

{f:translate(id: 'content.digital', extensionName: 'Thuecat')}

- {f:render(partial: 'Digital', arguments: {digitalOffer: entity.digitalOffer})} -
- -

{f:translate(id: 'content.traffic', extensionName: 'Thuecat')}

- {f:render(partial: 'Traffic', arguments: {trafficInfrastructures: entity.trafficInfrastructures})} -
- -

{f:translate(id: 'content.payment', extensionName: 'Thuecat')}

- {f:render(partial: 'Payment', arguments: {paymentAccepted: entity.paymentAccepted})} -
- -

{f:translate(id: 'content.languages', extensionName: 'Thuecat')}

- {f:render(partial: 'Payment', arguments: {availableLanguages: entity.availableLanguages})} -
- -

{f:translate(id: 'content.sanitation', extensionName: 'Thuecat')}

- sanitation - {f:render(partial: 'Sanitation', arguments: {sanitation: entity.sanitation})} -
- -

{f:translate(id: 'content.photography', extensionName: 'Thuecat')}

- {f:render(partial: 'Photography', arguments: {photography: entity.photography})} -
- -

{f:translate(id: 'content.construction', extensionName: 'Thuecat')}

-

- {entity.startOfConstruction} -

-
- -

{f:translate(id: 'content.architecture', extensionName: 'Thuecat')}

-

- - {f:translate(id: 'content.architecturalStyle.{style}', default: style, extensionName: 'Thuecat')} - -

-
-
-
- -
-

{f:translate(id: 'content.openingHours', extensionName: 'Thuecat')}

- {f:render(partial: 'Opening', arguments: {openingHours: entity.openingHours})} -
-
- -
-

{f:translate(id: 'content.specialOpeningHours', extensionName: 'Thuecat')}

- {f:render(partial: 'Opening', arguments: {openingHours: entity.specialOpeningHours})} -
-
-
- -
- -
-

{f:translate(id: 'content.offers', extensionName: 'Thuecat')}

- {f:render(partial: 'Offers', arguments: {offers: entity.offers, uid: entity.uid})} -
-
- -
-

{f:translate(id: 'content.accessibility', extensionName: 'Thuecat')}

- {f:render(partial: 'Accessibility', arguments: {specification: entity.accessibilitySpecification, uid: entity.uid})} -
-
-
-
-
- diff --git a/Resources/Private/Partials/Frontend/ContentElement/Accessibility.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Accessibility.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Accessibility.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Accessibility.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Address.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Address.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Address.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Address.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Digital.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Digital.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Digital.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Digital.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Languages.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Languages.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Languages.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Languages.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Museum.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Museum.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Museum.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Museum.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Offers.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Offers.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Offers.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Offers.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Opening.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Opening.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Opening.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Opening.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Parking.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Parking.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Parking.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Parking.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Payment.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Payment.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Payment.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Payment.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Photography.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Photography.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Photography.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Photography.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Sanitation.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Sanitation.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Sanitation.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Sanitation.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Service.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Service.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Service.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Service.html diff --git a/Resources/Private/Partials/Frontend/ContentElement/Traffic.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Traffic.html similarity index 100% rename from Resources/Private/Partials/Frontend/ContentElement/Traffic.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Partials/ContentElement/Traffic.html diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Templates/ContentElement/TouristAttraction.html b/Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Templates/ContentElement/TouristAttraction.html similarity index 100% rename from Tests/Functional/Fixtures/Frontend/Resources/Private/Templates/ContentElement/TouristAttraction.html rename to Tests/Functional/Fixtures/Frontend/Extensions/example/Resources/Private/Templates/ContentElement/TouristAttraction.html diff --git a/Tests/Functional/Fixtures/Frontend/Extensions/example/composer.json b/Tests/Functional/Fixtures/Frontend/Extensions/example/composer.json new file mode 100644 index 00000000..2d126fb5 --- /dev/null +++ b/Tests/Functional/Fixtures/Frontend/Extensions/example/composer.json @@ -0,0 +1,28 @@ +{ + "name": "werkraummedia/example", + "description": "Example extension for functional tests", + "type": "typo3-cms-extension", + "license": "GPL-2.0-or-later", + "homepage": "https://github.com/werkraum-media/thuecat", + "authors": [ + { + "name": "Daniel Siepmann", + "email": "coding@daniel-siepmann.de", + "homepage": "https://daniel-siepmann.de/", + "role": "Developer" + } + ], + "autoload": { + "psr-4": { + "WerkraumMedia\\Example\\": "Classes/" + } + }, + "require": { + "werkraummedia/thuecat": "*" + }, + "extra": { + "typo3/cms": { + "extension-key": "example" + } + } +} diff --git a/Tests/Functional/Fixtures/Frontend/Rendering.typoscript b/Tests/Functional/Fixtures/Frontend/Rendering.typoscript index 7ab01863..ef6526b3 100644 --- a/Tests/Functional/Fixtures/Frontend/Rendering.typoscript +++ b/Tests/Functional/Fixtures/Frontend/Rendering.typoscript @@ -1,11 +1,23 @@ page = PAGE page.10 < styles.content.get -lib.thuecatContentElement { +tt_content.thuecat_tourist_attraction =< lib.contentElement +tt_content.thuecat_tourist_attraction { + templateName = TouristAttraction + dataProcessing { + 10 = WerkraumMedia\ThueCat\Frontend\DataProcessing\ResolveEntities + 10 { + table = tx_thuecat_tourist_attraction + uids.data = field:records + } + } +} + +lib.contentElement { partialRootPaths { - 0 = EXT:thuecat/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/ + 10 = EXT:example/Resources/Private/Partials/ContentElement/ } templateRootPaths { - 0 = EXT:thuecat/Tests/Functional/Fixtures/Frontend/Resources/Private/Templates/ContentElement/ + 10 = EXT:example/Resources/Private/Templates/ContentElement/ } } diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Accessibility.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Accessibility.html deleted file mode 100644 index f7f294f5..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Accessibility.html +++ /dev/null @@ -1,136 +0,0 @@ - -
-
-
- -
    - - -
  • {f:translate(id: 'content.accessibilitySpecification.searchCriteria.criteria.{criteria}', default: criteria, extensionName: 'Thuecat')}
  • -
    -
    -
-
-
-
-
    - -
  • {f:translate(id: 'content.accessibilitySpecification.certification.{specification.certificationDeaf}', default: specification.certificationDeaf, extensionName: 'Thuecat')} {f:translate(id: 'content.accessibilitySpecification.certification.deaf', extensionName: 'Thuecat')}
  • -
    - -
  • {f:translate(id: 'content.accessibilitySpecification.certification.{specification.certificationMental}', default: specification.certificationMental, extensionName: 'Thuecat')} {f:translate(id: 'content.accessibilitySpecification.certification.mental', extensionName: 'Thuecat')}
  • -
    - -
  • {f:translate(id: 'content.accessibilitySpecification.certification.{specification.certificationPartiallyDeaf}', default: specification.certificationPartiallyDeaf, extensionName: 'Thuecat')} {f:translate(id: 'content.accessibilitySpecification.certification.partiallyDeaf', extensionName: 'Thuecat')}
  • -
    - -
  • {f:translate(id: 'content.accessibilitySpecification.certification.{specification.certificationPartiallyVisual}', default: specification.certificationPartiallyVisual, extensionName: 'Thuecat')} {f:translate(id: 'content.accessibilitySpecification.certification.partiallyVisual', extensionName: 'Thuecat')}
  • -
    - -
  • {f:translate(id: 'content.accessibilitySpecification.certification.{specification.certificationVisual}', default: specification.certificationVisual, extensionName: 'Thuecat')} {f:translate(id: 'content.accessibilitySpecification.certification.visual', extensionName: 'Thuecat')}
  • -
    - -
  • {f:translate(id: 'content.accessibilitySpecification.certification.{specification.certificationWalking}', default: specification.certificationWalking, extensionName: 'Thuecat')} {f:translate(id: 'content.accessibilitySpecification.certification.walking', extensionName: 'Thuecat')}
  • -
    - -
  • {f:translate(id: 'content.accessibilitySpecification.certification.{specification.certificationWheelchair}', default: specification.certificationWheelchair, extensionName: 'Thuecat')} {f:translate(id: 'content.accessibilitySpecification.certification.wheelchair', extensionName: 'Thuecat')}
  • -
    -
-
-
- -
- - -
-

- -

-
-
-

{specification.shortDescriptionAllGenerations -> f:format.nl2br()}

-
-
-
-
- - -
-

- -

-
-
-

{specification.shortDescriptionAllergic -> f:format.nl2br()}

-
-
-
-
- - -
-

- -

-
-
-

{specification.shortDescriptionDeaf -> f:format.nl2br()}

-
-
-
-
- - -
-

- -

-
-
-

{specification.shortDescriptionMental -> f:format.nl2br()}

-
-
-
-
- - -
-

- -

-
-
-

{specification.shortDescriptionVisual -> f:format.nl2br()}

-
-
-
-
- - -
-

- -

-
-
-

{specification.shortDescriptionWalking -> f:format.nl2br()}

-
-
-
-
-
- -
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Address.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Address.html deleted file mode 100644 index 2c726b82..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Address.html +++ /dev/null @@ -1,24 +0,0 @@ - -
-

- - {address.street}
-
- - {address.zip} {address.city}
-
- - {address.email}
-
- - {address.phone}
-
- - {address.fax} - - - {f:translate(id: 'content.url', extensionName: 'Thuecat')} - -

-
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Digital.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Digital.html deleted file mode 100644 index 33df9854..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Digital.html +++ /dev/null @@ -1,9 +0,0 @@ - -
-

- - {f:translate(id: 'content.digitalOffer.{offer}', default: offer, extensionName: 'Thuecat')} - -

-
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Languages.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Languages.html deleted file mode 100644 index 8c47cd42..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Languages.html +++ /dev/null @@ -1,9 +0,0 @@ - -
-

- - {f:translate(id: 'content.availableLanguage.{language}', default: language, extensionName: 'Thuecat')} - -

-
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Museum.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Museum.html deleted file mode 100644 index 61f79d0c..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Museum.html +++ /dev/null @@ -1,9 +0,0 @@ - -
-

- - {f:translate(id: 'content.museumService.{service}', default: service, extensionName: 'Thuecat')} - -

-
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Offers.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Offers.html deleted file mode 100644 index 445ba6b7..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Offers.html +++ /dev/null @@ -1,80 +0,0 @@ - -
- - - - - - - - -
- - -
- -
-

- -

-
-
- -
-

{offer.description}

-
-
-
-
- -
- {price.title} - {price.price -> f:format.currency(decimalSeparator: ',', thousandsSeparator: '.', decimals: 2, currencySign: price.currency)} - {f:translate(id: 'content.price.rule.{price.rules.0}', default: price.rule, extensionName: 'Thuecat')} - -

{price.description}

-
-
-
-
-
-
-
-
-
-
-
- - - -
-
-

{offer.title} ({f:translate(id: 'content.price.type.{offer.type}', default: offer.type, extensionName: 'Thuecat')})

-
- -
-

{offer.description}

-
-
-
-
- -
- {price.title} - {price.price -> f:format.currency(decimalSeparator: ',', thousandsSeparator: '.', decimals: 2, currencySign: price.currency)} - {f:translate(id: 'content.price.rule.{price.rules.0}', default: price.rule, extensionName: 'Thuecat')} - -

{price.description}

-
-
-
-
-
-
-
-
-
-
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Opening.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Opening.html deleted file mode 100644 index 469dcdba..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Opening.html +++ /dev/null @@ -1,27 +0,0 @@ - -
- - - -
- - -
- - -

- {openingHour.from -> f:format.date(format: 'd.m.Y')} - - {openingHour.through -> f:format.date(format: 'd.m.Y')} -

-
- -
-
{f:translate(id: 'content.openingHour.weekday.{weekday}', default: weekday, extensionName: 'Thuecat')}
-
{openingHour.opens} - {openingHour.closes}
-
-
-
-
- -
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Parking.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Parking.html deleted file mode 100644 index 6a2802ab..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Parking.html +++ /dev/null @@ -1,22 +0,0 @@ - -
- - -
- -
- {parkingFacility.title} - {f:render(partial: 'Address', arguments: {address: parkingFacility.address})} -
-
-
-
- - - {parkingFacility.title} - {f:render(partial: 'Address', arguments: {address: parkingFacility.address})} - - -
-
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Payment.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Payment.html deleted file mode 100644 index 9a7589f6..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Payment.html +++ /dev/null @@ -1,9 +0,0 @@ - -
-

- - {f:translate(id: 'content.paymentAccepted.{payment}', default: payment, extensionName: 'Thuecat')} - -

-
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Photography.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Photography.html deleted file mode 100644 index b76f9873..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Photography.html +++ /dev/null @@ -1,9 +0,0 @@ - -
-

- - {f:translate(id: 'content.photography.{photography}', default: photography, extensionName: 'Thuecat')} - -

-
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Sanitation.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Sanitation.html deleted file mode 100644 index 49167b3e..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Sanitation.html +++ /dev/null @@ -1,9 +0,0 @@ - -
-

- - {f:translate(id: 'content.sanitation.{sanitation}', default: sanitation, extensionName: 'Thuecat')} - -

-
- diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Service.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Service.html deleted file mode 100644 index 8ac667a0..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Service.html +++ /dev/null @@ -1,5 +0,0 @@ - - - {f:translate(id: 'content.otherService.{service}', default: service, extensionName: 'Thuecat')} - - diff --git a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Traffic.html b/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Traffic.html deleted file mode 100644 index 0939e886..00000000 --- a/Tests/Functional/Fixtures/Frontend/Resources/Private/Partials/ContentElement/Traffic.html +++ /dev/null @@ -1,9 +0,0 @@ - -
-

- - {f:translate(id: 'content.trafficInfrastructure.{trafficInfrastructure}', default: trafficInfrastructure, extensionName: 'Thuecat')} - -

-
- diff --git a/Tests/Functional/FrontendTest.php b/Tests/Functional/FrontendTest.php index 7848a6a4..3409a1e6 100644 --- a/Tests/Functional/FrontendTest.php +++ b/Tests/Functional/FrontendTest.php @@ -41,6 +41,7 @@ protected function setUp(): void $this->testExtensionsToLoad = [ 'werkraummedia/thuecat', + 'typo3conf/ext/thuecat/Tests/Functional/Fixtures/Frontend/Extensions/example/', ]; $this->pathsToLinkInTestInstance = [ diff --git a/composer.json b/composer.json index 0185bfce..5deac948 100644 --- a/composer.json +++ b/composer.json @@ -74,6 +74,12 @@ "phpstan/extension-installer": true } }, + "scripts": { + "post-autoload-dump": [ + "mkdir -p .Build/web/typo3conf/ext/", + "[ -L .Build/web/typo3conf/ext/thuecat ] || ln -snvf ../../../../. .Build/web/typo3conf/ext/thuecat" + ] + }, "extra": { "typo3/cms": { "cms-package-dir": "{$vendor-dir}/typo3/cms", diff --git a/ext_emconf.php b/ext_emconf.php index fd4bc2f6..5e1dbc51 100644 --- a/ext_emconf.php +++ b/ext_emconf.php @@ -10,7 +10,7 @@ 'author' => 'Daniel Siepmann', 'author_email' => 'coding@daniel-siepmann.de', 'author_company' => '', - 'version' => '3.1.0', + 'version' => '4.0.0', 'constraints' => [ 'depends' => [ 'core' => '', diff --git a/ext_tables.php b/ext_tables.php index 3808ffd2..1bc835e0 100644 --- a/ext_tables.php +++ b/ext_tables.php @@ -1,5 +1,7 @@ { } }: let - php = pkgs.php82.buildEnv { + php = pkgs.php83.buildEnv { extensions = { enabled, all }: enabled ++ (with all; [ xdebug ]); @@ -11,7 +11,7 @@ let memory_limit = 4G ''; }; - inherit(pkgs.php82Packages) composer; + inherit(php.packages) composer; projectInstall = pkgs.writeShellApplication { name = "project-install"; @@ -49,7 +49,7 @@ let ''; }; -in pkgs.mkShell { +in pkgs.mkShellNoCC { name = "TYPO3 Extension ThüCAT"; buildInputs = [ php From f77f1968ca58d7f0e36121870351ce495cab2631 Mon Sep 17 00:00:00 2001 From: Daniel Siepmann Date: Thu, 28 Nov 2024 07:46:07 +0100 Subject: [PATCH 3/3] Ease entity registration Remove phpstan errors and ease code. Directly call the static methods instead of wrapping the call with call_user_func() for no reason. --- Classes/DependencyInjection/EntityPass.php | 4 ++-- phpstan-baseline.neon | 10 ---------- 2 files changed, 2 insertions(+), 12 deletions(-) diff --git a/Classes/DependencyInjection/EntityPass.php b/Classes/DependencyInjection/EntityPass.php index 34288bd5..c541769b 100644 --- a/Classes/DependencyInjection/EntityPass.php +++ b/Classes/DependencyInjection/EntityPass.php @@ -47,8 +47,8 @@ public function process(ContainerBuilder $container): void 'registerEntityClass', [ $definition->getClass(), - call_user_func([$definition->getClass(), 'getPriority']), - call_user_func([$definition->getClass(), 'getSupportedTypes']), + $definition->getClass()::getPriority(), + $definition->getClass()::getSupportedTypes(), ] ); } diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index b2869ff0..cf86fa35 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -1,15 +1,5 @@ parameters: ignoreErrors: - - - message: "#^Parameter \\#1 \\$callback of function call_user_func expects callable\\(\\)\\: mixed, array\\{string, 'getPriority'\\} given\\.$#" - count: 1 - path: Classes/DependencyInjection/EntityPass.php - - - - message: "#^Parameter \\#1 \\$callback of function call_user_func expects callable\\(\\)\\: mixed, array\\{string, 'getSupportedTypes'\\} given\\.$#" - count: 1 - path: Classes/DependencyInjection/EntityPass.php - - message: "#^Method WerkraumMedia\\\\ThueCat\\\\Domain\\\\Import\\\\EntityMapper\\:\\:mapDataToEntity\\(\\) should return object but returns mixed\\.$#" count: 1