Symfony 函数“LoginFormAuthenticator::\uu construct()”的参数太少,0正好传递了4个

Symfony 函数“LoginFormAuthenticator::\uu construct()”的参数太少,0正好传递了4个,symfony,service,orm,doctrine,authenticator,Symfony,Service,Orm,Doctrine,Authenticator,需要连接到多个数据库,并遵循有关此问题的文档 我创建了多个条令连接和orm实体管理器,并禁用了自动布线 # config/packages/doctrine.yaml doctrine: dbal: default_connection: default connections: default: # configure these for your database server url: "%env(resolve:D

需要连接到多个数据库,并遵循有关此问题的文档

我创建了多个条令连接和orm实体管理器,并禁用了自动布线

# config/packages/doctrine.yaml
doctrine:
  dbal:
    default_connection: default
    connections:
      default:
        # configure these for your database server
        url: "%env(resolve:DATABASE_URL)%"
        driver: "pdo_mysql"
        server_version: "5.7"
        charset: utf8mb4
      lc_cvo:
        # configure these for your database server
        url: "%env(resolve:DATABASE_LC_CVO_URL)%"
        driver: "pdo_mysql"
        server_version: "5.7"
        charset: utf8mb4
      lc_cvt:
        # configure these for your database server
        url: "%env(resolve:DATABASE_LC_CVT_URL)%"
        driver: "pdo_mysql"
        server_version: "5.7"
        charset: utf8mb4
      lc_ewi:
        # configure these for your database server
        url: "%env(resolve:DATABASE_LC_EWI_URL)%"
        driver: "pdo_mysql"
        server_version: "5.7"
        charset: utf8mb4
      lc_tbo:
        # configure these for your database server
        url: "%env(resolve:DATABASE_LC_TBO_URL)%"
        driver: "pdo_mysql"
        server_version: "5.7"
        charset: utf8mb4
      lc_users:
        # configure these for your database server
        url: "%env(resolve:DATABASE_LC_USERS_URL)%"
        driver: "pdo_mysql"
        server_version: "5.7"
        charset: utf8mb4

orm:
    entity_managers:
      default:
        connection: default
        mappings:
          Main:
            is_bundle: false
            type: annotation
            dir: "%kernel.project_dir%/src/Entity/lmc_lemoncake"
            prefix: 'App\Entity\lmc_lemoncake'
            alias: Main
      lc_cvo:
        connection: lc_cvo
        mappings:
          lc_cvo:
            is_bundle: false
            type: annotation
            dir: "%kernel.project_dir%/src/Entity/lmc_lemoncake-cvo"
            prefix: 'App\Entity\lmc_lemoncake_cvo'
            alias: lc_cvo
      lc_cvt:
        connection: lc_cvt
        mappings:
          lc_cvt:
            is_bundle: false
            type: annotation
            dir: "%kernel.project_dir%/src/Entity/lmc_lemoncake-cvt"
            prefix: 'App\Entity\lmc_lemoncake_cvt'
            alias: lc_cvt
      lc_ewi:
        connection: lc_ewi
        mappings:
          lc_ewi:
            is_bundle: false
            type: annotation
            dir: "%kernel.project_dir%/src/Entity/lmc_lemoncake-ewi"
            prefix: 'App\Entity\lmc_lemoncake_ewi'
            alias: lc_ewi
      lc_tbo:
        connection: lc_tbo
        mappings:
          lc_tbo:
            is_bundle: false
            type: annotation
            dir: "%kernel.project_dir%/src/Entity/lmc_lemoncake-tbo"
            prefix: 'App\Entity\lmc_lemoncake_tbo'
            alias: lc_tbo
      lc_users:
        connection: lc_users
        mappings:
          lc_users:
            is_bundle: false
            type: annotation
            dir: "%kernel.project_dir%/src/Entity/lmc_users"
            prefix: 'App\Entity\lmc_users'
            alias: lc_users
我的services.yaml文件如下所示

# This file is the entry point to configure your own services.
# Files in the packages/ subdirectory configure your dependencies.

# Put parameters here that don't need to change on each machine where the app is deployed
# https://symfony.com/doc/current/best_practices/configuration.html#application-related-configuration
parameters:

services:
  # default configuration for services in *this* file
  _defaults:
    autowire: false # Automatically injects dependencies in your services.
    autoconfigure: true # Automatically registers your services as commands, event subscribers, etc.

  # makes classes in src/ available to be used as services
  # this creates a service per class whose id is the fully-qualified class name
  App\:
    resource: "../src/"
    exclude:
      - "../src/DependencyInjection/"
      - "../src/Entity/"
      - "../src/Kernel.php"
      - "../src/Tests/"

  # controllers are imported separately to make sure services can be injected
  # as action arguments even if you don't extend any base controller class
  App\Controller\:
    resource: "../src/Controller/"
    tags: ["controller.service_arguments"]

  # add more service definitions when explicit configuration is needed
  # please note that last definitions always *replace* previous ones
