Doctrine ODM在运行时选择数据库



我想在我的项目中使用条令ODM,我想为我的每个客户都有一个单独的数据库。我希望能够通过我的API在运行时管理客户端。我现在的问题是:

当我设置条令ODM时,我必须在parameters.yml中设置数据库设置,但我希望能够在运行时选择数据库。我将有一个主数据库,其中包含我所有的fixture集合和客户端索引,以知道要选择哪个数据库,但特定于客户端的内容将在这些客户端数据库中。每个Document类仍然会像在正常情况下一样链接到一个集合,但随后会链接到不同的数据库中。

有没有一种方法可以在运行时为Document类选择数据库?

假设我访问www.myproject.com/client1/item/list我将列出dbclient1.Items集合中的每个项目,如果我转到www.myproject.com/client2/item/list,我将列出dbclient2.Items集合的所有项目。

我希望我清楚地表明了我想在这里达到的目标。。。我找不到任何关于这件事的信息,但我认为如果我是第一个对此有疑问的人,那会很奇怪。。。在我之前肯定有人有同样的想法,对吧?

有没有办法在运行时为Document类选择数据库?

是的,每次需要更改数据库时都可以调用$dm->getConfiguration()->setDefaultDb('some_db_name'),但这可能会导致写入时出现意外行为。

根据我的经验(以及几年在多租户应用程序上的实践),最防弹的方法是为每个上下文创建单独的DocumentManager实例。您可以通过使用参数正常配置一个DocumentManager来实现这一点,但从未直接使用过-相反,您需要有一个类来管理其实例,fe:

use DoctrineODMMongoDBDocumentManager;
class DocumentManagerFactory
{
/**
* DocumentManager created by Symfony.
* 
* @var DocumentManager
*/
private $defaultDocumentManager;
/**
* All DocumentManagers created by Factory so far.
* 
* @var DocumentManager[]
*/
private $instances = array();
public function __construct(DocumentManager $dm) 
{
$this->defaultDocumentManager = $dm;
}
public function createFor(Context $ctx)
{
$databaseName = $ctx->getDatabaseName();
if (isset($this->instances[$databaseName])) {
return $this->instances[$databaseName];
}
$configuration = clone $this->defaultDocumentManager->getConfiguration();
$configuration->setDefaultDB($databaseName);
return $this->instances[$databaseName] = DocumentManager::create(
$this->defaultDocumentManager->getConnection(),
$configuration,
$this->defaultDocumentManager->getEventManager()
);
}
}

由于这种方法,来自少数数据库的文档永远不会由一个DocumentManager管理,您保留了一个负责干预ODM配置和框架方法的类(每个DocumentManager的事件订阅者等都相同)。

我还需要根据域名默认数据库名称(即seallot.mysite.com将使用seallote数据库)

我做了这个破解,虽然不是很好,但效果很好。我在DocumentManager构造中添加了这些行

$dbName = isset($_SERVER["HTTP_HOST"]) ? explode('.', $_SERVER["HTTP_HOST"])[0] : 'default_database_of_my_root_website';
$this->config->setDefaultDB($dbName);

相关内容

  • 没有找到相关文章

最新更新