【问题标题】:Symfony 2 : multiple and dynamic database connectionSymfony 2:多动态数据库连接
【发布时间】:2018-05-08 04:49:25
【问题描述】:

我对 SF2 很陌生,我想知道如何将多个数据库的连接管理到一个包中。 目前我有这个解决方案 - 效果很好 - 但我不知道这是否是正确的方法......

在 myBundle\Ressource\config\config.yml 中:

doctrine:
dbal:
    default_connection:       default
    connections:
        default:
            dbname:           SERVER
            user:             root
            password:         null
            host:             localhost
        client:
            dbname:           CLIENT_134
            user:             root
            password:         null
            host:             localhost
orm:
    default_entity_manager:   default
    entity_managers:
        default:
            connection:       default
            mappings:
                MyBundle: ~
        client:
            connection:       client
            mappings:
                MyBundle: ~

然后,为了切换到一个 BD 或另一个,我这样做了:

$O_ressource=  $this->get('doctrine')->getEntityManager('client');
$O_ressource=  $this->get('doctrine')->getEntityManager('default');

各位小伙伴们,你们认为这是管理这个问题的好方法吗?

我的第二个问题是:

如何建立动态数据库连接? 我的意思是我的系统中有 100 个数据库,我无法在我的 config.yml 文件中设置所有这些数据库。 所以我希望能够即时更改数据库。

感谢您的帮助!

【问题讨论】:

  • “动态数据库连接”是指从控制器创建 DBAL 连接吗?
  • 是的,没错!能够从一个数据库切换到另一个,这些数据库可能不会在config.yml文件中声明
  • getEntityManager 在最新版本的 symfony2 中已被弃用

标签: symfony


【解决方案1】:

如果您使用ConnectionFactory,则附加到连接的事件订阅者将停止工作,例如 stofDoctrineExtensions。

这是我的方法。我和ConnectionFactory 一样有空连接和EntityManager。在工作时,我只是用反射替换连接配置。适用于 SF 2.0.10 ;)

class YourService extends ContainerAware
{ 

  public function switchDatabase($dbName, $dbUser, $dbPass) 
  {
    $connection = $this->container->get(sprintf('doctrine.dbal.%s_connection', 'dynamic_conn'));
    $connection->close();

    $refConn = new \ReflectionObject($connection);
    $refParams = $refConn->getProperty('_params');
    $refParams->setAccessible('public'); //we have to change it for a moment

    $params = $refParams->getValue($connection);
    $params['dbname'] = $dbName;
    $params['user'] = $dbUser;
    $params['password'] = $dbPass;

    $refParams->setAccessible('private');
    $refParams->setValue($connection, $params);
    $this->container->get('doctrine')->resetEntityManager('dynamic_manager'); // for sure (unless you like broken transactions)
  }
}

更新

为 php5.4 创建的教义 2.2 / sf 2.3(无重新选择)更优雅的解决方案(我喜欢新的数组初始化器:D) 我们可以使用称为连接包装器的学说特性,请参阅http://docs.doctrine-project.org/projects/doctrine-dbal/en/latest/reference/portability.html

此示例使用会话服务来临时存储连接详细信息。

首先我们必须创建特殊的连接包装器:

namespace w3des\DoctrineBundle\Connection;

use Doctrine\DBAL\Connection;
use Symfony\Component\HttpFoundation\Session\Session;
use Doctrine\Common\EventManager;
use Doctrine\DBAL\Events;
use Doctrine\DBAL\Event\ConnectionEventArgs;

/*
 * @author Dawid zulus Pakula [zulus@w3des.net]
 */
class ConnectionWrapper extends Connection
{

const SESSION_ACTIVE_DYNAMIC_CONN = 'active_dynamic_conn';

/**
 * @var Session
 */
private $session;

/**
 * @var bool
 */
private $_isConnected = false;

/**
 * @param Session $sess
 */
public function setSession(Session $sess)
{
    $this->session = $sess;
}

public function forceSwitch($dbName, $dbUser, $dbPassword)
{
    if ($this->session->has(self::SESSION_ACTIVE_DYNAMIC_CONN)) {
        $current = $this->session->get(self::SESSION_ACTIVE_DYNAMIC_CONN);
        if ($current[0] === $dbName) {
            return;
        }
    }

    $this->session->set(self::SESSION_ACTIVE_DYNAMIC_CONN, [
        $dbName,
        $dbUser,
        $dbPass
    ]);

    if ($this->isConnected()) {
        $this->close();
    }
}

/**
 * {@inheritDoc}
 */
public function connect()
{
    if (! $this->session->has(self::SESSION_ACTIVE_DYNAMIC_CONN)) {
        throw new \InvalidArgumentException('You have to inject into valid context first');
    }
    if ($this->isConnected()) {
        return true;
    }

    $driverOptions = isset($params['driverOptions']) ? $params['driverOptions'] : array();

    $params = $this->getParams();
    $realParams = $this->session->get(self::SESSION_ACTIVE_DYNAMIC_CONN);
    $params['dbname'] = $realParams[0];
    $params['user'] = $realParams[1];
    $params['password'] = $realParams[2];

    $this->_conn = $this->_driver->connect($params, $params['user'], $params['password'], $driverOptions);

    if ($this->_eventManager->hasListeners(Events::postConnect)) {
        $eventArgs = new ConnectionEventArgs($this);
        $this->_eventManager->dispatchEvent(Events::postConnect, $eventArgs);
    }

    $this->_isConnected = true;

    return true;
}

/**
 * {@inheritDoc}
 */
public function isConnected()
{
    return $this->_isConnected;
}

/**
 * {@inheritDoc}
 */
public function close()
{
    if ($this->isConnected()) {
        parent::close();
        $this->_isConnected = false;
    }
}
}

