From c9d53fd473acc154b0404e906c23f352097af6cd Mon Sep 17 00:00:00 2001 From: Alex Pott <alex.a.pott@googlemail.com> Date: Thu, 4 Apr 2024 00:40:51 +0100 Subject: [PATCH] Issue #3424509 by godotislate, quietone, sorlov, smustgrave, benjifisher, alexpott: Update MigratePluginManager to include both attribute and annotation class --- .../AttributeDiscoveryWithAnnotations.php | 4 +- .../src/Plugin/migrate/destination/Book.php | 7 +- .../src/Plugin/migrate/field/DateField.php | 24 ++--- .../Plugin/migrate/destination/EntityFile.php | 6 +- core/modules/migrate/migrate.api.php | 19 ++-- core/modules/migrate/migrate.services.yml | 8 +- .../src/Attribute/MigrateDestination.php | 53 +++++++++++ .../migrate/src/Attribute/MigrateProcess.php | 50 ++++++++++ .../migrate/src/Attribute/MigrateSource.php | 93 +++++++++++++++++++ .../MultipleProviderAttributeInterface.php | 44 +++++++++ ...otatedClassDiscoveryAutomatedProviders.php | 29 +----- ...otatedDiscoveryAutomatedProvidersTrait.php | 59 ++++++++++++ ...ributeClassDiscoveryAutomatedProviders.php | 42 +++++++++ ...overyWithAnnotationsAutomatedProviders.php | 83 +++++++++++++++++ .../Plugin/MigrateDestinationInterface.php | 2 +- .../MigrateDestinationPluginManager.php | 10 +- .../src/Plugin/MigratePluginManager.php | 18 +++- .../src/Plugin/MigrateProcessInterface.php | 2 +- .../src/Plugin/MigrateSourceInterface.php | 2 +- .../src/Plugin/MigrateSourcePluginManager.php | 33 ++++++- .../migrate/destination/DestinationBase.php | 2 +- .../src/Plugin/migrate/id_map/NullIdMap.php | 4 +- .../migrate/src/Plugin/migrate/id_map/Sql.php | 4 +- .../src/Plugin/migrate/process/Explode.php | 6 +- .../migrate/source/EmbeddedDataSource.php | 10 +- .../src/Plugin/migrate/source/EmptySource.php | 11 ++- .../migrate/source/SourcePluginBase.php | 2 +- .../modules/migrate/src/ProcessPluginBase.php | 2 +- ...migrate_source_annotation_bc_test.info.yml | 5 + .../source/MigrateSourceWithAnnotations.php | 50 ++++++++++ ...SourceWithAnnotationsMultipleProviders.php | 29 ++++++ .../MigrateSourceAnnotationDiscoveryTest.php | 60 ++++++++++++ .../migrate_drupal.services.yml | 1 + .../src/Attribute/MigrateField.php | 74 +++++++++++++++ .../src/Plugin/MigrateFieldPluginManager.php | 4 +- .../Plugin/migrate/field/FieldPluginBase.php | 2 +- .../Plugin/migrate/source/ContentEntity.php | 14 +-- 37 files changed, 763 insertions(+), 105 deletions(-) create mode 100644 core/modules/migrate/src/Attribute/MigrateDestination.php create mode 100644 core/modules/migrate/src/Attribute/MigrateProcess.php create mode 100644 core/modules/migrate/src/Attribute/MigrateSource.php create mode 100644 core/modules/migrate/src/Attribute/MultipleProviderAttributeInterface.php create mode 100644 core/modules/migrate/src/Plugin/Discovery/AnnotatedDiscoveryAutomatedProvidersTrait.php create mode 100644 core/modules/migrate/src/Plugin/Discovery/AttributeClassDiscoveryAutomatedProviders.php create mode 100644 core/modules/migrate/src/Plugin/Discovery/AttributeDiscoveryWithAnnotationsAutomatedProviders.php create mode 100644 core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/migrate_source_annotation_bc_test.info.yml create mode 100644 core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/src/Plugin/migrate/source/MigrateSourceWithAnnotations.php create mode 100644 core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/src/Plugin/migrate/source/MigrateSourceWithAnnotationsMultipleProviders.php create mode 100644 core/modules/migrate/tests/src/Kernel/Plugin/source/MigrateSourceAnnotationDiscoveryTest.php create mode 100644 core/modules/migrate_drupal/src/Attribute/MigrateField.php diff --git a/core/lib/Drupal/Core/Plugin/Discovery/AttributeDiscoveryWithAnnotations.php b/core/lib/Drupal/Core/Plugin/Discovery/AttributeDiscoveryWithAnnotations.php index a54e86087552..e4d2a51a8533 100644 --- a/core/lib/Drupal/Core/Plugin/Discovery/AttributeDiscoveryWithAnnotations.php +++ b/core/lib/Drupal/Core/Plugin/Discovery/AttributeDiscoveryWithAnnotations.php @@ -114,7 +114,7 @@ protected function parseClass(string $class, \SplFileInfo $fileinfo): array { * @see \Drupal\Component\Annotation\Plugin\Discovery\AnnotatedClassDiscovery::prepareAnnotationDefinition() * @see \Drupal\Core\Plugin\Discovery\AnnotatedClassDiscovery::prepareAnnotationDefinition() */ - private function prepareAnnotationDefinition(AnnotationInterface $annotation, string $class): void { + protected function prepareAnnotationDefinition(AnnotationInterface $annotation, string $class): void { $annotation->setClass($class); if (!$annotation->getProvider()) { $annotation->setProvider($this->getProviderFromNamespace($class)); @@ -133,7 +133,7 @@ private function prepareAnnotationDefinition(AnnotationInterface $annotation, st * @see \Drupal\Component\Annotation\Plugin\Discovery\AnnotatedClassDiscovery::getAnnotationReader() * @see \Drupal\Core\Plugin\Discovery\AnnotatedClassDiscovery::getAnnotationReader() */ - private function getAnnotationReader() : SimpleAnnotationReader { + protected function getAnnotationReader() : SimpleAnnotationReader { if (!isset($this->annotationReader)) { $this->annotationReader = new SimpleAnnotationReader(); diff --git a/core/modules/book/src/Plugin/migrate/destination/Book.php b/core/modules/book/src/Plugin/migrate/destination/Book.php index dcc5056c89af..2f4ea3349066 100644 --- a/core/modules/book/src/Plugin/migrate/destination/Book.php +++ b/core/modules/book/src/Plugin/migrate/destination/Book.php @@ -3,15 +3,14 @@ namespace Drupal\book\Plugin\migrate\destination; use Drupal\Core\Entity\EntityInterface; +use Drupal\migrate\Attribute\MigrateDestination; use Drupal\migrate\Plugin\migrate\destination\EntityContentBase; use Drupal\migrate\Row; /** - * @MigrateDestination( - * id = "book", - * provider = "book" - * ) + * Provides migrate destination plugin for Book content. */ +#[MigrateDestination('book')] class Book extends EntityContentBase { /** diff --git a/core/modules/datetime/src/Plugin/migrate/field/DateField.php b/core/modules/datetime/src/Plugin/migrate/field/DateField.php index f6a33434d15b..9bfa3572e31e 100644 --- a/core/modules/datetime/src/Plugin/migrate/field/DateField.php +++ b/core/modules/datetime/src/Plugin/migrate/field/DateField.php @@ -6,25 +6,25 @@ use Drupal\migrate\Plugin\MigrationInterface; use Drupal\migrate\MigrateException; use Drupal\migrate\Row; +use Drupal\migrate_drupal\Attribute\MigrateField; use Drupal\migrate_drupal\Plugin\migrate\field\FieldPluginBase; // cspell:ignore todate /** * Provides a field plugin for date and time fields. - * - * @MigrateField( - * id = "datetime", - * type_map = { - * "date" = "datetime", - * "datestamp" = "timestamp", - * "datetime" = "datetime", - * }, - * core = {6,7}, - * source_module = "date", - * destination_module = "datetime" - * ) */ +#[MigrateField( + id: 'datetime', + core: [6, 7], + type_map: [ + 'date' => 'datetime', + 'datestamp' => 'timestamp', + 'datetime' => 'datetime', + ], + source_module: 'date', + destination_module: 'datetime', +)] class DateField extends FieldPluginBase { /** diff --git a/core/modules/file/src/Plugin/migrate/destination/EntityFile.php b/core/modules/file/src/Plugin/migrate/destination/EntityFile.php index 08b946f53f9a..c953a71869a9 100644 --- a/core/modules/file/src/Plugin/migrate/destination/EntityFile.php +++ b/core/modules/file/src/Plugin/migrate/destination/EntityFile.php @@ -3,15 +3,15 @@ namespace Drupal\file\Plugin\migrate\destination; use Drupal\Core\Field\Plugin\Field\FieldType\UriItem; +use Drupal\migrate\Attribute\MigrateDestination; use Drupal\migrate\Row; use Drupal\migrate\MigrateException; use Drupal\migrate\Plugin\migrate\destination\EntityContentBase; /** - * @MigrateDestination( - * id = "entity:file" - * ) + * Provides migrate destination plugin for File entities. */ +#[MigrateDestination('entity:file')] class EntityFile extends EntityContentBase { /** diff --git a/core/modules/migrate/migrate.api.php b/core/modules/migrate/migrate.api.php index 0339475ff0f7..9b3ab67bb980 100644 --- a/core/modules/migrate/migrate.api.php +++ b/core/modules/migrate/migrate.api.php @@ -47,8 +47,8 @@ * @section sec_source Migrate API source plugins * Migrate API source plugins implement * \Drupal\migrate\Plugin\MigrateSourceInterface and usually extend - * \Drupal\migrate\Plugin\migrate\source\SourcePluginBase. They are annotated - * with \Drupal\migrate\Annotation\MigrateSource annotation and must be in + * \Drupal\migrate\Plugin\migrate\source\SourcePluginBase. They have the + * \Drupal\migrate\Attribute\MigrateSource attribute and must be in * namespace subdirectory 'Plugin\migrate\source' under the namespace of the * module that defines them. Migrate API source plugins are managed by the * \Drupal\migrate\Plugin\MigrateSourcePluginManager class. @@ -59,8 +59,8 @@ * @section sec_process Migrate API process plugins * Migrate API process plugins implement * \Drupal\migrate\Plugin\MigrateProcessInterface and usually extend - * \Drupal\migrate\ProcessPluginBase. They are annotated with - * \Drupal\migrate\Annotation\MigrateProcessPlugin annotation and must be in + * \Drupal\migrate\ProcessPluginBase. They have the + * \Drupal\migrate\Attribute\MigrateProcess attribute and must be in * namespace subdirectory 'Plugin\migrate\process' under the namespace of the * module that defines them. Migrate API process plugins are managed by the * \Drupal\migrate\Plugin\MigratePluginManager class. @@ -70,12 +70,11 @@ * @section sec_destination Migrate API destination plugins * Migrate API destination plugins implement * \Drupal\migrate\Plugin\MigrateDestinationInterface and usually extend - * \Drupal\migrate\Plugin\migrate\destination\DestinationBase. They are - * annotated with \Drupal\migrate\Annotation\MigrateDestination annotation and - * must be in namespace subdirectory 'Plugin\migrate\destination' under the - * namespace of the module that defines them. Migrate API destination plugins - * are managed by the \Drupal\migrate\Plugin\MigrateDestinationPluginManager - * class. + * \Drupal\migrate\Plugin\migrate\destination\DestinationBase. They have the + * \Drupal\migrate\Attribute\MigrateDestination attribute and must be in + * namespace subdirectory 'Plugin\migrate\destination' under the namespace of + * the module that defines them. Migrate API destination plugins are managed by + * the \Drupal\migrate\Plugin\MigrateDestinationPluginManager class. * * @link https://api.drupal.org/api/drupal/namespace/Drupal!migrate!Plugin!migrate!destination List of destination plugins for Drupal configuration and content entities provided by the core Migrate module. @endlink * diff --git a/core/modules/migrate/migrate.services.yml b/core/modules/migrate/migrate.services.yml index d7ef2ded4b0d..b7068cf68250 100644 --- a/core/modules/migrate/migrate.services.yml +++ b/core/modules/migrate/migrate.services.yml @@ -14,7 +14,13 @@ services: arguments: [source, '@container.namespaces', '@cache.discovery', '@module_handler'] plugin.manager.migrate.process: class: Drupal\migrate\Plugin\MigratePluginManager - arguments: [process, '@container.namespaces', '@cache.discovery', '@module_handler', 'Drupal\migrate\Annotation\MigrateProcessPlugin'] + arguments: + - process + - '@container.namespaces' + - '@cache.discovery' + - '@module_handler' + - 'Drupal\migrate\Attribute\MigrateProcess' + - 'Drupal\migrate\Annotation\MigrateProcessPlugin' plugin.manager.migrate.destination: class: Drupal\migrate\Plugin\MigrateDestinationPluginManager arguments: [destination, '@container.namespaces', '@cache.discovery', '@module_handler', '@entity_type.manager'] diff --git a/core/modules/migrate/src/Attribute/MigrateDestination.php b/core/modules/migrate/src/Attribute/MigrateDestination.php new file mode 100644 index 000000000000..54e469b13814 --- /dev/null +++ b/core/modules/migrate/src/Attribute/MigrateDestination.php @@ -0,0 +1,53 @@ +<?php + +declare(strict_types=1); + +namespace Drupal\migrate\Attribute; + +use Drupal\Component\Plugin\Attribute\Plugin; + +/** + * Defines a MigrateDestination attribute. + * + * Plugin Namespace: Plugin\migrate\destination + * + * For a working example, see + * \Drupal\migrate\Plugin\migrate\destination\UrlAlias + * + * @see \Drupal\migrate\Plugin\MigrateDestinationPluginManager + * @see \Drupal\migrate\Plugin\MigrateDestinationInterface + * @see \Drupal\migrate\Plugin\migrate\destination\DestinationBase + * @see \Drupal\migrate\Attribute\MigrateProcess + * @see \Drupal\migrate\Attribute\MigrateSource + * @see plugin_api + * + * @ingroup migration + */ +#[\Attribute(\Attribute::TARGET_CLASS)] +class MigrateDestination extends Plugin { + + /** + * Constructs a migrate destination plugin attribute object. + * + * @param string $id + * A unique identifier for the destination plugin. + * @param bool $requirements_met + * (optional) Whether requirements are met. + * @param string|null $destination_module + * (optional) Identifies the system handling the data the destination plugin + * will write. The destination plugin itself determines how the value is + * used. For example, Migrate's destination plugins expect + * destination_module to be the name of a module that must be installed on + * the destination. + * @param class-string|null $deriver + * (optional) The deriver class. + */ + public function __construct( + public readonly string $id, + public bool $requirements_met = TRUE, + public readonly ?string $destination_module = NULL, + public readonly ?string $deriver = NULL, + ) { + } + +} diff --git a/core/modules/migrate/src/Attribute/MigrateProcess.php b/core/modules/migrate/src/Attribute/MigrateProcess.php new file mode 100644 index 000000000000..ba2cb65cd147 --- /dev/null +++ b/core/modules/migrate/src/Attribute/MigrateProcess.php @@ -0,0 +1,50 @@ +<?php + +declare(strict_types=1); + +namespace Drupal\migrate\Attribute; + +use Drupal\Component\Plugin\Attribute\Plugin; + +/** + * Defines a MigrateProcess attribute. + * + * Plugin Namespace: Plugin\migrate\process + * + * For a working example, see + * \Drupal\migrate\Plugin\migrate\process\DefaultValue + * + * @see \Drupal\migrate\Plugin\MigratePluginManager + * @see \Drupal\migrate\Plugin\MigrateProcessInterface + * @see \Drupal\migrate\ProcessPluginBase + * @see \Drupal\migrate\Attribute\MigrateDestination + * @see \Drupal\migrate\Attribute\MigrateSource + * @see plugin_api + * + * @ingroup migration + */ +#[\Attribute(\Attribute::TARGET_CLASS)] +class MigrateProcess extends Plugin { + + /** + * Constructs a migrate process plugin attribute object. + * + * @param string $id + * A unique identifier for the process plugin. + * @param bool $handle_multiples + * (optional) Whether the plugin handles multiples itself. Typically these + * plugins will expect an array as input and iterate over it themselves, + * changing the whole array. For example the 'sub_process' and the 'flatten' + * plugins. If the plugin only needs to change a single value, then it can + * skip setting this attribute and let + * \Drupal\migrate\MigrateExecutable::processRow() handle the iteration. + * @param class-string|null $deriver + * (optional) The deriver class. + */ + public function __construct( + public readonly string $id, + public readonly bool $handle_multiples = FALSE, + public readonly ?string $deriver = NULL, + ) {} + +} diff --git a/core/modules/migrate/src/Attribute/MigrateSource.php b/core/modules/migrate/src/Attribute/MigrateSource.php new file mode 100644 index 000000000000..1dd690035063 --- /dev/null +++ b/core/modules/migrate/src/Attribute/MigrateSource.php @@ -0,0 +1,93 @@ +<?php + +declare(strict_types=1); + +namespace Drupal\migrate\Attribute; + +use Drupal\Component\Plugin\Attribute\Plugin; + +/** + * Defines a MigrateSource attribute. + * + * Plugin Namespace: Plugin\migrate\source + * + * For a working example, see + * \Drupal\migrate\Plugin\migrate\source\EmptySource + * \Drupal\migrate_drupal\Plugin\migrate\source\UrlAlias + * + * @see \Drupal\migrate\Plugin\MigratePluginManager + * @see \Drupal\migrate\Plugin\MigrateSourceInterface + * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase + * @see \Drupal\migrate\Attribute\MigrateDestination + * @see \Drupal\migrate\Attribute\MigrateProcess + * @see plugin_api + * + * @ingroup migration + */ +#[\Attribute(\Attribute::TARGET_CLASS)] +class MigrateSource extends Plugin implements MultipleProviderAttributeInterface { + + /** + * The providers of the source plugin. + */ + protected array $providers = []; + + /** + * Constructs a migrate source plugin attribute object. + * + * @param string $id + * A unique identifier for the source plugin. + * @param string $source_module + * Identifies the system providing the data the source plugin will read. + * @param bool $requirements_met + * (optional) Whether requirements are met. Defaults to true. The source + * plugin itself determines how the value is used. For example, Migrate + * Drupal's source plugins expect source_module to be the name of a module + * that must be installed and enabled in the source database. + * @param mixed $minimum_version + * (optional) Specifies the minimum version of the source provider. This can + * be any type, and the source plugin itself determines how it is used. For + * example, Migrate Drupal's source plugins expect this to be an integer + * representing the minimum installed database schema version of the module + * specified by source_module. + * @param class-string|null $deriver + * (optional) The deriver class. + * + * @see \Drupal\migrate_drupal\Plugin\migrate\source\DrupalSqlBase::checkRequirements + */ + public function __construct( + public readonly string $id, + public readonly string $source_module, + public bool $requirements_met = TRUE, + public readonly mixed $minimum_version = '', + public readonly ?string $deriver = NULL, + ) {} + + /** + * {@inheritdoc} + */ + public function setProvider(string $provider): void { + $this->setProviders([$provider]); + } + + /** + * {@inheritdoc} + */ + public function getProviders(): array { + return $this->providers; + } + + /** + * {@inheritdoc} + */ + public function setProviders(array $providers): void { + if ($providers) { + parent::setProvider(reset($providers)); + } + else { + $this->provider = NULL; + } + $this->providers = $providers; + } + +} diff --git a/core/modules/migrate/src/Attribute/MultipleProviderAttributeInterface.php b/core/modules/migrate/src/Attribute/MultipleProviderAttributeInterface.php new file mode 100644 index 000000000000..437ddfb8b9ae --- /dev/null +++ b/core/modules/migrate/src/Attribute/MultipleProviderAttributeInterface.php @@ -0,0 +1,44 @@ +<?php + +declare(strict_types=1); + +namespace Drupal\migrate\Attribute; + +use Drupal\Component\Plugin\Attribute\AttributeInterface; + +/** + * Defines a common interface for attributes with multiple providers. + * + * @internal + * This is a temporary solution to the fact that migration source plugins have + * more than one provider. This functionality will be moved to core in + * https://www.drupal.org/node/2786355. + */ +interface MultipleProviderAttributeInterface extends AttributeInterface { + + /** + * Gets the name of the provider of the attribute class. + * + * @return string|null + * The provider of the attribute. If there are multiple providers the first + * is returned. + */ + public function getProvider(): ?string; + + /** + * Gets the provider names of the attribute class. + * + * @return string[] + * The providers of the attribute. + */ + public function getProviders(): array; + + /** + * Sets the provider names of the attribute class. + * + * @param string[] $providers + * The providers of the attribute. + */ + public function setProviders(array $providers): void; + +} diff --git a/core/modules/migrate/src/Plugin/Discovery/AnnotatedClassDiscoveryAutomatedProviders.php b/core/modules/migrate/src/Plugin/Discovery/AnnotatedClassDiscoveryAutomatedProviders.php index c51468911645..edea32fefda2 100644 --- a/core/modules/migrate/src/Plugin/Discovery/AnnotatedClassDiscoveryAutomatedProviders.php +++ b/core/modules/migrate/src/Plugin/Discovery/AnnotatedClassDiscoveryAutomatedProviders.php @@ -3,12 +3,10 @@ namespace Drupal\migrate\Plugin\Discovery; use Doctrine\Common\Annotations\AnnotationRegistry; -use Drupal\Component\Annotation\AnnotationInterface; use Drupal\Component\Annotation\Doctrine\StaticReflectionParser as BaseStaticReflectionParser; use Drupal\Component\Annotation\Reflection\MockFileFinder; use Drupal\Component\ClassFinder\ClassFinder; use Drupal\Core\Plugin\Discovery\AnnotatedClassDiscovery; -use Drupal\migrate\Annotation\MultipleProviderAnnotationInterface; /** * Determines providers based on a class's and its parent's namespaces. @@ -20,12 +18,7 @@ */ class AnnotatedClassDiscoveryAutomatedProviders extends AnnotatedClassDiscovery { - /** - * A utility object that can use active autoloaders to find files for classes. - * - * @var \Drupal\Component\ClassFinder\ClassFinderInterface - */ - protected $finder; + use AnnotatedDiscoveryAutomatedProvidersTrait; /** * Constructs an AnnotatedClassDiscoveryAutomatedProviders object. @@ -48,26 +41,6 @@ public function __construct($subdir, \Traversable $root_namespaces, $plugin_defi $this->finder = new ClassFinder(); } - /** - * {@inheritdoc} - */ - protected function prepareAnnotationDefinition(AnnotationInterface $annotation, $class, BaseStaticReflectionParser $parser = NULL) { - if (!($annotation instanceof MultipleProviderAnnotationInterface)) { - throw new \LogicException('AnnotatedClassDiscoveryAutomatedProviders annotations must implement \Drupal\migrate\Annotation\MultipleProviderAnnotationInterface'); - } - $annotation->setClass($class); - $providers = $annotation->getProviders(); - // Loop through all the parent classes and add their providers (which we - // infer by parsing their namespaces) to the $providers array. - do { - $providers[] = $this->getProviderFromNamespace($parser->getNamespaceName()); - } while ($parser = StaticReflectionParser::getParentParser($parser, $this->finder)); - $providers = array_unique(array_filter($providers, function ($provider) { - return $provider && $provider !== 'component'; - })); - $annotation->setProviders($providers); - } - /** * {@inheritdoc} */ diff --git a/core/modules/migrate/src/Plugin/Discovery/AnnotatedDiscoveryAutomatedProvidersTrait.php b/core/modules/migrate/src/Plugin/Discovery/AnnotatedDiscoveryAutomatedProvidersTrait.php new file mode 100644 index 000000000000..aa9d8745a677 --- /dev/null +++ b/core/modules/migrate/src/Plugin/Discovery/AnnotatedDiscoveryAutomatedProvidersTrait.php @@ -0,0 +1,59 @@ +<?php + +namespace Drupal\migrate\Plugin\Discovery; + +use Drupal\Component\Annotation\AnnotationInterface; +use Drupal\Component\Annotation\Doctrine\StaticReflectionParser as BaseStaticReflectionParser; +use Drupal\migrate\Annotation\MultipleProviderAnnotationInterface; + +/** + * Provides method for annotation discovery with multiple providers. + */ +trait AnnotatedDiscoveryAutomatedProvidersTrait { + + /** + * A utility object that can use active autoloaders to find files for classes. + * + * @var \Drupal\Component\ClassFinder\ClassFinderInterface + */ + protected $finder; + + /** + * Prepares the annotation definition. + * + * This is modified from the prepareAnnotationDefinition method from annotated + * class discovery to account for multiple providers. + * + * @param \Drupal\Component\Annotation\AnnotationInterface $annotation + * The annotation derived from the plugin. + * @param class-string $class + * The class used for the plugin. + * @param \Drupal\Component\Annotation\Doctrine\StaticReflectionParser|null $parser + * Static reflection parser. + * + * @see \Drupal\Component\Annotation\Plugin\Discovery\AnnotatedClassDiscovery::prepareAnnotationDefinition() + * @see \Drupal\Core\Plugin\Discovery\AnnotatedClassDiscovery::prepareAnnotationDefinition() + */ + protected function prepareAnnotationDefinition(AnnotationInterface $annotation, $class, ?BaseStaticReflectionParser $parser = NULL): void { + if (!($annotation instanceof MultipleProviderAnnotationInterface)) { + throw new \LogicException('AnnotatedClassDiscoveryAutomatedProviders annotations must implement ' . MultipleProviderAnnotationInterface::class); + } + if (!$parser) { + throw new \LogicException('Parser argument must be passed for automated providers discovery.'); + } + if (!method_exists($this, 'getProviderFromNamespace')) { + throw new \LogicException('Classes using \Drupal\migrate\Plugin\Discovery\AnnotatedDiscoveryAutomatedProvidersTrait must have getProviderFromNamespace() method.'); + } + // @see \Drupal\Component\Annotation\Plugin\Discovery\AnnotatedClassDiscovery::prepareAnnotationDefinition() + $annotation->setClass($class); + $providers = $annotation->getProviders(); + // Loop through all the parent classes and add their providers (which we + // infer by parsing their namespaces) to the $providers array. + do { + $providers[] = $this->getProviderFromNamespace($parser->getNamespaceName()); + } while ($parser = StaticReflectionParser::getParentParser($parser, $this->finder)); + $providers = array_diff(array_unique(array_filter($providers)), ['component']); + $annotation->setProviders($providers); + } + +} diff --git a/core/modules/migrate/src/Plugin/Discovery/AttributeClassDiscoveryAutomatedProviders.php b/core/modules/migrate/src/Plugin/Discovery/AttributeClassDiscoveryAutomatedProviders.php new file mode 100644 index 000000000000..c5691a57bf9e --- /dev/null +++ b/core/modules/migrate/src/Plugin/Discovery/AttributeClassDiscoveryAutomatedProviders.php @@ -0,0 +1,42 @@ +<?php + +declare(strict_types=1); + +namespace Drupal\migrate\Plugin\Discovery; + +use Drupal\Component\Plugin\Attribute\AttributeInterface; +use Drupal\Core\Plugin\Discovery\AttributeClassDiscovery; +use Drupal\migrate\Attribute\MultipleProviderAttributeInterface; + +/** + * Determines providers based on the namespaces of a class and its ancestors. + * + * @internal + * This is a temporary solution to the fact that migration source plugins have + * more than one provider. This functionality will be moved to core in + * https://www.drupal.org/node/2786355. + */ +class AttributeClassDiscoveryAutomatedProviders extends AttributeClassDiscovery { + + /** + * {@inheritdoc} + */ + protected function prepareAttributeDefinition(AttributeInterface $attribute, string $class): void { + if (!($attribute instanceof MultipleProviderAttributeInterface)) { + throw new \LogicException('AttributeClassDiscoveryAutomatedProviders must implement ' . MultipleProviderAttributeInterface::class); + } + // @see Drupal\Component\Plugin\Discovery\AttributeClassDiscovery::prepareAttributeDefinition() + $attribute->setClass($class); + + // Loop through all the parent classes and add their providers (which we + // infer by parsing their namespaces) to the $providers array. + $providers = $attribute->getProviders(); + do { + $providers[] = $this->getProviderFromNamespace($class); + } while (($class = get_parent_class($class)) !== FALSE); + + $providers = array_diff(array_unique(array_filter($providers)), ['component']); + $attribute->setProviders($providers); + } + +} diff --git a/core/modules/migrate/src/Plugin/Discovery/AttributeDiscoveryWithAnnotationsAutomatedProviders.php b/core/modules/migrate/src/Plugin/Discovery/AttributeDiscoveryWithAnnotationsAutomatedProviders.php new file mode 100644 index 000000000000..fab94bf307fe --- /dev/null +++ b/core/modules/migrate/src/Plugin/Discovery/AttributeDiscoveryWithAnnotationsAutomatedProviders.php @@ -0,0 +1,83 @@ +<?php + +declare(strict_types=1); + +namespace Drupal\migrate\Plugin\Discovery; + +use Drupal\Component\Annotation\Doctrine\StaticReflectionParser as BaseStaticReflectionParser; +use Drupal\Component\Annotation\Reflection\MockFileFinder; +use Drupal\Component\ClassFinder\ClassFinder; +use Drupal\Component\Plugin\Attribute\AttributeInterface; +use Drupal\Core\Plugin\Discovery\AttributeDiscoveryWithAnnotations; + +/** + * Enables both attribute and annotation discovery for plugin definitions. + * + * @internal + * This is a temporary solution to the fact that migration source plugins have + * more than one provider. This functionality will be moved to core in + * https://www.drupal.org/node/2786355. + */ +class AttributeDiscoveryWithAnnotationsAutomatedProviders extends AttributeDiscoveryWithAnnotations { + + use AnnotatedDiscoveryAutomatedProvidersTrait; + + /** + * Instance of attribute class discovery with automatic providers. + * + * Since there isn't multiple inheritance, instantiate the attribute only + * discovery for code reuse. + * + * @var \Drupal\migrate\Plugin\Discovery\AttributeClassDiscoveryAutomatedProviders + */ + private AttributeClassDiscoveryAutomatedProviders $attributeDiscovery; + + public function __construct( + string $subdir, + \Traversable $rootNamespaces, + string $pluginDefinitionAttributeName = 'Drupal\Component\Plugin\Attribute\Plugin', + string $pluginDefinitionAnnotationName = 'Drupal\Component\Annotation\Plugin', + array $additionalNamespaces = [], + ) { + parent::__construct($subdir, $rootNamespaces, $pluginDefinitionAttributeName, $pluginDefinitionAnnotationName, $additionalNamespaces); + $this->finder = new ClassFinder(); + $this->attributeDiscovery = new AttributeClassDiscoveryAutomatedProviders($subdir, $rootNamespaces, $pluginDefinitionAttributeName); + } + + /** + * {@inheritdoc} + */ + protected function prepareAttributeDefinition(AttributeInterface $attribute, string $class): void { + $this->attributeDiscovery->prepareAttributeDefinition($attribute, $class); + } + + /** + * {@inheritdoc} + */ + protected function parseClass(string $class, \SplFileInfo $fileinfo): array { + // The filename is already known, so there is no need to find the + // file. However, StaticReflectionParser needs a finder, so use a + // mock version. + $finder = MockFileFinder::create($fileinfo->getPathName()); + $parser = new BaseStaticReflectionParser($class, $finder, FALSE); + + $reflection_class = $parser->getReflectionClass(); + // @todo Handle deprecating definitions discovery via annotations in + // https://www.drupal.org/project/drupal/issues/3265945. + /** @var \Drupal\Component\Annotation\AnnotationInterface $annotation */ + if ($annotation = $this->getAnnotationReader()->getClassAnnotation($reflection_class, $this->pluginDefinitionAnnotationName)) { + $this->prepareAnnotationDefinition($annotation, $class, $parser); + return ['id' => $annotation->getId(), 'content' => $annotation->get()]; + } + + // Annotations use static reflection and are able to analyze a class that + // extends classes or uses traits that do not exist. Attribute discovery + // will trigger a fatal error with such classes, so only call it if the + // class has a class attribute. + if ($reflection_class->hasClassAttribute($this->pluginDefinitionAttributeName)) { + return parent::parseClass($class, $fileinfo); + } + return ['id' => NULL, 'content' => NULL]; + } + +} diff --git a/core/modules/migrate/src/Plugin/MigrateDestinationInterface.php b/core/modules/migrate/src/Plugin/MigrateDestinationInterface.php index d2c7fb7ee994..d2dbe2f68abd 100644 --- a/core/modules/migrate/src/Plugin/MigrateDestinationInterface.php +++ b/core/modules/migrate/src/Plugin/MigrateDestinationInterface.php @@ -13,7 +13,7 @@ * * @see \Drupal\migrate\Plugin\migrate\destination\DestinationBase * @see \Drupal\migrate\Plugin\MigrateDestinationPluginManager - * @see \Drupal\migrate\Annotation\MigrateDestination + * @see \Drupal\migrate\Attribute\MigrateDestination * @see plugin_api * * @ingroup migration diff --git a/core/modules/migrate/src/Plugin/MigrateDestinationPluginManager.php b/core/modules/migrate/src/Plugin/MigrateDestinationPluginManager.php index b65515c3afed..6b7514435f29 100644 --- a/core/modules/migrate/src/Plugin/MigrateDestinationPluginManager.php +++ b/core/modules/migrate/src/Plugin/MigrateDestinationPluginManager.php @@ -5,13 +5,14 @@ use Drupal\Core\Cache\CacheBackendInterface; use Drupal\Core\Entity\EntityTypeManagerInterface; use Drupal\Core\Extension\ModuleHandlerInterface; +use Drupal\migrate\Attribute\MigrateDestination; /** * Plugin manager for migrate destination plugins. * * @see \Drupal\migrate\Plugin\MigrateDestinationInterface * @see \Drupal\migrate\Plugin\migrate\destination\DestinationBase - * @see \Drupal\migrate\Annotation\MigrateDestination + * @see \Drupal\migrate\Attribute\MigrateDestination * @see plugin_api * * @ingroup migration @@ -40,12 +41,15 @@ class MigrateDestinationPluginManager extends MigratePluginManager { * The module handler to invoke the alter hook with. * @param \Drupal\Core\Entity\EntityTypeManagerInterface $entity_type_manager * The entity type manager. + * @param string $attribute + * (optional) The attribute class name. Defaults to + * 'Drupal\migrate\Attribute\MigrateDestination'. * @param string $annotation * (optional) The annotation class name. Defaults to * 'Drupal\migrate\Annotation\MigrateDestination'. */ - public function __construct($type, \Traversable $namespaces, CacheBackendInterface $cache_backend, ModuleHandlerInterface $module_handler, EntityTypeManagerInterface $entity_type_manager, $annotation = 'Drupal\migrate\Annotation\MigrateDestination') { - parent::__construct($type, $namespaces, $cache_backend, $module_handler, $annotation); + public function __construct($type, \Traversable $namespaces, CacheBackendInterface $cache_backend, ModuleHandlerInterface $module_handler, EntityTypeManagerInterface $entity_type_manager, $attribute = MigrateDestination::class, $annotation = 'Drupal\migrate\Annotation\MigrateDestination') { + parent::__construct($type, $namespaces, $cache_backend, $module_handler, $attribute, $annotation); $this->entityTypeManager = $entity_type_manager; } diff --git a/core/modules/migrate/src/Plugin/MigratePluginManager.php b/core/modules/migrate/src/Plugin/MigratePluginManager.php index b3645dbdd0e4..9eba756ab1be 100644 --- a/core/modules/migrate/src/Plugin/MigratePluginManager.php +++ b/core/modules/migrate/src/Plugin/MigratePluginManager.php @@ -2,6 +2,8 @@ namespace Drupal\migrate\Plugin; +use Drupal\Component\Plugin\Attribute\AttributeInterface; +use Drupal\Component\Plugin\Attribute\PluginID; use Drupal\Component\Plugin\Factory\DefaultFactory; use Drupal\Core\Cache\CacheBackendInterface; use Drupal\Core\Extension\ModuleHandlerInterface; @@ -11,10 +13,10 @@ * Manages migrate plugins. * * @see hook_migrate_info_alter() - * @see \Drupal\migrate\Annotation\MigrateSource + * @see \Drupal\migrate\Attribute\MigrateSource * @see \Drupal\migrate\Plugin\MigrateSourceInterface * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase - * @see \Drupal\migrate\Annotation\MigrateProcessPlugin + * @see \Drupal\migrate\Attribute\MigrateProcess * @see \Drupal\migrate\Plugin\MigrateProcessInterface * @see \Drupal\migrate\Plugin\migrate\process\ProcessPluginBase * @see plugin_api @@ -36,12 +38,20 @@ class MigratePluginManager extends DefaultPluginManager implements MigratePlugin * Cache backend instance to use. * @param \Drupal\Core\Extension\ModuleHandlerInterface $module_handler * The module handler to invoke the alter hook with. + * @param string $attribute + * (optional) The attribute class name. Defaults to + * 'Drupal\Component\Plugin\Attribute\PluginID'. * @param string $annotation * (optional) The annotation class name. Defaults to * 'Drupal\Component\Annotation\PluginID'. */ - public function __construct($type, \Traversable $namespaces, CacheBackendInterface $cache_backend, ModuleHandlerInterface $module_handler, $annotation = 'Drupal\Component\Annotation\PluginID') { - parent::__construct("Plugin/migrate/$type", $namespaces, $module_handler, NULL, $annotation); + public function __construct($type, \Traversable $namespaces, CacheBackendInterface $cache_backend, ModuleHandlerInterface $module_handler, $attribute = PluginID::class, $annotation = 'Drupal\Component\Annotation\PluginID') { + if (!is_subclass_of($attribute, AttributeInterface::class)) { + // Backward compatibility. + $annotation = $attribute; + $attribute = PluginID::class; + } + parent::__construct("Plugin/migrate/$type", $namespaces, $module_handler, NULL, $attribute, $annotation); $this->alterInfo('migrate_' . $type . '_info'); $this->setCacheBackend($cache_backend, 'migrate_plugins_' . $type); } diff --git a/core/modules/migrate/src/Plugin/MigrateProcessInterface.php b/core/modules/migrate/src/Plugin/MigrateProcessInterface.php index 6b3832c89b6f..7d2d365d8817 100644 --- a/core/modules/migrate/src/Plugin/MigrateProcessInterface.php +++ b/core/modules/migrate/src/Plugin/MigrateProcessInterface.php @@ -15,7 +15,7 @@ * * @see \Drupal\migrate\Plugin\MigratePluginManager * @see \Drupal\migrate\ProcessPluginBase - * @see \Drupal\migrate\Annotation\MigrateProcessPlugin + * @see \Drupal\migrate\Attribute\MigrateProcess * @see plugin_api * * @ingroup migration diff --git a/core/modules/migrate/src/Plugin/MigrateSourceInterface.php b/core/modules/migrate/src/Plugin/MigrateSourceInterface.php index f33dc68cbc80..b05236ad369a 100644 --- a/core/modules/migrate/src/Plugin/MigrateSourceInterface.php +++ b/core/modules/migrate/src/Plugin/MigrateSourceInterface.php @@ -9,7 +9,7 @@ * Defines an interface for migrate sources. * * @see \Drupal\migrate\Plugin\MigratePluginManager - * @see \Drupal\migrate\Annotation\MigrateSource + * @see \Drupal\migrate\Attribute\MigrateSource * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase * @see plugin_api * diff --git a/core/modules/migrate/src/Plugin/MigrateSourcePluginManager.php b/core/modules/migrate/src/Plugin/MigrateSourcePluginManager.php index 965da5dd766e..74fb59b59447 100644 --- a/core/modules/migrate/src/Plugin/MigrateSourcePluginManager.php +++ b/core/modules/migrate/src/Plugin/MigrateSourcePluginManager.php @@ -4,8 +4,10 @@ use Drupal\Core\Cache\CacheBackendInterface; use Drupal\Core\Extension\ModuleHandlerInterface; -use Drupal\migrate\Plugin\Discovery\AnnotatedClassDiscoveryAutomatedProviders; use Drupal\Core\Plugin\Discovery\ContainerDerivativeDiscoveryDecorator; +use Drupal\migrate\Plugin\Discovery\AnnotatedClassDiscoveryAutomatedProviders; +use Drupal\migrate\Plugin\Discovery\AttributeClassDiscoveryAutomatedProviders; +use Drupal\migrate\Plugin\Discovery\AttributeDiscoveryWithAnnotationsAutomatedProviders; use Drupal\migrate\Plugin\Discovery\ProviderFilterDecorator; /** @@ -13,7 +15,7 @@ * * @see \Drupal\migrate\Plugin\MigrateSourceInterface * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase - * @see \Drupal\migrate\Annotation\MigrateSource + * @see \Drupal\migrate\Attribute\MigrateSource * @see plugin_api * * @ingroup migration @@ -35,7 +37,7 @@ class MigrateSourcePluginManager extends MigratePluginManager { * The module handler to invoke the alter hook with. */ public function __construct($type, \Traversable $namespaces, CacheBackendInterface $cache_backend, ModuleHandlerInterface $module_handler) { - parent::__construct($type, $namespaces, $cache_backend, $module_handler, 'Drupal\migrate\Annotation\MigrateSource'); + parent::__construct($type, $namespaces, $cache_backend, $module_handler, 'Drupal\migrate\Attribute\MigrateSource', 'Drupal\migrate\Annotation\MigrateSource'); } /** @@ -43,7 +45,30 @@ public function __construct($type, \Traversable $namespaces, CacheBackendInterfa */ protected function getDiscovery() { if (!$this->discovery) { - $discovery = new AnnotatedClassDiscoveryAutomatedProviders($this->subdir, $this->namespaces, $this->pluginDefinitionAnnotationName, $this->additionalAnnotationNamespaces); + if (isset($this->pluginDefinitionAttributeName) && isset($this->pluginDefinitionAnnotationName)) { + $discovery = new AttributeDiscoveryWithAnnotationsAutomatedProviders( + $this->subdir, + $this->namespaces, + $this->pluginDefinitionAttributeName, + $this->pluginDefinitionAnnotationName, + $this->additionalAnnotationNamespaces, + ); + } + elseif (isset($this->pluginDefinitionAttributeName)) { + $discovery = new AttributeClassDiscoveryAutomatedProviders( + $this->subdir, + $this->namespaces, + $this->pluginDefinitionAttributeName, + ); + } + else { + $discovery = new AnnotatedClassDiscoveryAutomatedProviders( + $this->subdir, + $this->namespaces, + $this->pluginDefinitionAnnotationName, + $this->additionalAnnotationNamespaces, + ); + } $this->discovery = new ContainerDerivativeDiscoveryDecorator($discovery); } return $this->discovery; diff --git a/core/modules/migrate/src/Plugin/migrate/destination/DestinationBase.php b/core/modules/migrate/src/Plugin/migrate/destination/DestinationBase.php index 4a7fc59d8270..5712613e218b 100644 --- a/core/modules/migrate/src/Plugin/migrate/destination/DestinationBase.php +++ b/core/modules/migrate/src/Plugin/migrate/destination/DestinationBase.php @@ -21,7 +21,7 @@ * information, refer to \Drupal\migrate\Plugin\MigrateDestinationInterface. * * @see \Drupal\migrate\Plugin\MigrateDestinationPluginManager - * @see \Drupal\migrate\Annotation\MigrateDestination + * @see \Drupal\migrate\Attribute\MigrateDestination * @see plugin_api * * @ingroup migration diff --git a/core/modules/migrate/src/Plugin/migrate/id_map/NullIdMap.php b/core/modules/migrate/src/Plugin/migrate/id_map/NullIdMap.php index 80c5da68a5a1..a6b2e3339e52 100644 --- a/core/modules/migrate/src/Plugin/migrate/id_map/NullIdMap.php +++ b/core/modules/migrate/src/Plugin/migrate/id_map/NullIdMap.php @@ -2,6 +2,7 @@ namespace Drupal\migrate\Plugin\migrate\id_map; +use Drupal\Component\Plugin\Attribute\PluginID; use Drupal\Core\Plugin\PluginBase; use Drupal\migrate\MigrateMessageInterface; use Drupal\migrate\Plugin\MigrateIdMapInterface; @@ -12,9 +13,8 @@ * Defines the null ID map implementation. * * This serves as a dummy in order to not store anything. - * - * @PluginID("null") */ +#[PluginID('null')] class NullIdMap extends PluginBase implements MigrateIdMapInterface { /** diff --git a/core/modules/migrate/src/Plugin/migrate/id_map/Sql.php b/core/modules/migrate/src/Plugin/migrate/id_map/Sql.php index b9261d098943..d66d3ce2fb2b 100644 --- a/core/modules/migrate/src/Plugin/migrate/id_map/Sql.php +++ b/core/modules/migrate/src/Plugin/migrate/id_map/Sql.php @@ -2,6 +2,7 @@ namespace Drupal\migrate\Plugin\migrate\id_map; +use Drupal\Component\Plugin\Attribute\PluginID; use Drupal\Core\Database\DatabaseException; use Drupal\Core\Database\DatabaseExceptionWrapper; use Drupal\Core\Database\Exception\SchemaTableKeyTooLargeException; @@ -30,9 +31,8 @@ * * It creates one map and one message table per migration entity to store the * relevant information. - * - * @PluginID("sql") */ +#[PluginID('sql')] class Sql extends PluginBase implements MigrateIdMapInterface, ContainerFactoryPluginInterface, HighestIdInterface { /** diff --git a/core/modules/migrate/src/Plugin/migrate/process/Explode.php b/core/modules/migrate/src/Plugin/migrate/process/Explode.php index 38a58ed4a405..c6d902492659 100644 --- a/core/modules/migrate/src/Plugin/migrate/process/Explode.php +++ b/core/modules/migrate/src/Plugin/migrate/process/Explode.php @@ -2,6 +2,7 @@ namespace Drupal\migrate\Plugin\migrate\process; +use Drupal\migrate\Attribute\MigrateProcess; use Drupal\migrate\ProcessPluginBase; use Drupal\migrate\MigrateException; use Drupal\migrate\MigrateExecutableInterface; @@ -86,11 +87,8 @@ * configuration, if foo is '', NULL or FALSE, then bar will be []. * * @see \Drupal\migrate\Plugin\MigrateProcessInterface - * - * @MigrateProcessPlugin( - * id = "explode" - * ) */ +#[MigrateProcess('explode')] class Explode extends ProcessPluginBase { /** diff --git a/core/modules/migrate/src/Plugin/migrate/source/EmbeddedDataSource.php b/core/modules/migrate/src/Plugin/migrate/source/EmbeddedDataSource.php index a61a951f3e05..251df66accba 100644 --- a/core/modules/migrate/src/Plugin/migrate/source/EmbeddedDataSource.php +++ b/core/modules/migrate/src/Plugin/migrate/source/EmbeddedDataSource.php @@ -2,6 +2,7 @@ namespace Drupal\migrate\Plugin\migrate\source; +use Drupal\migrate\Attribute\MigrateSource; use Drupal\migrate\Plugin\MigrationInterface; /** @@ -40,12 +41,11 @@ * * For additional configuration keys, refer to the parent class: * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase - * - * @MigrateSource( - * id = "embedded_data", - * source_module = "migrate" - * ) */ +#[MigrateSource( + id: 'embedded_data', + source_module: 'migrate' +)] class EmbeddedDataSource extends SourcePluginBase { /** diff --git a/core/modules/migrate/src/Plugin/migrate/source/EmptySource.php b/core/modules/migrate/src/Plugin/migrate/source/EmptySource.php index 2991aeec7a26..ee56876cdc0b 100644 --- a/core/modules/migrate/src/Plugin/migrate/source/EmptySource.php +++ b/core/modules/migrate/src/Plugin/migrate/source/EmptySource.php @@ -2,6 +2,8 @@ namespace Drupal\migrate\Plugin\migrate\source; +use Drupal\migrate\Attribute\MigrateSource; + /** * Source returning a row based on the constants provided. * @@ -21,12 +23,11 @@ * * For additional configuration keys, refer to the parent class: * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase - * - * @MigrateSource( - * id = "empty", - * source_module = "migrate" - * ) */ +#[MigrateSource( + id: 'empty', + source_module: 'migrate', +)] class EmptySource extends SourcePluginBase { /** diff --git a/core/modules/migrate/src/Plugin/migrate/source/SourcePluginBase.php b/core/modules/migrate/src/Plugin/migrate/source/SourcePluginBase.php index e6d5d786dd7f..e98d41975d03 100644 --- a/core/modules/migrate/src/Plugin/migrate/source/SourcePluginBase.php +++ b/core/modules/migrate/src/Plugin/migrate/source/SourcePluginBase.php @@ -105,7 +105,7 @@ * In this example, the constant 'foo' is defined with a value of 'bar'. It is * later used in the process pipeline to set the value of the field baz. * - * @see \Drupal\migrate\Annotation\MigrateSource + * @see \Drupal\migrate\Attribute\MigrateSource * @see \Drupal\migrate\Plugin\MigrateIdMapInterface * @see \Drupal\migrate\Plugin\MigratePluginManager * @see \Drupal\migrate\Plugin\MigrateSourceInterface diff --git a/core/modules/migrate/src/ProcessPluginBase.php b/core/modules/migrate/src/ProcessPluginBase.php index 5c6978ec335d..e7b1fcc83126 100644 --- a/core/modules/migrate/src/ProcessPluginBase.php +++ b/core/modules/migrate/src/ProcessPluginBase.php @@ -21,7 +21,7 @@ * @see https://www.drupal.org/node/2129651 * @see \Drupal\migrate\Plugin\MigratePluginManager * @see \Drupal\migrate\Plugin\MigrateProcessInterface - * @see \Drupal\migrate\Annotation\MigrateProcessPlugin + * @see \Drupal\migrate\Attribute\MigrateProcess * @see \Drupal\migrate\Plugin\migrate\process\SkipOnEmpty * @see d7_field_formatter_settings.yml * @see plugin_api diff --git a/core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/migrate_source_annotation_bc_test.info.yml b/core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/migrate_source_annotation_bc_test.info.yml new file mode 100644 index 000000000000..a87b6c047ad2 --- /dev/null +++ b/core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/migrate_source_annotation_bc_test.info.yml @@ -0,0 +1,5 @@ +name: 'Migrate module source annotation bc tests' +type: module +description: 'Support module for source plugin annotation discovery backwards compatibility tests' +package: Testing +version: VERSION diff --git a/core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/src/Plugin/migrate/source/MigrateSourceWithAnnotations.php b/core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/src/Plugin/migrate/source/MigrateSourceWithAnnotations.php new file mode 100644 index 000000000000..145adcaa59a8 --- /dev/null +++ b/core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/src/Plugin/migrate/source/MigrateSourceWithAnnotations.php @@ -0,0 +1,50 @@ +<?php + +namespace Drupal\migrate_source_annotation_bc_test\Plugin\migrate\source; + +use Drupal\migrate\Plugin\migrate\source\SourcePluginBase; + +/** + * A migration source plugin with annotations and a single provider. + * + * This plugin exists to test backwards compatibility of source plugin discovery + * for plugin classes using annotations. This class has no providers other than + * 'migrate_source_annotation_bc_test' and 'core'. This class and its annotation + * should remain until annotation support is completely removed. + * + * @MigrateSource( + * id = "annotated", + * source_module = "migrate" + * ) + */ +class MigrateSourceWithAnnotations extends SourcePluginBase { + + /** + * {@inheritdoc} + */ + public function fields() { + return []; + } + + /** + * {@inheritdoc} + */ + public function __toString() { + return 'Annotated'; + } + + /** + * {@inheritdoc} + */ + public function getIds() { + return []; + } + + /** + * {@inheritdoc} + */ + protected function initializeIterator() { + return new \ArrayIterator(); + } + +} diff --git a/core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/src/Plugin/migrate/source/MigrateSourceWithAnnotationsMultipleProviders.php b/core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/src/Plugin/migrate/source/MigrateSourceWithAnnotationsMultipleProviders.php new file mode 100644 index 000000000000..dabc2c3d70cc --- /dev/null +++ b/core/modules/migrate/tests/modules/migrate_source_annotation_bc_test/src/Plugin/migrate/source/MigrateSourceWithAnnotationsMultipleProviders.php @@ -0,0 +1,29 @@ +<?php + +namespace Drupal\migrate_source_annotation_bc_test\Plugin\migrate\source; + +use Drupal\migrate_drupal\Plugin\migrate\source\EmptySource; + +/** + * A migration source plugin with annotations and multiple providers. + * + * This plugin exists to test backwards compatibility of source plugin discovery + * for plugin classes using annotations. This class has an additional provider, + * because it extends a plugin in migrate_drupal. This class and its annotation + * should remain until annotation support is completely removed. + * + * @MigrateSource( + * id = "annotated_multiple_providers", + * source_module = "migrate" + * ) + */ +class MigrateSourceWithAnnotationsMultipleProviders extends EmptySource { + + /** + * {@inheritdoc} + */ + public function __toString() { + return 'Annotated multiple providers'; + } + +} diff --git a/core/modules/migrate/tests/src/Kernel/Plugin/source/MigrateSourceAnnotationDiscoveryTest.php b/core/modules/migrate/tests/src/Kernel/Plugin/source/MigrateSourceAnnotationDiscoveryTest.php new file mode 100644 index 000000000000..324a79706017 --- /dev/null +++ b/core/modules/migrate/tests/src/Kernel/Plugin/source/MigrateSourceAnnotationDiscoveryTest.php @@ -0,0 +1,60 @@ +<?php + +namespace Drupal\Tests\migrate\Kernel\Plugin\source; + +use Drupal\KernelTests\KernelTestBase; + +/** + * Tests discovery of source plugins with annotations. + * + * Migrate source plugins use a specific discovery class to accommodate multiple + * providers. This is a backwards compatibility test that discovery for plugin + * classes that have annotations still works even after all core plugins have + * been converted to attributes. + * + * @group migrate + */ +class MigrateSourceAnnotationDiscoveryTest extends KernelTestBase { + + /** + * {@inheritdoc} + */ + protected static $modules = ['migrate']; + + /** + * @covers \Drupal\migrate\Plugin\MigrateSourcePluginManager::getDefinitions + */ + public function testGetDefinitions(): void { + // First, test attribute-only discovery. + $expected = ['embedded_data', 'empty']; + $source_plugins = $this->container->get('plugin.manager.migrate.source')->getDefinitions(); + ksort($source_plugins); + $this->assertSame($expected, array_keys($source_plugins)); + + // Next, test discovery of both attributed and annotated plugins. The + // annotated plugin with multiple providers depends on migrate_drupal and + // should not be discovered with it uninstalled. + $expected = ['annotated', 'embedded_data', 'empty']; + $this->enableModules(['migrate_source_annotation_bc_test']); + $source_plugins = $this->container->get('plugin.manager.migrate.source')->getDefinitions(); + ksort($source_plugins); + $this->assertSame($expected, array_keys($source_plugins)); + + // Install migrate_drupal and now the annotated plugin that depends on it + // should be discovered. + $expected = [ + 'annotated', + 'annotated_multiple_providers', + 'embedded_data', + 'empty', + ]; + $this->enableModules(['migrate_drupal']); + $source_plugins = $this->container->get('plugin.manager.migrate.source')->getDefinitions(); + // Confirming here the that the source plugins that migrate and + // migrate_source_annotation_bc_test are discovered. There are additional + // plugins provided by migrate_drupal, but they do not need to be enumerated + // here. + $this->assertSame(array_diff($expected, array_keys($source_plugins)), []); + } + +} diff --git a/core/modules/migrate_drupal/migrate_drupal.services.yml b/core/modules/migrate_drupal/migrate_drupal.services.yml index 80444d04750b..9e3662341b98 100644 --- a/core/modules/migrate_drupal/migrate_drupal.services.yml +++ b/core/modules/migrate_drupal/migrate_drupal.services.yml @@ -6,6 +6,7 @@ services: - '@container.namespaces' - '@cache.discovery' - '@module_handler' + - '\Drupal\migrate_drupal\Attribute\MigrateField' - '\Drupal\migrate_drupal\Annotation\MigrateField' Drupal\migrate_drupal\Plugin\MigrateFieldPluginManagerInterface: '@plugin.manager.migrate.field' logger.channel.migrate_drupal: diff --git a/core/modules/migrate_drupal/src/Attribute/MigrateField.php b/core/modules/migrate_drupal/src/Attribute/MigrateField.php new file mode 100644 index 000000000000..ffb529e98000 --- /dev/null +++ b/core/modules/migrate_drupal/src/Attribute/MigrateField.php @@ -0,0 +1,74 @@ +<?php + +declare(strict_types=1); + +namespace Drupal\migrate_drupal\Attribute; + +use Drupal\Component\Plugin\Attribute\Plugin; + +/** + * Defines a field plugin attribute object. + * + * Field plugins are responsible for handling the migration of custom fields + * (provided by Field API in Drupal 7) to Drupal 8+. They are allowed to alter + * fieldable entity migrations when these migrations are being generated, and + * can compute destination field types for individual fields during the actual + * migration process. + * + * Plugin Namespace: Plugin\migrate\field + * + * For a working example, see + * \Drupal\datetime\Plugin\migrate\field\DateField + * + * @see \Drupal\migrate\Plugin\MigratePluginManager + * @see \Drupal\migrate_drupal\Plugin\MigrateFieldInterface; + * @see \Drupal\migrate_drupal\Plugin\migrate\field\FieldPluginBase + * @see plugin_api + * + * @ingroup migration + */ +#[\Attribute(\Attribute::TARGET_CLASS)] +class MigrateField extends Plugin { + + /** + * The plugin definition. + * + * @var array + */ + protected $definition; + + /** + * Constructs a migrate field attribute object. + * + * @param string $id + * A unique identifier for the field plugin. + * @param int[] $core + * (optional) The Drupal core version(s) this plugin applies to. + * @param int $weight + * (optional) The weight of this plugin relative to other plugins servicing + * the same field type and core version. The lowest weighted applicable + * plugin will be used for each field. + * @param string[] $type_map + * (optional) Map of D6 and D7 field types to D8+ field type plugin IDs. + * @param string|null $source_module + * (optional) Identifies the system providing the data the field plugin will + * read. The source_module is expected to be the name of a Drupal module + * that must be installed in the source database. + * @param string|null $destination_module + * (optional) Identifies the system handling the data the destination plugin + * will write. The destination_module is expected to be the name of a Drupal + * module on the destination site that must be installed. + * @param class-string|null $deriver + * (optional) The deriver class. + */ + public function __construct( + public readonly string $id, + public readonly array $core = [6], + public readonly int $weight = 0, + public readonly array $type_map = [], + public readonly ?string $source_module = NULL, + public readonly ?string $destination_module = NULL, + public readonly ?string $deriver = NULL + ) {} + +} diff --git a/core/modules/migrate_drupal/src/Plugin/MigrateFieldPluginManager.php b/core/modules/migrate_drupal/src/Plugin/MigrateFieldPluginManager.php index 5859ff3aa7dc..cce492cfc34c 100644 --- a/core/modules/migrate_drupal/src/Plugin/MigrateFieldPluginManager.php +++ b/core/modules/migrate_drupal/src/Plugin/MigrateFieldPluginManager.php @@ -11,7 +11,7 @@ * Plugin manager for migrate field plugins. * * @see \Drupal\migrate_drupal\Plugin\MigrateFieldInterface - * @see \Drupal\migrate\Annotation\MigrateField + * @see \Drupal\migrate\Attribute\MigrateField * @see plugin_api * * @ingroup migration @@ -48,7 +48,7 @@ class MigrateFieldPluginManager extends MigratePluginManager implements MigrateF * @throws \Drupal\Component\Plugin\Exception\PluginNotFoundException * If the plugin cannot be determined, such as if the field type is invalid. * - * @see \Drupal\migrate_drupal\Annotation\MigrateField + * @see \Drupal\migrate_drupal\Attribute\MigrateField */ public function getPluginIdFromFieldType($field_type, array $configuration = [], MigrationInterface $migration = NULL) { $core = static::DEFAULT_CORE_VERSION; diff --git a/core/modules/migrate_drupal/src/Plugin/migrate/field/FieldPluginBase.php b/core/modules/migrate_drupal/src/Plugin/migrate/field/FieldPluginBase.php index b62ac0f994cc..36a25211ece9 100644 --- a/core/modules/migrate_drupal/src/Plugin/migrate/field/FieldPluginBase.php +++ b/core/modules/migrate_drupal/src/Plugin/migrate/field/FieldPluginBase.php @@ -11,7 +11,7 @@ * The base class for all field plugins. * * @see \Drupal\migrate\Plugin\MigratePluginManager - * @see \Drupal\migrate_drupal\Annotation\MigrateField + * @see \Drupal\migrate_drupal\Attribute\MigrateField * @see \Drupal\migrate_drupal\Plugin\MigrateFieldInterface * @see plugin_api * diff --git a/core/modules/migrate_drupal/src/Plugin/migrate/source/ContentEntity.php b/core/modules/migrate_drupal/src/Plugin/migrate/source/ContentEntity.php index 5d80f6160a77..bc7ff421ff7a 100644 --- a/core/modules/migrate_drupal/src/Plugin/migrate/source/ContentEntity.php +++ b/core/modules/migrate_drupal/src/Plugin/migrate/source/ContentEntity.php @@ -9,6 +9,7 @@ use Drupal\Core\Entity\EntityTypeBundleInfoInterface; use Drupal\Core\Entity\EntityTypeManagerInterface; use Drupal\Core\Plugin\ContainerFactoryPluginInterface; +use Drupal\migrate\Attribute\MigrateSource; use Drupal\migrate\EntityFieldDefinitionTrait; use Drupal\migrate\Plugin\migrate\source\SourcePluginBase; use Drupal\migrate\Plugin\MigrateSourceInterface; @@ -61,13 +62,12 @@ * * For additional configuration keys, refer to the parent class: * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase - * - * @MigrateSource( - * id = "content_entity", - * source_module = "migrate_drupal", - * deriver = "\Drupal\migrate_drupal\Plugin\migrate\source\ContentEntityDeriver", - * ) - */ + */ +#[MigrateSource( + id: "content_entity", + source_module: "migrate_drupal", + deriver: ContentEntityDeriver::class, +)] class ContentEntity extends SourcePluginBase implements ContainerFactoryPluginInterface { use EntityFieldDefinitionTrait; -- GitLab