From 9df9e020e734fc4e3a31c4177aa7bc36e94704fe Mon Sep 17 00:00:00 2001 From: webchick <drupal@webchick.net> Date: Tue, 6 Jan 2015 11:16:02 -0800 Subject: [PATCH] Issue #2379419 by Mile23, DuaelFr: Expand unit testing for Drupal\Component\Plugin\Discovery\StaticDiscoveryDecorator --- .../StaticDiscoveryDecoratorTest.php | 234 ++++++++++++++++++ 1 file changed, 234 insertions(+) create mode 100644 core/tests/Drupal/Tests/Component/Plugin/Discovery/StaticDiscoveryDecoratorTest.php diff --git a/core/tests/Drupal/Tests/Component/Plugin/Discovery/StaticDiscoveryDecoratorTest.php b/core/tests/Drupal/Tests/Component/Plugin/Discovery/StaticDiscoveryDecoratorTest.php new file mode 100644 index 000000000000..8c472c31c752 --- /dev/null +++ b/core/tests/Drupal/Tests/Component/Plugin/Discovery/StaticDiscoveryDecoratorTest.php @@ -0,0 +1,234 @@ +<?php + +/** + * @file + * Contains \Drupal\Tests\Component\Plugin\Discovery\DiscoveryTraitTest. + */ + +namespace Drupal\Tests\Component\Plugin\Discovery; + +use Drupal\Tests\UnitTestCase; + +/** + * @group Plugin + * @coversDefaultClass Drupal\Component\Plugin\Discovery\StaticDiscoveryDecorator + */ +class StaticDiscoveryDecoratorTest extends UnitTestCase { + + /** + * Helper method to provide a mocked callback object with expectations. + * + * If there should be a registered definition, then we have to place a + * \Callable in the mock object. The return value of this callback is + * never used. + * + * @return mock + * Mocked object with expectation of registerDefinitionsCallback() being + * called once. + */ + public function getRegisterDefinitionsCallback() { + $mock_callable = $this->getMockBuilder('\stdClass') + ->setMethods(array('registerDefinitionsCallback')) + ->getMock(); + // Set expectations for the callback method. + $mock_callable->expects($this->once()) + ->method('registerDefinitionsCallback'); + return $mock_callable; + } + + /** + * Data provider for testGetDefinitions(). + * + * @return array + * - Expected plugin definition. + * - Whether we require the method to register definitions through a + * callback. + * - Whether to throw an exception if the definition is invalid. + * - A plugin definition. + * - Base plugin ID. + */ + public function providerGetDefinition() { + return [ + ['is_defined', TRUE, FALSE, ['plugin-definition' => 'is_defined'], 'plugin-definition'], + // Make sure we don't call the decorated method if we shouldn't. + ['is_defined', FALSE, FALSE, ['plugin-definition' => 'is_defined'], 'plugin-definition'], + // Return NULL for bad plugin id. + [NULL, FALSE, FALSE, ['plugin-definition' => 'is_defined'], 'BAD-plugin-definition'], + // Generate an exception. + [NULL, FALSE, TRUE, ['plugin-definition' => 'is_defined'], 'BAD-plugin-definition'], + ]; + } + + /** + * @covers ::getDefinition + * @dataProvider providerGetDefinition + */ + public function testGetDefinition($expected, $has_register_definitions, $exception_on_invalid, $definitions, $base_plugin_id) { + // Mock our StaticDiscoveryDecorator. + $mock_decorator = $this->getMockBuilder('Drupal\Component\Plugin\Discovery\StaticDiscoveryDecorator') + ->disableOriginalConstructor() + ->setMethods(array('registeredDefintionCallback')) + ->getMock(); + + // Set up the ::$registerDefinitions property. + $ref_register_definitions = new \ReflectionProperty($mock_decorator, 'registerDefinitions'); + $ref_register_definitions->setAccessible(TRUE); + if ($has_register_definitions) { + // Set the callback object on the mocked decorator. + $ref_register_definitions->setValue( + $mock_decorator, + array($this->getRegisterDefinitionsCallback(), 'registerDefinitionsCallback') + ); + } + else { + // There should be no registerDefinitions callback. + $ref_register_definitions->setValue($mock_decorator, NULL); + } + + // Set up ::$definitions to an empty array. + $ref_definitions = new \ReflectionProperty($mock_decorator, 'definitions'); + $ref_definitions->setAccessible(TRUE); + $ref_definitions->setValue($mock_decorator, array()); + + // Mock a decorated object. + $mock_decorated = $this->getMockBuilder('Drupal\Component\Plugin\Discovery\DiscoveryInterface') + ->setMethods(array('getDefinitions')) + ->getMockForAbstractClass(); + // Return our definitions from getDefinitions(). + $mock_decorated->expects($this->once()) + ->method('getDefinitions') + ->willReturn($definitions); + + // Set up ::$decorated to our mocked decorated object. + $ref_decorated = new \ReflectionProperty($mock_decorator, 'decorated'); + $ref_decorated->setAccessible(TRUE); + $ref_decorated->setValue($mock_decorator, $mock_decorated); + + if ($exception_on_invalid) { + $this->setExpectedException('Drupal\Component\Plugin\Exception\PluginNotFoundException'); + } + + // Exercise getDefinition(). It calls parent::getDefinition(). + $this->assertEquals( + $expected, + $mock_decorator->getDefinition($base_plugin_id, $exception_on_invalid) + ); + } + + /** + * Data provider for testGetDefinitions(). + * + * @return array + * - bool Whether the test mock has a callback. + * - array Plugin definitions. + */ + public function providerGetDefinitions() { + return [ + [TRUE, ['definition' => 'is_fake']], + [FALSE, ['definition' => 'array_of_stuff']], + ]; + } + + /** + * @covers ::getDefinitions + * @dataProvider providerGetDefinitions + */ + public function testGetDefinitions($has_register_definitions, $definitions) { + // Mock our StaticDiscoveryDecorator. + $mock_decorator = $this->getMockBuilder('Drupal\Component\Plugin\Discovery\StaticDiscoveryDecorator') + ->disableOriginalConstructor() + ->setMethods(array('registeredDefintionCallback')) + ->getMock(); + + // Set up the ::$registerDefinitions property. + $ref_register_definitions = new \ReflectionProperty($mock_decorator, 'registerDefinitions'); + $ref_register_definitions->setAccessible(TRUE); + if ($has_register_definitions) { + // Set the callback object on the mocked decorator. + $ref_register_definitions->setValue( + $mock_decorator, + array($this->getRegisterDefinitionsCallback(), 'registerDefinitionsCallback') + ); + } + else { + // There should be no registerDefinitions callback. + $ref_register_definitions->setValue($mock_decorator, NULL); + } + + // Set up ::$definitions to an empty array. + $ref_definitions = new \ReflectionProperty($mock_decorator, 'definitions'); + $ref_definitions->setAccessible(TRUE); + $ref_definitions->setValue($mock_decorator, array()); + + // Mock a decorated object. + $mock_decorated = $this->getMockBuilder('Drupal\Component\Plugin\Discovery\DiscoveryInterface') + ->setMethods(array('getDefinitions')) + ->getMockForAbstractClass(); + // Our mocked method will return any arguments sent to it. + $mock_decorated->expects($this->once()) + ->method('getDefinitions') + ->willReturn($definitions); + + // Set up ::$decorated to our mocked decorated object. + $ref_decorated = new \ReflectionProperty($mock_decorator, 'decorated'); + $ref_decorated->setAccessible(TRUE); + $ref_decorated->setValue($mock_decorator, $mock_decorated); + + // Exercise getDefinitions(). It calls parent::getDefinitions() but in this + // case there will be no side-effects. + $this->assertArrayEquals( + $definitions, + $mock_decorator->getDefinitions() + ); + } + + /** + * Data provider for testCall(). + * + * @return array + * - Method name. + * - Array of arguments to pass to the method, with the expectation that our + * mocked __call() will return them. + */ + public function providerCall() { + return [ + ['complexArguments', ['1', 2.0, 3, ['4' => 'five']]], + ['noArguments', []], + ]; + } + + /** + * @covers ::__call + * @dataProvider providerCall + */ + public function testCall($method, $args) { + // Mock a decorated object. + $mock_decorated = $this->getMockBuilder('Drupal\Component\Plugin\Discovery\DiscoveryInterface') + ->setMethods(array($method)) + ->getMockForAbstractClass(); + // Our mocked method will return any arguments sent to it. + $mock_decorated->expects($this->once()) + ->method($method) + ->willReturnCallback( + function () { + return \func_get_args(); + } + ); + + // Create a mock decorator. + $mock_decorator = $this->getMockBuilder('Drupal\Component\Plugin\Discovery\StaticDiscoveryDecorator') + ->disableOriginalConstructor() + ->getMock(); + // Poke the decorated object into our decorator. + $ref_decorated = new \ReflectionProperty($mock_decorator, 'decorated'); + $ref_decorated->setAccessible(TRUE); + $ref_decorated->setValue($mock_decorator, $mock_decorated); + + // Exercise __call. + $this->assertArrayEquals( + $args, + \call_user_func_array(array($mock_decorated, $method), $args) + ); + } + +} -- GitLab