Как настроить и внедрить несколько подключений к базе данных PDO в slim 4?

Я мог бы создать экземпляр PDO и успешно внедрить его. Я определил PDO::class напрямую и ввел его в конструктор с помощью __construct(PDO $pdo). Мне нужно что-то вроде PDO1::class и PDO2::class, чтобы ввести его следующим образом: __construct(PDO1 $pdo1, PDO2 $pdo2) но это, очевидно, не работает. Существует только один класс PDO, и мне нужно сделать 2 его экземпляра с разными учетными данными базы данных.
Как лучше всего это сделать?

Я установил одно определение базы данных через PDO, и оно работает:

Файл: dependencies.php

use DI\ContainerBuilder;
use Psr\Container\ContainerInterface;

return function (ContainerBuilder $containerBuilder) {
    $containerBuilder->addDefinitions([
        PDO::class => function (ContainerInterface $c) {
            $dbSettings = $c->get('settings')['db1'];
            $dsn = 'mysql:host=' . $dbSettings['host'] . ';dbname=' . $dbSettings['dbname'];
            $options = [
                PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
                PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
                PDO::ATTR_EMULATE_PREPARES => false,
            ];
            return new PDO($dsn, $dbSettings['user'], $dbSettings['pass'], $options);
        },
    ]);
};

Файл: index.php

...
// Set up dependencies
$dependencies = require __DIR__ . '/../app/dependencies.php';
$dependencies($containerBuilder);
// Build PHP-DI Container instance
$container = $containerBuilder->build();
// Set container to create App with on AppFactory
AppFactory::setContainer($container);
// Instantiate the app
$app = AppFactory::create();
...

Файл SomeRepository.php

use PDO;

class SomeRepository{

    protected $pdo;

    public function __construct(PDO $pdo) {
        $this->pdo = $pdo;
    }
}

Я видел нечто подобное в этой статье:

return function (ContainerBuilder $containerBuilder) {
    $containerBuilder->addDefinitions([
        'db1' => function (ContainerInterface $c) {
            $db1Settings = $c->get('settings')['db1'];
            $dsn = 'mysql:host=' . $db1Settings['host'] . ';dbname=' . $db1Settings['dbname'];
            $options = [ ... ];
            return new PDO($dsn, $db1Settings['user'], $db1Settings['pass'],$options);
        },
        'db2' => function (ContainerInterface $c) {
            $db2Settings = $c->get('settings')['db2'];
            $dsn = 'mysql:host=' . $db2Settings['host'] . ';dbname=' . $db2Settings['dbname'];
            $options = [ ... ];
            return new PDO($dsn, $db2Settings['user'], $db2Settings['pass'],$options);
        },

    ]);
};

Но лучший ли это способ сделать это? И как я могу получить доступ к соединениям в классе репозитория без необходимости внедрения всего контейнера?


person Samuel Gfeller    schedule 02.09.2019    source источник
comment
Что не так с __construct(PDO $pdo1, PDO $pdo2) помните, что PDO - это подсказка типа, так что намекните, что все параметры являются PDO   -  person RiggsFolly    schedule 02.09.2019


Ответы (2)


У вас есть несколько вариантов:

  1. Расширение PDO
  2. Автосвязывание объектов

<сильный>1. Расширение PDO

use PDO;

class PDO2 extends PDO
{
    // must be empty
}

Определение контейнера:

use PDO2;

// ...

return [
    PDO::class => function (ContainerInterface $container) {
        return new PDO(...);
    },

    PDO2::class => function (ContainerInterface $container) {
        return new PDO2(...);
    },
];

Использование

use PDO;
use PDO2;

class MyRepository
{
    private $pdo;

    private $pdo2;
    
    public function __construct(PDO $pdo, PDO2 $pdo2)
    {
        $this->pdo = $pdo;
        $this->pdo2 = $pdo2;
    }
}

<сильный>2. Автосвязывание объектов

См. ответ Матье Наполи: https://stackoverflow.com/a/57758106/1461181

person odan    schedule 03.09.2019

Если у вас есть несколько экземпляров класса в вашем приложении (здесь у вас есть несколько экземпляров класса PDO), то вы должны настроить, какой из них вводить каждый раз.

Это означает, что PDO нельзя подключить автоматически, потому что PHP-DI не может решить, какой экземпляр вам нужен, в зависимости от службы/контроллера/и т. д.

Вам необходимо использовать конфигурацию (см. http://php-di.org/doc/php-definitions.html#autowired-objects), чтобы указать, какой экземпляр (db1 или db2 в вашем примере) следует внедрять для каждой службы.

return [
    MyService::class => DI\autowire()
        ->constructorParameter('pdo', DI\get('db1'))
        ->constructorParameter('pdo2', DI\get('db2')),

    'db1' => function (ContainerInterface $c) {
        return new PDO();
    },
    'db2' => function (ContainerInterface $c) {
        return new PDO();
    },
];
person Matthieu Napoli    schedule 02.09.2019
comment
Спасибо за ваш ответ! Не могли бы вы добавить пример (определение и использование в службе), который соответствует моему коду, потому что я немного пытаюсь понять, что такое автопроводка и как я могу реализовать ее в своем примере? - person Samuel Gfeller; 02.09.2019
comment
@SamuelGfeller Я добавил пример. - person Matthieu Napoli; 03.09.2019