接下来在你的学说配置中注册它:

…

connections:
  dynamic:
    driver:   %database_driver%
    host:     %database_host%
    port:     %database_port%
    dbname:   'empty_database'
    charset:  UTF8
    wrapper_class: 'w3des\DoctrineBundle\Connection\ConnectionWrapper'

我们的 ConnectionWrapper 已正确注册。现在会话注入。

首先创建特殊的 CompilerPass 类:

namespace w3des\DoctrineBundle\DependencyInjection\CompilerPass;

use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\Definition;
use Symfony\Component\DependencyInjection\Reference;

class ConnectionCompilerPass implements CompilerPassInterface
{

/**
 * {@inheritDoc}
 */
public function process(ContainerBuilder $container)
{
    $connection = $container
    ->getDefinition('doctrine.dbal.dynamic_connection')
    ->addMethodCall('setSession', [
        new Reference('session')
    ]);
}
}

我们在 *Bundle 类中记录我们新的编译器类:

public function build(ContainerBuilder $container)
{
    parent::build($container);
    $container->addCompilerPass(new ConnectionCompilerPass());
}

仅此而已!

将根据会话属性按需创建连接。

要切换数据库,只需使用:

$this->get('doctrine.dbal.dynamic_connection')->forceSwitch($dbname, $dbuser, $dbpass);

优势

  1. 不再反思
  2. 按需创作
  3. 优雅而强大

缺点

  1. 您必须手动清理您的实体管理器,或为此创建特殊的学说事件
  2. 更多代码

【讨论】:

  • dynamic_conn 或 dynamic_manager 的用途是什么?你能详细说明他们的目的是什么吗?谢谢
  • 我有应用程序,每个客户端都有自己的数据库,其架构与其他客户端相同。此方法允许我通过 for ex 切换到真实数据库。会话权限
  • 感谢您的回复。在knpuniversity.com/screencast/question-answer-day/… 的帮助下,我能够根据给定的子域选择数据库
  • 好的,但是 ParamConverter 呢?我也可以在注释中使用它吗? /** * @Route("/page/{id}.{_format}", defaults={"_format"="json"}) * @ParamConverter("page", class="BlogBundle:Rest\Page", options={"entity_manager" = "dynamic"}) */ public function showAction(\BlogBundle\Entity\Rest\Page $page) { //... } 我也可以这样工作吗?正确的实体管理器应该通过注解注入到 ParamConverter...
  • 您必须通过 CompilerPass 执行此操作。直接(richardmiller.co.uk/2012/02/15/…)或通过将整个容器(就像我对会话服务所做的那样)注入到您的包装器中。 ID:service_container。
【解决方案2】:

你可以查看Symfony\Bundle\DoctrineBundle\ConnectionFactory,使用容器服务doctrine.dbal.connection_factory

$connectionFactory = $this->container->get('doctrine.dbal.connection_factory');
$connection = $connectionFactory->createConnection(array(
    'driver' => 'pdo_mysql',
    'user' => 'root',
    'password' => '',
    'host' => 'localhost',
    'dbname' => 'foo_database',
));

这只是一个简单的示例,但它应该可以帮助您入门。

【讨论】:

  • 谢谢你,它帮助了我!我想知道是否有办法更改在我的一个配置文件中定义的参数的值?例如:# app/config/config.yml 参数:my_mailer.class: Acme\HelloBundle\Mailer my_mailer.transport: sendmail services: my_mailer: class: %my_mailer.class% arguments: [%my_mailer.transport%] 我可以改变吗的值:%my_mailer.transport%
  • 我设法通过在我的 config.yml 文件中定义一个与假 dbal.connection 项目相关联的实体来“即时”创建连接,然后我使用了学说.dbal。 connection_factory 以创建正确的连接,最后我将这个新连接设置为 $this->container $this->container->set('doctrine.dbal.custom_client_connection', $connection); $myObject = $this->get('doctrine') ->getEntityManager('custom_client') ->getRepository('FooBarBundle:MyObject) ->find($id); 我不知道这是否是正确的方式/最好的方式,但它可以工作
  • 在Symfony 2.1中好像不能直接访问容器属性,需要使用helper方法$this->getContainer()->get('...')
  • 关于如何让多数据库工厂运转的很棒的简短示例,+1 先生。
【解决方案3】:

我遇到了同样的问题,需要为每个客户端使用具有相同架构的不同数据库。从 symfony 2.3 开始,在方法 resetEntityManager 被弃用后,我注意到代码运行良好,无需关闭连接,也无需重置(旧实体)管理器。

这是我当前的工作代码:

public function switchDatabase($dbName, $dbUser, $dbPass) {
    $connection = $this->container->get(sprintf('doctrine.dbal.%s_connection', 'dynamic_conn'));

    $refConn = new \ReflectionObject($connection);
    $refParams = $refConn->getProperty('_params');
    $refParams->setAccessible('public'); //we have to change it for a moment

    $params = $refParams->getValue($connection);
    $params['dbname'] = $dbName;
    $params['user'] = $dbUser;
    $params['password'] = $dbPass;

    $refParams->setAccessible('private');
    $refParams->setValue($connection, $params);
}

【讨论】:

  • 我正在构建一个仪表板系统,它将不同的数据库集成到一个中,从而提供一个界面来使用 Symfony3.0 管理这些数据库。问题是学说使用parameters.yml 文件来声明连接,我还没有看到有人谈论如何从这个角度实现与该线程中包含的相同的事情。文档也没有说清楚,因为我已经尝试过它们,我一直在绊倒错误。有人可以为 Symfony3.0 提供解决方案吗...谢谢。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2018-04-29
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多