From 92537b39df36d31f8dd9ab0ed1e86d02be02fad9 Mon Sep 17 00:00:00 2001 From: dereine <dereine@99340.no-reply.drupal.org> Date: Tue, 2 Oct 2012 12:20:48 -0400 Subject: [PATCH] Issue #1785770 by dawehner: Improve the handler base test. --- lib/Drupal/views/Plugin/views/HandlerBase.php | 5 +- .../views/Tests/Handler/HandlerTest.php | 187 +++++++++++++++++- .../views.view.test_handler_relationships.yml | 30 +++ .../views.view.test_handler_test_access.yml | 47 +++++ .../Plugin/views/field/FieldTest.php | 10 + tests/views_test_data/views_test_data.module | 37 ++++ views_ui/admin.inc | 2 +- 7 files changed, 310 insertions(+), 8 deletions(-) create mode 100644 tests/views_test_config/config/views.view.test_handler_relationships.yml create mode 100644 tests/views_test_data/config/views.view.test_handler_test_access.yml diff --git a/lib/Drupal/views/Plugin/views/HandlerBase.php b/lib/Drupal/views/Plugin/views/HandlerBase.php index cc7419870b85..08280ac4fe3c 100644 --- a/lib/Drupal/views/Plugin/views/HandlerBase.php +++ b/lib/Drupal/views/Plugin/views/HandlerBase.php @@ -496,9 +496,12 @@ public function postExecute(&$values) { } /** * Provides a unique placeholders for handlers. + * + * @return string + * A placeholder which contains the table and the fieldname. */ protected function placeholder() { - return $this->query->placeholder($this->options['table'] . '_' . $this->options['field']); + return $this->query->placeholder($this->table . '_' . $this->field); } /** diff --git a/lib/Drupal/views/Tests/Handler/HandlerTest.php b/lib/Drupal/views/Tests/Handler/HandlerTest.php index 865aaded4bf7..ad2c30a60bd3 100644 --- a/lib/Drupal/views/Tests/Handler/HandlerTest.php +++ b/lib/Drupal/views/Tests/Handler/HandlerTest.php @@ -7,6 +7,7 @@ namespace Drupal\views\Tests\Handler; +use Drupal\views\ViewExecutable; use Drupal\views\Tests\ViewTestBase; use Drupal\views\Plugin\views\HandlerBase; @@ -20,13 +21,13 @@ class HandlerTest extends ViewTestBase { * * @var array */ - public static $modules = array('views_ui'); + public static $modules = array('views_ui', 'comment', 'node'); public static function getInfo() { return array( - 'name' => 'Handlers tests', + 'name' => 'Handler: Base', 'description' => 'Tests abstract handler definitions.', - 'group' => 'Views', + 'group' => 'Views Handlers', ); } @@ -36,6 +37,34 @@ protected function setUp() { $this->enableViewsTestModule(); } + /** + * Overrides Drupal\views\Tests\ViewTestBase::viewsData(). + */ + protected function viewsData() { + $data = parent::viewsData(); + // Override the name handler to be able to call placeholder() from outside. + $data['views_test_data']['name']['field']['id'] = 'test_field'; + + // Setup one field with an access callback and one with an access callback + // and arguments. + $data['views_test_data']['access_callback'] = $data['views_test_data']['id']; + $data['views_test_data']['access_callback_arguments'] = $data['views_test_data']['id']; + foreach (ViewExecutable::viewsHandlerTypes() as $type => $info) { + if (isset($data['views_test_data']['access_callback'][$type]['id'])) { + $data['views_test_data']['access_callback'][$type]['access callback'] = 'views_test_data_handler_test_access_callback'; + + $data['views_test_data']['access_callback_arguments'][$type]['access callback'] = 'views_test_data_handler_test_access_callback_argument'; + $data['views_test_data']['access_callback_arguments'][$type]['access arguments'] = array(TRUE); + } + } + + return $data; + } + + /** + * @todo + * This should probably moved to a filter related test. + */ function testFilterInOperatorUi() { $admin_user = $this->drupalCreateUser(array('administer views', 'administer site configuration')); $this->drupalLogin($admin_user); @@ -109,7 +138,7 @@ function testBreakPhraseString() { } /** - * Tests views_break_phrase function. + * Tests Drupal\views\Plugin\views\HandlerBase::breakPhrase() function. */ function testBreakPhrase() { $empty_stdclass = new \stdClass(); @@ -145,11 +174,12 @@ function testBreakPhrase() { } /** - * Check to see if two values are equal. + * Check to see if a value is the same as the value on a certain handler. * * @param $first * The first value to check. - * @param views_handler $handler + * @param Drupal\views\Plugin\views\HandlerBase $handler + * The handler that has the $handler->value property to compare with first. * @param string $message * The message to display along with the assertion. * @param string $group @@ -162,4 +192,149 @@ protected function assertEqualValue($first, $handler, $message = '', $group = 'O return $this->assert($first == $handler->value, $message ? $message : t('First value is equal to second value'), $group); } + /** + * Tests the relationship ui for field/filter/argument/relationship. + */ + public function testRelationshipUI() { + $views_admin = $this->drupalCreateUser(array('administer views')); + $this->drupalLogin($views_admin); + + // Make sure the link to the field options exists. + $handler_options_path = 'admin/structure/views/nojs/config-item/test_handler_relationships/default/field/title'; + $view_edit_path = 'admin/structure/views/view/test_handler_relationships/edit'; + $this->drupalGet($view_edit_path); + $this->assertLinkByHref($handler_options_path); + + // The test view has a relationship to node_revision so the field should + // show a relationship selection. + + $this->drupalGet($handler_options_path); + $relationship_name = 'options[relationship]'; + $this->assertFieldByName($relationship_name); + + // Check for available options. + $xpath = $this->constructFieldXpath('name', $relationship_name); + $fields = $this->xpath($xpath); + $options = array(); + foreach ($fields as $field) { + $items = $this->getAllOptions($field); + foreach ($items as $item) { + $options[] = $item->attributes()->value; + } + } + $expected_options = array('none', 'nid'); + $this->assertEqual($options, $expected_options); + + // Remove the relationship and take sure no relationship option appears. + $this->drupalPost('admin/structure/views/nojs/config-item/test_handler_relationships/default/relationship/nid', array(), t('Remove')); + $this->drupalGet($handler_options_path); + $this->assertNoFieldByName($relationship_name, 'Make sure that no relationship option is available'); + } + + /** + * Tests the relationship method on the base class. + */ + public function testSetRelationship() { + $view = $this->createViewFromConfig('test_handler_relationships'); + // Setup a broken relationship. + $view->addItem('default', 'relationship', $this->randomName(), $this->randomName(), array(), 'broken_relationship'); + // Setup a valid relationship. + $view->addItem('default', 'relationship', 'comment', 'nid', array('relationship' => 'cid'), 'valid_relationship'); + $view->initHandlers(); + $field = $view->field['title']; + + $field->options['relationship'] = NULL; + $field->setRelationship(); + $this->assertFalse($field->relationship, 'Make sure that an empty relationship does not create a relationship on the field.'); + + $field->options['relationship'] = $this->randomName(); + $field->setRelationship(); + $this->assertFalse($field->relationship, 'Make sure that a random relationship does not create a relationship on the field.'); + + $field->options['relationship'] = 'broken_relationship'; + $field->setRelationship(); + $this->assertFalse($field->relationship, 'Make sure that a broken relationship does not create a relationship on the field.'); + + $field->options['relationship'] = 'valid_relationship'; + $field->setRelationship(); + $this->assertFalse(!empty($field->relationship), 'Make sure that the relationship alias was not set without building a views query before.'); + + // Remove the invalid relationship. + unset($view->relationship['broken_relationship']); + + $view->build(); + $field->setRelationship(); + $this->assertEqual($field->relationship, $view->relationship['valid_relationship']->alias, 'Make sure that a valid relationship does create the right relationship query alias.'); + } + + /** + * Tests the placeholder function. + * + * @see Drupal\views\Plugin\views\HandlerBase::placeholder() + */ + public function testPlaceholder() { + $view = $this->getView(); + $view->initHandlers(); + $view->initQuery(); + + $handler = $view->field['name']; + $table = $handler->table; + $field = $handler->field; + $string = ':' . $table . '_' . $field; + + // Make sure the placeholder variables are like expected. + $this->assertEqual($handler->getPlaceholder(), $string); + $this->assertEqual($handler->getPlaceholder(), $string . 1); + $this->assertEqual($handler->getPlaceholder(), $string . 2); + + // Set another table/field combination and make sure there are new + // placeholders. + $table = $handler->table = $this->randomName(); + $field = $handler->field = $this->randomName(); + $string = ':' . $table . '_' . $field; + + // Make sure the placeholder variables are like expected. + $this->assertEqual($handler->getPlaceholder(), $string); + $this->assertEqual($handler->getPlaceholder(), $string . 1); + $this->assertEqual($handler->getPlaceholder(), $string . 2); + } + + /** + * Tests access to a handler. + * + * @see views_test_data_handler_test_access_callback + */ + public function testAccess() { + $view = views_get_view('test_handler_test_access'); + $views_data = $this->viewsData(); + $views_data = $views_data['views_test_data']; + + // Enable access to callback only field and deny for callback + arguments. + config('views_test_data.tests')->set('handler_access_callback', TRUE)->save(); + config('views_test_data.tests')->set('handler_access_callback_argument', FALSE)->save(); + $view->initDisplay(); + $view->initHandlers(); + + foreach ($views_data['access_callback'] as $type => $info) { + if (!in_array($type, array('title', 'help'))) { + $this->assertTrue($view->field['access_callback'] instanceof HandlerBase, 'Make sure the user got access to the access_callback field '); + $this->assertFalse(isset($view->field['access_callback_arguments']), 'Make sure the user got no access to the access_callback_arguments field '); + } + } + + // Enable access to the callback + argument handlers and deny for callback. + config('views_test_data.tests')->set('handler_access_callback', FALSE)->save(); + config('views_test_data.tests')->set('handler_access_callback_argument', TRUE)->save(); + $view->destroy(); + $view->initDisplay(); + $view->initHandlers(); + + foreach ($views_data['access_callback'] as $type => $info) { + if (!in_array($type, array('title', 'help'))) { + $this->assertFalse(isset($view->field['access_callback']), 'Make sure the user got no access to the access_callback field '); + $this->assertTrue($view->field['access_callback_arguments'] instanceof HandlerBase, 'Make sure the user got access to the access_callback_arguments field '); + } + } + } + } diff --git a/tests/views_test_config/config/views.view.test_handler_relationships.yml b/tests/views_test_config/config/views.view.test_handler_relationships.yml new file mode 100644 index 000000000000..409fb2c0c376 --- /dev/null +++ b/tests/views_test_config/config/views.view.test_handler_relationships.yml @@ -0,0 +1,30 @@ +api_version: '3.0' +base_table: node +core: '8' +description: '' +disabled: '0' +display: + default: + display_options: + fields: + title: + id: title + table: node + field: title + relationships: + cid: + id: cid + table: node + field: cid + nid: + id: nid + table: comment + field: nid + relationship: cid + display_plugin: default + display_title: Master + id: default + position: '0' +human_name: '' +name: test_handler_relationships +tag: '' diff --git a/tests/views_test_data/config/views.view.test_handler_test_access.yml b/tests/views_test_data/config/views.view.test_handler_test_access.yml new file mode 100644 index 000000000000..f7cbce55c91b --- /dev/null +++ b/tests/views_test_data/config/views.view.test_handler_test_access.yml @@ -0,0 +1,47 @@ +api_version: '3.0' +base_table: views_test_data +core: 8 +display: + default: + display_options: + fields: + access_callback: + id: access_callback + table: views_test_data + field: access_callback + access_callback_arguments: + id: access_callback_arguments + table: views_test_data + field: access_callback_arguments + filters: + access_callback: + id: access_callback + table: views_test_data + field: access_callback + access_callback_arguments: + id: access_callback_arguments + table: views_test_data + field: access_callback_arguments + arguments: + access_callback: + id: access_callback + table: views_test_data + field: access_callback + access_callback_arguments: + id: access_callback_arguments + table: views_test_data + field: access_callback_arguments + sorts: + access_callback: + id: access_callback + table: views_test_data + field: access_callback + access_callback_arguments: + id: access_callback_arguments + table: views_test_data + field: access_callback_arguments + display_plugin: default + display_title: Master + id: default + position: '0' +name: test_handler_test_access diff --git a/tests/views_test_data/lib/Drupal/views_test_data/Plugin/views/field/FieldTest.php b/tests/views_test_data/lib/Drupal/views_test_data/Plugin/views/field/FieldTest.php index 434f80edb9f5..929d9a4100ec 100644 --- a/tests/views_test_data/lib/Drupal/views_test_data/Plugin/views/field/FieldTest.php +++ b/tests/views_test_data/lib/Drupal/views_test_data/Plugin/views/field/FieldTest.php @@ -61,4 +61,14 @@ function render($values) { return $this->sanitizeValue($this->getTestValue()); } + /** + * A mock function which allows to call placeholder from public. + * + * @return string + * The result of the placeholder method. + */ + public function getPlaceholder() { + return $this->placeholder(); + } + } diff --git a/tests/views_test_data/views_test_data.module b/tests/views_test_data/views_test_data.module index ef4b86f5a522..c4a28f954167 100644 --- a/tests/views_test_data/views_test_data.module +++ b/tests/views_test_data/views_test_data.module @@ -44,6 +44,43 @@ function views_test_data_test_dynamic_access_callback($access, $argument1, $argu return $access && $argument1 == variable_get('test_dynamic_access_argument1', NULL) && $argument2 == variable_get('test_dynamic_access_argument2', NULL); } +/**+ + * Access callback for the generic handler test. + * + * @return bool + * Returns views_test_data.tests->handler_access_callback config. so the user + * has access to the handler. + * + * @see Drupal\views\Tests\Handler\HandlerTest + */ +function views_test_data_handler_test_access_callback() { + return config('views_test_data.tests')->get('handler_access_callback'); +} + +/** + * Access callback with an argument for the generic handler test. + * + * @param bool $argument + * A parameter to test that an argument got passed. + * + * @return bool + * Returns views_test_data.tests->handler_access_callback_argument, so the + * use has access to the handler. + * + * @see Drupal\views\Tests\Handler\HandlerTest + */ +function views_test_data_handler_test_access_callback_argument($argument = FALSE) { + // Check the argument to be sure that access arguments are passed into the + // callback. + if ($argument) { + return config('views_test_data.tests')->get('handler_access_callback_argument'); + } + else { + return FALSE; + } +} + + /** * Implements hook_views_pre_render(). */ diff --git a/views_ui/admin.inc b/views_ui/admin.inc index 861ce7d7cf4d..0e6172fd72e1 100644 --- a/views_ui/admin.inc +++ b/views_ui/admin.inc @@ -1740,7 +1740,7 @@ function views_ui_config_item_form($form, &$form_state) { if (!empty($relationship_options)) { // Make sure the existing relationship is even valid. If not, force // it to none. - $base_fields = views_fetch_fields($view->base_table, $form_state['type'], $view->display_handler->useGroupBy()); + $base_fields = views_fetch_fields($view->storage->base_table, $form_state['type'], $view->display_handler->useGroupBy()); if (isset($base_fields[$item['table'] . '.' . $item['field']])) { $relationship_options = array_merge(array('none' => t('Do not use a relationship')), $relationship_options); } -- GitLab