Issue #1921544 by kim.pepper, EclipseGc, xjm, katbailey, arnested, dawehner, neclimdul: Create a Current Path Condition.

8.0.x
Alex Pott 2014-06-06 14:49:51 -05:00
parent c993e0df92
commit acb97e876b
2 changed files with 283 additions and 0 deletions

View File

@ -0,0 +1,149 @@
<?php
/**
* @file
* Contains \Drupal\system\Plugin\Condition\RequestPath.
*/
namespace Drupal\system\Plugin\Condition;
use Drupal\Component\Utility\Unicode;
use Drupal\Core\Condition\ConditionPluginBase;
use Drupal\Core\Path\AliasManagerInterface;
use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
use Drupal\Core\StringTranslation\StringTranslationTrait;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\HttpFoundation\RequestStack;
/**
* Provides a 'Request Path' condition.
*
* @Condition(
* id = "request_path",
* label = @Translation("Request Path"),
* )
*/
class RequestPath extends ConditionPluginBase implements ContainerFactoryPluginInterface {
/**
* An alias manager to find the alias for the current system path.
*
* @var \Drupal\Core\Path\AliasManagerInterface
*/
protected $aliasManager;
/**
* The request stack.
*
* @var \Symfony\Component\HttpFoundation\RequestStack
*/
protected $requestStack;
/**
* Constructs a RequestPath condition plugin.
*
* @param \Drupal\Core\Path\AliasManagerInterface $alias_manager
* An alias manager to find the alias for the current system path.
* @param \Symfony\Component\HttpFoundation\RequestStack $request_stack
* The request stack.
* @param array $configuration
* A configuration array containing information about the plugin instance.
* @param string $plugin_id
* The plugin_id for the plugin instance.
* @param array $plugin_definition
* The plugin implementation definition.
*/
public function __construct(AliasManagerInterface $alias_manager, RequestStack $request_stack, array $configuration, $plugin_id, array $plugin_definition) {
parent::__construct($configuration, $plugin_id, $plugin_definition);
$this->aliasManager = $alias_manager;
$this->requestStack = $request_stack;
}
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
return new static(
$container->get('path.alias_manager.cached'),
$container->get('request_stack'),
$configuration,
$plugin_id,
$plugin_definition);
}
/**
* {@inheritdoc}
*/
public function defaultConfiguration() {
return array('pages' => '');
}
/**
* {@inheritdoc}
*/
public function buildConfigurationForm(array $form, array &$form_state) {
$form['pages'] = array(
'#type' => 'textarea',
'#title' => $this->t('Pages'),
'#default_value' => $this->configuration['pages'],
'#description' => $this->t("Specify pages by using their paths. Enter one path per line. The '*' character is a wildcard. Example paths are %user for the current user's page and %user-wildcard for every user page. %front is the front page.", array(
'%user' => 'user',
'%user-wildcard' => 'user/*',
'%front' => '<front>',
)),
);
return parent::buildConfigurationForm($form, $form_state);
}
/**
* {@inheritdoc}
*/
public function submitConfigurationForm(array &$form, array &$form_state) {
$this->configuration['pages'] = $form_state['values']['pages'];
parent::submitConfigurationForm($form, $form_state);
}
/**
* {@inheritdoc}
*/
public function summary() {
$pages = array_map('trim', explode("\n", $this->configuration['pages']));
$pages = implode(', ', $pages);
if (!empty($this->configuration['negate'])) {
return $this->t('Do not return true on the following pages: @pages', array('@pages' => $pages));
}
return $this->t('Return true on the following pages: @pages', array('@pages' => $pages));
}
/**
* {@inheritdoc}
*/
public function evaluate() {
// Convert path to lowercase. This allows comparison of the same path
// with different case. Ex: /Page, /page, /PAGE.
$pages = Unicode::strtolower($this->configuration['pages']);
$request = $this->requestStack->getCurrentRequest();
// Compare the lowercase path alias (if any) and internal path.
$path = $request->attributes->get('_system_path');
$path_alias = Unicode::strtolower($this->aliasManager->getAliasByPath($path));
return $this->matchPath($path_alias, $pages) || (($path != $path_alias) && $this->matchPath($path, $pages));
}
/**
* Check if a path matches any pattern in a set of patterns.
*
* @param string $path
* The path to match.
* @param string $patterns
* String containing a set of patterns separated by \n, \r or \r\n.
*
* @return bool
* TRUE if the path matches a pattern, FALSE otherwise.
*/
protected function matchPath($path, $patterns) {
return drupal_match_path($path, $patterns);
}
}

