Skip to content
Snippets Groups Projects
Verified Commit ac7227ef authored by Lee Rowlands's avatar Lee Rowlands
Browse files

Issue #3394870 by longwave, smustgrave, alexpott: Allow controller service...

Issue #3394870 by longwave, smustgrave, alexpott: Allow controller service wiring via constructor parameter attributes

(cherry picked from commit a3edb28e)
parent d30fedcc
No related branches found
No related tags found
20 merge requests!8376Drupal views: adding more granularity to the ‘use ajax’ functionality,!8300Issue #3443586 View area displays even when parent view has no results.,!7567Issue #3153723 by quietone, Hardik_Patel_12: Change the scaffolding...,!7565Issue #3153723 by quietone, Hardik_Patel_12: Change the scaffolding...,!7509Change label "Block description" to "Block type",!7344Issue #3292350 by O'Briat, KlemenDEV, hswong3i, smustgrave, quietone: Update...,!6922Issue #3412959 by quietone, smustgrave, longwave: Fix 12 'un' words,!6848Issue #3417553 by longwave: Remove withConsecutive() in CacheCollectorTest,!6720Revert "Issue #3358581 by pfrenssen, _tarik_, a.dmitriiev, smustgrave:...,!6560Update ClaroPreRender.php, confirming classes provided are in array format,!6528Issue #3414261 by catch: Add authenticated user umami performance tests,!6501Issue #3263668 by omkar-pd, Wim Leers, hooroomoo: Re-enable inline form errors...,!6354Draft: Issue #3380392 by phma: Updating language weight from the overview reverts label if translated,!6324Issue #3416723 by Ludo.R: Provide a "node type" views default argument,!6119Issue #3405704 by Spokje, longwave: symfony/psr-http-message-bridge major version bump,!5950Issue #3403653 by alexpott, longwave: Incorporate improvements to how contrib runs PHPStan to core,!5858Issue #3401971 by fjgarlin: Test-only job shouldn't require constant rebases...,!5716Draft: Issue #3401102 by Spokje, longwave, smustgrave: Nightwatch artifacts on GitLab not retained,!5674Transaction autocommit during shutdown relies on unreliable object destruction order,!5644Issue #3395563 by nireneko, marvil07, lauriii, borisson_, smustgrave, Wim...
Checking pipeline status
......@@ -2,12 +2,12 @@
namespace Drupal\Core\Controller;
use Drupal\Core\DependencyInjection\AutowireTrait;
use Drupal\Core\DependencyInjection\ContainerInjectionInterface;
use Drupal\Core\Logger\LoggerChannelTrait;
use Drupal\Core\Routing\RedirectDestinationTrait;
use Drupal\Core\StringTranslation\StringTranslationTrait;
use Drupal\Core\Url;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Drupal\Core\Messenger\MessengerTrait;
use Symfony\Component\HttpFoundation\RedirectResponse;
......@@ -34,6 +34,7 @@
*/
abstract class ControllerBase implements ContainerInjectionInterface {
use AutowireTrait;
use LoggerChannelTrait;
use MessengerTrait;
use RedirectDestinationTrait;
......@@ -102,13 +103,6 @@ abstract class ControllerBase implements ContainerInjectionInterface {
*/
protected $formBuilder;
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container) {
return new static();
}
/**
* Retrieves the entity type manager.
*
......
<?php
namespace Drupal\Core\DependencyInjection;
use Symfony\Component\DependencyInjection\Attribute\Autowire;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\DependencyInjection\Exception\AutowiringFailedException;
/**
* Defines a trait for automatically wiring dependencies from the container.
*
* This trait uses reflection and may cause performance issues with classes
* that will be instantiated multiple times.
*/
trait AutowireTrait {
/**
* Instantiates a new instance of the implementing class using autowiring.
*
* @param \Symfony\Component\DependencyInjection\ContainerInterface $container
* The service container this instance should use.
*
* @return static
*/
public static function create(ContainerInterface $container) {
$args = [];
if (method_exists(static::class, '__construct')) {
$constructor = new \ReflectionMethod(static::class, '__construct');
foreach ($constructor->getParameters() as $parameter) {
$service = (string) $parameter->getType();
foreach ($parameter->getAttributes(Autowire::class) as $attribute) {
$service = (string) $attribute->newInstance()->value;
}
if (!$container->has($service)) {
throw new AutowiringFailedException($service, sprintf('Cannot autowire service "%s": argument "$%s" of method "%s::_construct()", you should configure its value explicitly.', $service, $parameter->getName(), static::class));
}
$args[] = $container->get($service);
}
}
return new static(...$args);
}
}
......@@ -13,7 +13,6 @@
use Drupal\node\NodeStorageInterface;
use Drupal\node\NodeTypeInterface;
use Drupal\node\NodeInterface;
use Symfony\Component\DependencyInjection\ContainerInterface;
/**
* Returns responses for Node routes.
......@@ -57,17 +56,6 @@ public function __construct(DateFormatterInterface $date_formatter, RendererInte
$this->entityRepository = $entity_repository;
}
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container) {
return new static(
$container->get('date.formatter'),
$container->get('renderer'),
$container->get('entity.repository')
);
}
/**
* Displays add content links for available content types.
*
......
<?php
namespace Drupal\system_test\Controller;
use Drupal\Core\Controller\ControllerBase;
use Drupal\Core\Lock\LockBackendInterface;
/**
* A controller that does not specify its autowired dependencies correctly.
*/
class BrokenSystemTestController extends ControllerBase {
/**
* Constructs the BrokenSystemTestController.
*
* @param \Drupal\Core\Lock\LockBackendInterface $lock
* The lock service.
*/
public function __construct(
protected LockBackendInterface $lock,
) {}
}
......@@ -11,11 +11,11 @@
use Drupal\Core\Render\Markup;
use Drupal\Core\Session\AccountInterface;
use Drupal\Core\Url;
use Symfony\Component\DependencyInjection\Attribute\Autowire;
use Symfony\Component\HttpFoundation\RedirectResponse;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;
use Drupal\Core\Lock\LockBackendInterface;
use Symfony\Component\DependencyInjection\ContainerInterface;
/**
* Controller routines for system_test routes.
......@@ -71,7 +71,15 @@ class SystemTestController extends ControllerBase implements TrustedCallbackInte
* @param \Drupal\Core\Messenger\MessengerInterface $messenger
* The messenger service.
*/
public function __construct(LockBackendInterface $lock, LockBackendInterface $persistent_lock, AccountInterface $current_user, RendererInterface $renderer, MessengerInterface $messenger) {
public function __construct(
#[Autowire(service: 'lock')]
LockBackendInterface $lock,
#[Autowire(service: 'lock.persistent')]
LockBackendInterface $persistent_lock,
AccountInterface $current_user,
RendererInterface $renderer,
MessengerInterface $messenger,
) {
$this->lock = $lock;
$this->persistentLock = $persistent_lock;
$this->currentUser = $current_user;
......@@ -79,19 +87,6 @@ public function __construct(LockBackendInterface $lock, LockBackendInterface $pe
$this->messenger = $messenger;
}
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container) {
return new static(
$container->get('lock'),
$container->get('lock.persistent'),
$container->get('current_user'),
$container->get('renderer'),
$container->get('messenger')
);
}
/**
* Tests main content fallback.
*
......
<?php
namespace Drupal\KernelTests\Core\Controller;
use Drupal\KernelTests\KernelTestBase;
use Drupal\system_test\Controller\BrokenSystemTestController;
use Drupal\system_test\Controller\SystemTestController;
use Symfony\Component\DependencyInjection\Exception\AutowiringFailedException;
/**
* Tests \Drupal\Core\Controller\ControllerBase.
*
* @coversDefaultClass \Drupal\Core\Controller\ControllerBase
* @group Controller
*/
class ControllerBaseTest extends KernelTestBase {
/**
* Modules to enable.
*
* @var array
*/
protected static $modules = ['system_test', 'system'];
/**
* @covers ::create
*/
public function testCreate() {
$controller = $this->container->get('class_resolver')->getInstanceFromDefinition(SystemTestController::class);
$property = new \ReflectionProperty(SystemTestController::class, 'lock');
$this->assertSame($this->container->get('lock'), $property->getValue($controller));
$property = new \ReflectionProperty(SystemTestController::class, 'persistentLock');
$this->assertSame($this->container->get('lock.persistent'), $property->getValue($controller));
$property = new \ReflectionProperty(SystemTestController::class, 'currentUser');
$this->assertSame($this->container->get('current_user'), $property->getValue($controller));
}
/**
* @covers ::create
*/
public function testCreateException() {
$this->expectException(AutowiringFailedException::class);
$this->expectExceptionMessage('Cannot autowire service "Drupal\Core\Lock\LockBackendInterface": argument "$lock" of method "Drupal\system_test\Controller\BrokenSystemTestController::_construct()", you should configure its value explicitly.');
$this->container->get('class_resolver')->getInstanceFromDefinition(BrokenSystemTestController::class);
}
}
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment