diff --git a/src/Symfony/Bundle/FrameworkBundle/DependencyInjection/FrameworkExtension.php b/src/Symfony/Bundle/FrameworkBundle/DependencyInjection/FrameworkExtension.php
index a29a0ceadcc5d..bf529ab5c8bc3 100644
--- a/src/Symfony/Bundle/FrameworkBundle/DependencyInjection/FrameworkExtension.php
+++ b/src/Symfony/Bundle/FrameworkBundle/DependencyInjection/FrameworkExtension.php
@@ -492,6 +492,7 @@ private function registerWorkflowConfiguration(array $config, ContainerBuilder $
foreach ($config['workflows'] as $name => $workflow) {
$type = $workflow['type'];
+ $workflowId = sprintf('%s.%s', $type, $name);
// Process Metadata (workflow + places (transition is done in the "create transition" block))
$metadataStoreDefinition = new Definition(Workflow\Metadata\InMemoryMetadataStore::class, array(array(), array(), null));
@@ -510,28 +511,53 @@ private function registerWorkflowConfiguration(array $config, ContainerBuilder $
// Create transitions
$transitions = array();
+ $guardsConfiguration = array();
$transitionsMetadataDefinition = new Definition(\SplObjectStorage::class);
- foreach ($workflow['transitions'] as $transition) {
+ // Global transition counter per workflow
+ $transitionCounter = 0;
+ foreach (array_values($workflow['transitions']) as $number => $transition) {
if ('workflow' === $type) {
$transitionDefinition = new Definition(Workflow\Transition::class, array($transition['name'], $transition['from'], $transition['to']));
- $transitions[] = $transitionDefinition;
+ $transitionId = sprintf('%s.transition.%s', $workflowId, $transitionCounter++);
+ $container->setDefinition($transitionId, $transitionDefinition);
+ $transitions[] = new Reference($transitionId);
+
if ($transition['metadata']) {
$transitionsMetadataDefinition->addMethodCall('attach', array(
$transitionDefinition,
$transition['metadata'],
));
}
+
+ if (isset($transition['guard'])) {
+ $configuration = new Definition(Workflow\EventListener\GuardExpression::class);
+ $configuration->addArgument(new Reference($transitionId));
+ $configuration->addArgument($transition['guard']);
+ $eventName = sprintf('workflow.%s.guard.%s', $name, $transition['name']);
+ $guardsConfiguration[$eventName][] = $configuration;
+ }
} elseif ('state_machine' === $type) {
foreach ($transition['from'] as $from) {
foreach ($transition['to'] as $to) {
$transitionDefinition = new Definition(Workflow\Transition::class, array($transition['name'], $from, $to));
- $transitions[] = $transitionDefinition;
+ $transitionId = sprintf('%s.transition.%s', $workflowId, $transitionCounter++);
+ $container->setDefinition($transitionId, $transitionDefinition);
+ $transitions[] = new Reference($transitionId);
+
if ($transition['metadata']) {
$transitionsMetadataDefinition->addMethodCall('attach', array(
$transitionDefinition,
$transition['metadata'],
));
}
+
+ if (isset($transition['guard'])) {
+ $configuration = new Definition(Workflow\EventListener\GuardExpression::class);
+ $configuration->addArgument(new Reference($transitionId));
+ $configuration->addArgument($transition['guard']);
+ $eventName = sprintf('workflow.%s.guard.%s', $name, $transition['name']);
+ $guardsConfiguration[$eventName][] = $configuration;
+ }
}
}
}
@@ -542,7 +568,6 @@ private function registerWorkflowConfiguration(array $config, ContainerBuilder $
$places = array_map(function (array $place) {
return $place['name'];
}, $workflow['places']);
-
// Create a Definition
$definitionDefinition = new Definition(Workflow\Definition::class);
$definitionDefinition->setPublic(false);
@@ -567,7 +592,7 @@ private function registerWorkflowConfiguration(array $config, ContainerBuilder $
}
// Create Workflow
- $workflowId = sprintf('%s.%s', $type, $name);
+
$workflowDefinition = new ChildDefinition(sprintf('%s.abstract', $type));
$workflowDefinition->replaceArgument(0, new Reference(sprintf('%s.definition', $workflowId)));
if (isset($markingStoreDefinition)) {
@@ -605,12 +630,8 @@ private function registerWorkflowConfiguration(array $config, ContainerBuilder $
// Add Guard Listener
$guard = new Definition(Workflow\EventListener\GuardListener::class);
$guard->setPrivate(true);
- $configuration = array();
- foreach ($workflow['transitions'] as $transitionName => $config) {
- if (!isset($config['guard'])) {
- continue;
- }
+ if ($guardsConfiguration) {
if (!class_exists(ExpressionLanguage::class)) {
throw new LogicException('Cannot guard workflows as the ExpressionLanguage component is not installed.');
}
@@ -619,13 +640,8 @@ private function registerWorkflowConfiguration(array $config, ContainerBuilder $
throw new LogicException('Cannot guard workflows as the Security component is not installed.');
}
- $eventName = sprintf('workflow.%s.guard.%s', $name, $transitionName);
- $guard->addTag('kernel.event_listener', array('event' => $eventName, 'method' => 'onTransition'));
- $configuration[$eventName] = $config['guard'];
- }
- if ($configuration) {
$guard->setArguments(array(
- $configuration,
+ $guardsConfiguration,
new Reference('workflow.security.expression_language'),
new Reference('security.token_storage'),
new Reference('security.authorization_checker'),
@@ -634,6 +650,10 @@ private function registerWorkflowConfiguration(array $config, ContainerBuilder $
new Reference('validator', ContainerInterface::NULL_ON_INVALID_REFERENCE),
));
+ foreach ($guardsConfiguration as $eventName => $config) {
+ $guard->addTag('kernel.event_listener', array('event' => $eventName, 'method' => 'onTransition'));
+ }
+
$container->setDefinition(sprintf('%s.listener.guard', $workflowId), $guard);
$container->setParameter('workflow.has_guard_listeners', true);
}
diff --git a/src/Symfony/Bundle/FrameworkBundle/Resources/config/schema/symfony-1.0.xsd b/src/Symfony/Bundle/FrameworkBundle/Resources/config/schema/symfony-1.0.xsd
index eeebffb1d3ad7..4be4d7d670313 100644
--- a/src/Symfony/Bundle/FrameworkBundle/Resources/config/schema/symfony-1.0.xsd
+++ b/src/Symfony/Bundle/FrameworkBundle/Resources/config/schema/symfony-1.0.xsd
@@ -311,6 +311,7 @@
+
diff --git a/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/Fixtures/php/workflow_with_guard_expression.php b/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/Fixtures/php/workflow_with_guard_expression.php
new file mode 100644
index 0000000000000..89c86339afe15
--- /dev/null
+++ b/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/Fixtures/php/workflow_with_guard_expression.php
@@ -0,0 +1,51 @@
+loadFromExtension('framework', array(
+ 'workflows' => array(
+ 'article' => array(
+ 'type' => 'workflow',
+ 'marking_store' => array(
+ 'type' => 'multiple_state',
+ ),
+ 'supports' => array(
+ FrameworkExtensionTest::class,
+ ),
+ 'initial_place' => 'draft',
+ 'places' => array(
+ 'draft',
+ 'wait_for_journalist',
+ 'approved_by_journalist',
+ 'wait_for_spellchecker',
+ 'approved_by_spellchecker',
+ 'published',
+ ),
+ 'transitions' => array(
+ 'request_review' => array(
+ 'from' => 'draft',
+ 'to' => array('wait_for_journalist', 'wait_for_spellchecker'),
+ ),
+ 'journalist_approval' => array(
+ 'from' => 'wait_for_journalist',
+ 'to' => 'approved_by_journalist',
+ ),
+ 'spellchecker_approval' => array(
+ 'from' => 'wait_for_spellchecker',
+ 'to' => 'approved_by_spellchecker',
+ ),
+ 'publish' => array(
+ 'from' => array('approved_by_journalist', 'approved_by_spellchecker'),
+ 'to' => 'published',
+ 'guard' => '!!true',
+ ),
+ 'publish_editor_in_chief' => array(
+ 'name' => 'publish',
+ 'from' => 'draft',
+ 'to' => 'published',
+ 'guard' => '!!false',
+ ),
+ ),
+ ),
+ ),
+));
diff --git a/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/Fixtures/xml/workflow_with_guard_expression.xml b/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/Fixtures/xml/workflow_with_guard_expression.xml
new file mode 100644
index 0000000000000..cf129e45c33c0
--- /dev/null
+++ b/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/Fixtures/xml/workflow_with_guard_expression.xml
@@ -0,0 +1,48 @@
+
+
+
+
+
+
+
+ a
+ a
+
+ Symfony\Bundle\FrameworkBundle\Tests\DependencyInjection\FrameworkExtensionTest
+
+
+
+
+
+
+
+ draft
+ wait_for_journalist
+ wait_for_spellchecker
+
+
+ wait_for_journalist
+ approved_by_journalist
+
+
+ wait_for_spellchecker
+ approved_by_spellchecker
+
+
+ approved_by_journalist
+ approved_by_spellchecker
+ published
+ !!true
+
+
+ draft
+ published
+ !!false
+
+
+
+
diff --git a/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/Fixtures/yml/workflow_with_guard_expression.yml b/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/Fixtures/yml/workflow_with_guard_expression.yml
new file mode 100644
index 0000000000000..458cb4ae1ee77
--- /dev/null
+++ b/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/Fixtures/yml/workflow_with_guard_expression.yml
@@ -0,0 +1,35 @@
+framework:
+ workflows:
+ article:
+ type: workflow
+ marking_store:
+ type: multiple_state
+ supports:
+ - Symfony\Bundle\FrameworkBundle\Tests\DependencyInjection\FrameworkExtensionTest
+ initial_place: draft
+ places:
+ - draft
+ - wait_for_journalist
+ - approved_by_journalist
+ - wait_for_spellchecker
+ - approved_by_spellchecker
+ - published
+ transitions:
+ request_review:
+ from: [draft]
+ to: [wait_for_journalist, wait_for_spellchecker]
+ journalist_approval:
+ from: [wait_for_journalist]
+ to: [approved_by_journalist]
+ spellchecker_approval:
+ from: [wait_for_spellchecker]
+ to: [approved_by_spellchecker]
+ publish:
+ from: [approved_by_journalist, approved_by_spellchecker]
+ to: [published]
+ guard: "!!true"
+ publish_editor_in_chief:
+ name: publish
+ from: [draft]
+ to: [published]
+ guard: "!!false"
diff --git a/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/FrameworkExtensionTest.php b/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/FrameworkExtensionTest.php
index 69d5647e2fec4..185a1bcd3c84e 100644
--- a/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/FrameworkExtensionTest.php
+++ b/src/Symfony/Bundle/FrameworkBundle/Tests/DependencyInjection/FrameworkExtensionTest.php
@@ -322,14 +322,84 @@ public function testWorkflowMultipleTransitionsWithSameName()
$this->assertCount(5, $transitions);
- $this->assertSame('request_review', $transitions[0]->getArgument(0));
- $this->assertSame('journalist_approval', $transitions[1]->getArgument(0));
- $this->assertSame('spellchecker_approval', $transitions[2]->getArgument(0));
- $this->assertSame('publish', $transitions[3]->getArgument(0));
- $this->assertSame('publish', $transitions[4]->getArgument(0));
-
- $this->assertSame(array('approved_by_journalist', 'approved_by_spellchecker'), $transitions[3]->getArgument(1));
- $this->assertSame(array('draft'), $transitions[4]->getArgument(1));
+ $this->assertSame('workflow.article.transition.0', (string) $transitions[0]);
+ $this->assertSame(array(
+ 'request_review',
+ array(
+ 'draft',
+ ),
+ array(
+ 'wait_for_journalist', 'wait_for_spellchecker',
+ ),
+ ), $container->getDefinition($transitions[0])->getArguments());
+
+ $this->assertSame('workflow.article.transition.1', (string) $transitions[1]);
+ $this->assertSame(array(
+ 'journalist_approval',
+ array(
+ 'wait_for_journalist',
+ ),
+ array(
+ 'approved_by_journalist',
+ ),
+ ), $container->getDefinition($transitions[1])->getArguments());
+
+ $this->assertSame('workflow.article.transition.2', (string) $transitions[2]);
+ $this->assertSame(array(
+ 'spellchecker_approval',
+ array(
+ 'wait_for_spellchecker',
+ ),
+ array(
+ 'approved_by_spellchecker',
+ ),
+ ), $container->getDefinition($transitions[2])->getArguments());
+
+ $this->assertSame('workflow.article.transition.3', (string) $transitions[3]);
+ $this->assertSame(array(
+ 'publish',
+ array(
+ 'approved_by_journalist',
+ 'approved_by_spellchecker',
+ ),
+ array(
+ 'published',
+ ),
+ ), $container->getDefinition($transitions[3])->getArguments());
+
+ $this->assertSame('workflow.article.transition.4', (string) $transitions[4]);
+ $this->assertSame(array(
+ 'publish',
+ array(
+ 'draft',
+ ),
+ array(
+ 'published',
+ ),
+ ), $container->getDefinition($transitions[4])->getArguments());
+ }
+
+ public function testGuardExpressions()
+ {
+ $container = $this->createContainerFromFile('workflow_with_guard_expression');
+
+ $this->assertTrue($container->hasDefinition('workflow.article.listener.guard'), 'Workflow guard listener is registered as a service');
+ $this->assertTrue($container->hasParameter('workflow.has_guard_listeners'), 'Workflow guard listeners parameter exists');
+ $this->assertTrue(true === $container->getParameter('workflow.has_guard_listeners'), 'Workflow guard listeners parameter is enabled');
+ $guardDefinition = $container->getDefinition('workflow.article.listener.guard');
+ $this->assertSame(array(
+ array(
+ 'event' => 'workflow.article.guard.publish',
+ 'method' => 'onTransition',
+ ),
+ ), $guardDefinition->getTag('kernel.event_listener'));
+ $guardsConfiguration = $guardDefinition->getArgument(0);
+ $this->assertTrue(1 === \count($guardsConfiguration), 'Workflow guard configuration contains one element per transition name');
+ $transitionGuardExpressions = $guardsConfiguration['workflow.article.guard.publish'];
+ $this->assertSame('workflow.article.transition.3', (string) $transitionGuardExpressions[0]->getArgument(0));
+ $this->assertSame('!!true', $transitionGuardExpressions[0]->getArgument(1));
+ $this->assertSame('workflow.article.transition.4', (string) $transitionGuardExpressions[1]->getArgument(0));
+ $this->assertSame('!!false', $transitionGuardExpressions[1]->getArgument(1));
}
public function testWorkflowServicesCanBeEnabled()
diff --git a/src/Symfony/Component/Workflow/EventListener/GuardExpression.php b/src/Symfony/Component/Workflow/EventListener/GuardExpression.php
new file mode 100644
index 0000000000000..cf3b8c7e18694
--- /dev/null
+++ b/src/Symfony/Component/Workflow/EventListener/GuardExpression.php
@@ -0,0 +1,37 @@
+
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Symfony\Component\Workflow\EventListener;
+
+use Symfony\Component\Workflow\Transition;
+
+class GuardExpression
+{
+ private $transition;
+
+ private $expression;
+
+ public function getTransition(): Transition
+ {
+ return $this->transition;
+ }
+
+ public function getExpression(): string
+ {
+ return $this->expression;
+ }
+
+ public function __construct(Transition $transition, string $expression)
+ {
+ $this->transition = $transition;
+ $this->expression = $expression;
+ }
+}
diff --git a/src/Symfony/Component/Workflow/EventListener/GuardListener.php b/src/Symfony/Component/Workflow/EventListener/GuardListener.php
index 912dc5dada540..4f1c229e51b6f 100644
--- a/src/Symfony/Component/Workflow/EventListener/GuardListener.php
+++ b/src/Symfony/Component/Workflow/EventListener/GuardListener.php
@@ -50,8 +50,21 @@ public function onTransition(GuardEvent $event, $eventName)
return;
}
- $expression = $this->configuration[$eventName];
+ $eventConfiguration = (array) $this->configuration[$eventName];
+ foreach ($eventConfiguration as $guard) {
+ if ($guard instanceof GuardExpression) {
+ if ($guard->getTransition() !== $event->getTransition()) {
+ continue;
+ }
+ $this->validateGuardExpression($event, $guard->getExpression());
+ } else {
+ $this->validateGuardExpression($event, $guard);
+ }
+ }
+ }
+ private function validateGuardExpression(GuardEvent $event, string $expression)
+ {
if (!$this->expressionLanguage->evaluate($expression, $this->getVariables($event))) {
$blocker = TransitionBlocker::createBlockedByExpressionGuardListener($expression);
$event->addTransitionBlocker($blocker);
diff --git a/src/Symfony/Component/Workflow/Tests/EventListener/GuardListenerTest.php b/src/Symfony/Component/Workflow/Tests/EventListener/GuardListenerTest.php
index f0cc707220cbf..253f21743b9b9 100644
--- a/src/Symfony/Component/Workflow/Tests/EventListener/GuardListenerTest.php
+++ b/src/Symfony/Component/Workflow/Tests/EventListener/GuardListenerTest.php
@@ -11,6 +11,7 @@
use Symfony\Component\Validator\Validator\ValidatorInterface;
use Symfony\Component\Workflow\Event\GuardEvent;
use Symfony\Component\Workflow\EventListener\ExpressionLanguage;
+use Symfony\Component\Workflow\EventListener\GuardExpression;
use Symfony\Component\Workflow\EventListener\GuardListener;
use Symfony\Component\Workflow\Marking;
use Symfony\Component\Workflow\Transition;
@@ -21,12 +22,17 @@ class GuardListenerTest extends TestCase
private $authenticationChecker;
private $validator;
private $listener;
+ private $transition;
protected function setUp()
{
$configuration = array(
'test_is_granted' => 'is_granted("something")',
'test_is_valid' => 'is_valid(subject)',
+ 'test_expression' => array(
+ new GuardExpression($this->getTransition(true), '!is_valid(subject)'),
+ new GuardExpression($this->getTransition(true), 'is_valid(subject)'),
+ ),
);
$expressionLanguage = new ExpressionLanguage();
$token = $this->getMockBuilder(TokenInterface::class)->getMock();
@@ -97,11 +103,38 @@ public function testWithValidatorSupportedEventAndAccept()
$this->assertFalse($event->isBlocked());
}
- private function createEvent()
+ public function testWithGuardExpressionWithNotSupportedTransition()
+ {
+ $event = $this->createEvent(true);
+ $this->configureValidator(false, false);
+ $this->listener->onTransition($event, 'test_expression');
+
+ $this->assertFalse($event->isBlocked());
+ }
+
+ public function testWithGuardExpressionWithSupportedTransition()
+ {
+ $event = $this->createEvent();
+ $this->configureValidator(true, true);
+ $this->listener->onTransition($event, 'test_expression');
+
+ $this->assertFalse($event->isBlocked());
+ }
+
+ public function testGuardExpressionBlocks()
+ {
+ $event = $this->createEvent();
+ $this->configureValidator(true, false);
+ $this->listener->onTransition($event, 'test_expression');
+
+ $this->assertTrue($event->isBlocked());
+ }
+
+ private function createEvent($newTransition = false)
{
$subject = new \stdClass();
$subject->marking = new Marking();
- $transition = new Transition('name', 'from', 'to');
+ $transition = $this->getTransition($newTransition);
$workflow = $this->getMockBuilder(WorkflowInterface::class)->getMock();
@@ -143,4 +176,13 @@ private function configureValidator($isUsed, $valid = true)
->willReturn($valid ? array() : array('a violation'))
;
}
+
+ private function getTransition($new = false)
+ {
+ if ($new || !$this->transition) {
+ $this->transition = new Transition('name', 'from', 'to');
+ }
+
+ return $this->transition;
+ }
}