Skip to content

Commit

Permalink
add support for multiple service annotations
Browse files Browse the repository at this point in the history
  • Loading branch information
wodka committed Feb 14, 2016
1 parent 36f8b06 commit 4ea63dc
Show file tree
Hide file tree
Showing 5 changed files with 118 additions and 42 deletions.
50 changes: 45 additions & 5 deletions Metadata/ClassMetadata.php
Original file line number Diff line number Diff line change
Expand Up @@ -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)
{
Expand All @@ -60,6 +101,7 @@ public function serialize()
$this->initMethod,
parent::serialize(),
$this->environments,
$this->services,
));
}

Expand All @@ -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);
}
}
14 changes: 8 additions & 6 deletions Metadata/Driver/AnnotationDriver.php
Original file line number Diff line number Diff line change
Expand Up @@ -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) {
Expand Down
69 changes: 38 additions & 31 deletions Metadata/MetadataConverter.php
Original file line number Diff line number Diff line change
Expand Up @@ -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)
{}
}
14 changes: 14 additions & 0 deletions Tests/Metadata/Driver/AnnotationDriverTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -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());
Expand Down
13 changes: 13 additions & 0 deletions Tests/Metadata/Driver/Fixture/MultiService.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
<?php

namespace JMS\DiExtraBundle\Tests\Metadata\Driver\Fixture;

use JMS\DiExtraBundle as DI; // Use this alias in order to not have this class picked up by the finder

/**
* @DI\Annotation\Service("first.service")
* @DI\Annotation\Service("second.service")
*/
class MultiService
{
}

0 comments on commit 4ea63dc

Please sign in to comment.