View File

@ -0,0 +1,134 @@
<?php
/**
* @file
* Contains Drupal\system\Tests\Plugin\Core\Condition\RequestPathTest
*/
namespace Drupal\system\Tests\Plugin\Condition;
use Drupal\simpletest\KernelTestBase;
use Drupal\system\Tests\Routing\MockAliasManager;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\RequestStack;
/**
* Tests the Request Path condition plugin.
*/
class RequestPathTest extends KernelTestBase {
/**
* The condition plugin manager under test.
*
* @var \Drupal\Core\Condition\ConditionManager
*/
protected $pluginManager;
/**
* The path alias manager used for testing.
*
* @var \Drupal\system\Tests\Routing\MockAliasManager
*/
protected $aliasManager;
/**
* The request stack used for testing.
*
* @var \Symfony\Component\HttpFoundation\RequestStack
*/
protected $requestStack;
/**
* Modules to enable.
*
* @var array
*/
public static $modules = array('system', 'user', 'field', 'path');
/**
* {@inheritdoc}
*/
public static function getInfo() {
return array(
'name' => 'Request Path Condition Plugin',
'description' => 'Tests that the Request Path Condition, provided by the system module, is working properly.',
'group' => 'Condition API',
);
}
/**
* {@inheritdoc}
*/
protected function setUp() {
parent::setUp();
$this->installSchema('system', array('sequences', 'url_alias'));
$this->pluginManager = $this->container->get('plugin.manager.condition');
// Set a mock alias manager in the container.
$this->aliasManager = new MockAliasManager();
$this->container->set('path.alias_manager.cached', $this->aliasManager);
// Set the test request stack in the container.
$this->requestStack = new RequestStack();
$this->container->set('request_stack', $this->requestStack);
}
/**
* Tests the request path condition.
*/
public function testConditions() {
// Get the request path condition and test and configure it to check against
// different patterns and requests.
$pages = "my/pass/page\r\nmy/pass/page2\r\nfoo";
$request = Request::create('/my/pass/page2');
$request->attributes->set('_system_path', 'my/pass/page2');
$this->requestStack->push($request);
/* @var \Drupal\system\Plugin\Condition\RequestPath $condition */
$condition = $this->pluginManager->createInstance('request_path');
$condition->setConfig('pages', $pages);
$this->aliasManager->addAlias('my/pass/page2', 'my/pass/page2');
$this->assertTrue($condition->execute(), 'The request path matches a standard path');
$this->assertEqual($condition->summary(), 'Return true on the following pages: my/pass/page, my/pass/page2, foo', 'The condition summary matches for a standard path');
// Test an aliased path.
$request->attributes->set('_system_path', 'my/aliased/page');
$this->requestStack->pop();
$this->requestStack->push($request);
$this->aliasManager->addAlias('my/aliased/page', 'my/pass/page');
$this->assertTrue($condition->execute(), 'The request path matches an aliased path');
$this->assertEqual($condition->summary(), 'Return true on the following pages: my/pass/page, my/pass/page2, foo', 'The condition summary matches for an aliased path');
// Test a wildcard path.
$this->aliasManager->addAlias('my/pass/page3', 'my/pass/page3');
$request->attributes->set('_system_path', 'my/pass/page3');
$this->requestStack->pop();
$this->requestStack->push($request);
$condition->setConfig('pages', 'my/pass/*');
$this->assertTrue($condition->evaluate(), 'The system_path my/pass/page3 passes for wildcard paths.');
$this->assertEqual($condition->summary(), 'Return true on the following pages: my/pass/*', 'The condition summary matches for a wildcard path');
// Test a missing path.
$request->attributes->set('_system_path', 'my/fail/page4');
$this->requestStack->pop();
$this->requestStack->push($request);
$condition->setConfig('pages', 'my/pass/*');
$this->aliasManager->addAlias('my/fail/page4', 'my/fail/page4');
$this->assertFalse($condition->evaluate(), 'The system_path my/pass/page4 fails for a missing path.');
}
}