不幸的是,我在尝试访问登录页面时收到以下错误

Too few arguments to function App\Security\LoginFormAuthenticator::__construct(), 0 passed in /var/www/html/app/var/cache/dev/Container12fc4el/getSecurity_Firewall_Map_Context_MainService.php on line 53 and exactly 4 expected
此处列出了它所指的LoginFormAuthenticator,它需要连接到用户信息(用户名,pw)所在的lc_用户连接。 我将需要其他连接来获取客户端数据

class LoginFormAuthenticator extends AbstractFormLoginAuthenticator implements PasswordAuthenticatedInterface
{
    use TargetPathTrait;

    public const LOGIN_ROUTE = 'app_login';

    private $entityManager;
    private $urlGenerator;
    private $csrfTokenManager;
    private $passwordEncoder;

    public function __construct(EntityManagerInterface $entityManager, UrlGeneratorInterface $urlGenerator, CsrfTokenManagerInterface $csrfTokenManager, UserPasswordEncoderInterface $passwordEncoder)
    {
        $this->entityManager = $entityManager;
        $this->urlGenerator = $urlGenerator;
        $this->csrfTokenManager = $csrfTokenManager;
        $this->passwordEncoder = $passwordEncoder;
    }
我相信我需要在我的服务中添加一些东西,以便身份验证器能够检索到正确的连接,不幸的是,我对此事的了解还不够

我需要为多个客户端使用多个数据库

  • 我如何解决手头的问题
  • 如何防止其他连接发生此问题
  • 我是以正确的方式处理这个问题,还是有更好的方法连接到多个数据库
提前感谢您的帮助,请随时询问更多信息

编辑: 感谢@msg的回答;我已通过以下代码使其正常工作:

app/config/services.yaml:

App\Security\LoginFormAuthenticator:
    autowire: true
    tags: ["doctrine.repository_service"]
    arguments:
      $entityManager: "@doctrine.orm.lc_users_entity_manager"
app/config/doctrine.yaml:

  orm:
    default_entity_manager: default
    entity_managers:
      auto_generate_proxy_classes: "%kernel.debug%"
      auto_mapping: true
      default:
         ...

      lc_users:
        connection: lc_users
        mappings:
          App\Entity\lmc_lemoncake:
            is_bundle: false
            type: annotation
            dir: "%kernel.project_dir%/src/Entity/lmc_users"
            prefix: 'App\Entity\lmc_users'
            alias: lc_users
(部分)LoginFormAuthenticator的getUser函数:

$em = $this->entityManager;
$repo = $em->getRepository(Users::class, 'lc_users');
$user = $repo->findOneBy(['username' => $credentials['username']]);

如前所述,注入
EntityManager
将获得
默认值。条令将为每个经理注册一个名为
条令.orm的服务。%manager\u name%\u entity\u manager

因此,如果您需要另一个,您有几个选项:

1。明确配置您的服务以使用所需的管理器:

services:
    App\Security\LoginFormAuthenticator: 
        arguments:
            # Override the Manager, other arguments will be autowired.
            $entityManager: '@doctrine.orm.lc_users_entity_manager'

2.为不同的实体经理创建不同的:

_defauts:
    bind:
        $cvoManager: '@doctrine.orm.lc_cvo_entity_manager'
        $usersManager: '@doctrine.orm.lc_users_entity_manager'
        # Other managers...

根据依赖项上的变量名,将注入适当的管理器:

public function __construct(
    EntityManagerInterface $entityManager, // Default manager
    EntityManagerInterface $usersManager,
)
3.如果您有许多依赖项,还可以插入
ManagerRegistry
(调用
getdoctor()
时在
AbstractController
中得到的对象)。它充当服务定位器,您可以:


如果您有许多具有不同依赖关系的服务,那么第一个选项可能会很快变得笨拙,而第三个选项将掩盖您的真实依赖关系,并且如果您需要模拟注册表,则会使测试复杂化。

尝试更改
autowire:true
如前所述,您需要将autowire设置为true以进行自动注入。但是,容器无法知道要注入哪个实体管理器。因此,您需要进一步绑定entity manager变量或手动定义LoginFormAuthenticator。我担心您将不得不花费一些高质量的时间阅读服务上的文档。注入
EntityManager
将为您提供默认的文档。如果您需要访问类中的多个连接,您可以插入
ManagerRegistry
并检索所需的服务,但请记住,它可能会带来自身的一系列问题。将autowire设置为true会调用错误的数据库。我会花一些时间阅读像@Cerad says这样的服务,然后在找到答案后将答案放在这里。我尝试从不同的资源中学习,但他们没有让我更聪明。如果有人能举一个我可以学习的例子,我将不胜感激。每当我尝试来自不同来源的解决方案时,我都会遇到不同错误。非常感谢@msg,我已成功登录,使用您提供的第一种方法。
    use Doctrine\Persistence\ManagerRegistry;

    public function __construct(ManagerRegistry $registry) {
        $userManager = $registry->getManager('users');
    }