From 4ea63dc82d7434bf8607e2fb264d51491b140a2b Mon Sep 17 00:00:00 2001 From: wodka Date: Sun, 14 Feb 2016 18:48:42 +0100 Subject: [PATCH] add support for multiple service annotations https://github.com/schmittjoh/JMSDiExtraBundle/issues/232 --- Metadata/ClassMetadata.php | 50 ++++++++++++-- Metadata/Driver/AnnotationDriver.php | 14 ++-- Metadata/MetadataConverter.php | 69 ++++++++++--------- .../Metadata/Driver/AnnotationDriverTest.php | 14 ++++ .../Metadata/Driver/Fixture/MultiService.php | 13 ++++ 5 files changed, 118 insertions(+), 42 deletions(-) create mode 100644 Tests/Metadata/Driver/Fixture/MultiService.php diff --git a/Metadata/ClassMetadata.php b/Metadata/ClassMetadata.php index 7cdbbf5..9f0a7ae 100644 --- a/Metadata/ClassMetadata.php +++ b/Metadata/ClassMetadata.php @@ -34,6 +34,47 @@ class ClassMetadata extends BaseClassMetadata public $properties = array(); public $initMethod; public $environments = array(); + public $services = []; + + /** + * on first call also populate legacy fields + * + * @param string[] $service + */ + public function addService(array $service) + { + if (empty($this->id)) { + $this->id = $service['id']; + $this->parent = $service['parent']; + $this->public = $service['public']; + $this->scope = $service['scope']; + $this->abstract = $service['abstract']; + // TODO update call for other tags (there are several pull requests) + } + + $this->services[$service['id']] = $service; + } + + /** + * get list of defined services, use fallback of original fields + * + * return array[] + */ + public function getServices() + { + // TODO remove fallback for next major version + if (empty($this->services) || !isset($this->services[$this->id])) { + $this->services[] = array( + 'id' => $this->id, + 'parent' => $this->parent, + 'public' => $this->public, + 'scope' => $this->scope, + 'abstract' => $this->abstract, + ); + } + + return $this->services; + } public function isLoadedInEnvironment($env) { @@ -60,6 +101,7 @@ public function serialize() $this->initMethod, parent::serialize(), $this->environments, + $this->services, )); } @@ -79,13 +121,11 @@ public function unserialize($str) $this->lookupMethods, $this->properties, $this->initMethod, - $parentStr + $parentStr, + $this->environments, + $this->services, ) = $data; - if (isset($data[12])) { - $this->environments = $data[12]; - } - parent::unserialize($parentStr); } } diff --git a/Metadata/Driver/AnnotationDriver.php b/Metadata/Driver/AnnotationDriver.php index 1cbd27b..6c689d6 100644 --- a/Metadata/Driver/AnnotationDriver.php +++ b/Metadata/Driver/AnnotationDriver.php @@ -71,16 +71,18 @@ public function loadMetadataForClass(\ReflectionClass $class) foreach ($this->reader->getClassAnnotations($class) as $annot) { if ($annot instanceof Service) { + $service = array(); if (null === $annot->id) { - $metadata->id = $this->namingStrategy->classToServiceName($className); + $service['id'] = $this->namingStrategy->classToServiceName($className); } else { - $metadata->id = $annot->id; + $service['id'] = $annot->id; } - $metadata->parent = $annot->parent; - $metadata->public = $annot->public; - $metadata->scope = $annot->scope; - $metadata->abstract = $annot->abstract; + $service['parent'] = $annot->parent; + $service['public'] = $annot->public; + $service['scope'] = $annot->scope; + $service['abstract'] = $annot->abstract; + $metadata->addService($service); } else if ($annot instanceof Tag) { $metadata->tags[$annot->name][] = $annot->attributes; } else if ($annot instanceof Validator) { diff --git a/Metadata/MetadataConverter.php b/Metadata/MetadataConverter.php index cb67ca7..a1ada92 100644 --- a/Metadata/MetadataConverter.php +++ b/Metadata/MetadataConverter.php @@ -37,49 +37,56 @@ public function convert(ClassHierarchyMetadata $metadata) $definitions = array(); $previous = null; + /** @var ClassMetadata $classMetadata */ foreach ($metadata->classMetadata as $classMetadata) { - if (null === $previous && null === $classMetadata->parent) { - $definition = new Definition(); - } else { - $definition = new DefinitionDecorator( - $classMetadata->parent ?: $previous->id - ); - } + foreach ($classMetadata->getServices() as $service) { + if (null === $previous && null === @$service['parent']) { + $definition = new Definition(); + } else { + $definition = new DefinitionDecorator( + @$service['parent'] ?: $previous->id + ); + } - $definition->setClass($classMetadata->name); - if (null !== $classMetadata->scope) { - $definition->setScope($classMetadata->scope); - } - if (null !== $classMetadata->public) { - $definition->setPublic($classMetadata->public); - } - if (null !== $classMetadata->abstract) { - $definition->setAbstract($classMetadata->abstract); - } - if (null !== $classMetadata->arguments) { - $definition->setArguments($classMetadata->arguments); - } + $definition->setClass($classMetadata->name); + if (null !== @$service['scope']) { + $definition->setScope(@$service['scope']); + } + if (null !== @$service['public']) { + $definition->setPublic(@$service['public']); + } + if (null !== @$service['abstract']) { + $definition->setAbstract(@$service['abstract']); + } + if (null !== $classMetadata->arguments) { + $definition->setArguments($classMetadata->arguments); + } - $definition->setMethodCalls($classMetadata->methodCalls); - $definition->setTags($classMetadata->tags); - $definition->setProperties($classMetadata->properties); + $definition->setMethodCalls($classMetadata->methodCalls); + $definition->setTags($classMetadata->tags); + $definition->setProperties($classMetadata->properties); - if (null === $classMetadata->id) { - $classMetadata->id = '_jms_di_extra.unnamed.service_'.$count++; - } + if (null === @$service['id']) { + $classMetadata->id = '_jms_di_extra.unnamed.service_' . $count++; + } + + if ($classMetadata->initMethod) { + if (!method_exists($definition, 'setInitMethod')) { + throw new \RuntimeException(sprintf('@AfterSetup is not available on your Symfony version.')); + } - if ($classMetadata->initMethod) { - if (!method_exists($definition, 'setInitMethod')) { - throw new \RuntimeException(sprintf('@AfterSetup is not available on your Symfony version.')); + $definition->setInitMethod($classMetadata->initMethod); } - $definition->setInitMethod($classMetadata->initMethod); + $definitions[$classMetadata->id] = $definition; } - $definitions[$classMetadata->id] = $definition; $previous = $classMetadata; } return $definitions; } + + protected function buildServices(ClassMetadata $classMetadata, $previous) + {} } diff --git a/Tests/Metadata/Driver/AnnotationDriverTest.php b/Tests/Metadata/Driver/AnnotationDriverTest.php index 459657f..3cdf4dc 100644 --- a/Tests/Metadata/Driver/AnnotationDriverTest.php +++ b/Tests/Metadata/Driver/AnnotationDriverTest.php @@ -31,6 +31,20 @@ public function testFormTypeWithExplicitAlias() ), $metadata->tags); } + public function testMultiService() + { + $metadata = $this->getDriver()->loadMetadataForClass(new \ReflectionClass('JMS\DiExtraBundle\Tests\Metadata\Driver\Fixture\MultiService')); + + $this->assertEquals(array( + 'first.service' => array( + 'id' => 'first.service', + ), + 'second.service' => array( + 'id' => 'first.service', + ) + ), $metadata->getServices()); + } + private function getDriver() { return new AnnotationDriver(new AnnotationReader(), new DefaultNamingStrategy()); diff --git a/Tests/Metadata/Driver/Fixture/MultiService.php b/Tests/Metadata/Driver/Fixture/MultiService.php new file mode 100644 index 0000000..dbdc045 --- /dev/null +++ b/Tests/Metadata/Driver/Fixture/MultiService.php @@ -0,0 +1,13 @@ +