gpt4 book ai didi

php - Symfony 2 中的可配置动态 Doctrine 数据库实体

转载 作者:可可西里 更新时间:2023-10-31 22:16:57 25 4
gpt4 key购买 nike

我正在努力实现的目标

  • 用户将能够通过网站上的 HTML 表单配置 Doctrine 实体。
  • 用户将能够定义新实体,以及为现有实体添加和删除字段。 (类似于 Drupal's content types )
  • Doctrine 实体将根据用户通过 Web UI 提供的配置获取动态属性。
  • 要么在实体配置发生变化时动态更改每个 Doctrine 实体的单个 DB 表;要么或者每个实体可以使用多个表(每个新实体字段都会有自己的表)。

到目前为止完成

过去几天我一直在研究这个,但没有取得太大成功,但我偶然发现了 this answer这似乎与我想要实现的目标非常相关。

我已经注册并添加了映射字段 foo 的 loadClassMetadata 监听器:

// src/DynamicMappingTest/AdminBundle/EventListener/MappingListener.php

namespace DynamicMappingTest\AdminBundle\EventListener;

use Doctrine\ORM\Event\LoadClassMetadataEventArgs;

class MappingListener
{
public function loadClassMetadata(LoadClassMetadataEventArgs $eventArgs)
{
$classMetadata = $eventArgs->getClassMetadata();
if ($classMetadata->getName() != 'DynamicMappingTest\\AdminBundle\\Entity\\CustomNode')
{
// Not the CustomNode test class. Do not alter the class metadata.
return;
}
$table = $classMetadata->table;

$oldName = $table['name']; // ... or $classMetaData->getTableName()

// your logic here ...

$table['name'] = 'custom_node';

$classMetadata->setPrimaryTable($table);

$reflClass = $classMetadata->getReflectionClass();
dump($reflClass);

// ... or add a field-mapping like this

$fieldMapping = array(
'fieldName' => 'foo',
'type' => 'string',
'length' => 255
);
$classMetadata->mapField($fieldMapping);
}
}

现在,只要我在 DynamicMappingTest\AdminBundle\Entity\CustomNode 类中声明了 foo 属性,这一切都有效:

// src/DynamicMappingTest/AdminBundle/Entity/CustomNode.php

namespace DynamicMappingTest\AdminBundle\Entity;

use Doctrine\ORM\Mapping as ORM;

/**
* CustomNode
*
* @ORM\Table()
* @ORM\Entity(repositoryClass="DynamicMappingTest\AdminBundle\Entity\CustomNodeRepository")
*/
class CustomNode
{
...
private $foo;
}

问题

但是,我无法知道用户将为他们的自定义实体定义哪些属性。如果我从 CustomNode 类中删除 foo 属性,我从 ClassMetadata 获得的 ReflectionClass 自然不会包含 foo 属性,所以我每当执行 MappingListener 中的 mapField() 时,都会出现以下异常:

ReflectionException: Property DynamicMappingTest\AdminBundle\Entity\CustomNode::$foo does not exist
in vendor/doctrine/common/lib/Doctrine/Common/Persistence/Mapping/RuntimeReflectionService.php at line 80

77.         */
78. public function getAccessibleProperty($class, $property)
79. {
80. $reflectionProperty = new ReflectionProperty($class, $property);
81.
82. if ($reflectionProperty->isPublic()) {
83. $reflectionProperty = new RuntimePublicReflectionProperty($class, $property);

问题

  • 是否有可能拥有完全可配置的动态 Doctrine 实体?
  • 我的方法是否正确?如果没有,你能推荐一个替代方案吗?
  • 我怎样才能拥有真正动态的类属性?或者我是否应该在用户更改实体配置时生成新的 Doctrine 实体 PHP 类?

最佳答案

是否有可能拥有完全可配置的动态 Doctrine 实体?

Doctrine 为您的实体生成代理类。这意味着 doctrine 生成带有类的 PHP 代码,它扩展您的 Entity 类并覆盖方法 - 放置一些自定义逻辑然后调用父方法。

因此,我认为真正实现这一目标的唯一方法是为代码中的实体生成 PHP 代码。也就是说,每次在您的网站中创建实体时,您都应该使用该实体生成 PHP 文件,然后运行迁移。

我的方法是否正确?如果没有,你能推荐一个替代方案吗?

我认为在这种情况下你根本不应该使用 Doctrine ORM,至少在你尝试这样做的方式上是这样。

通常,ORM 用于更容易/更易于管理的编程。也就是说,您可以设置关系、使用延迟加载、工作单元(更改实体属性然后刷新)等。如果您的实体是动态生成的,您将使用哪些功能?开发人员不会为这些实体编写代码,因为正如您所说,无法知道它将具有哪些字段。

您没有提供具体的用例 - 为什么您首先要这样做。但我想这真的可以用更简单的方式完成。

如果用户可以存储任何结构,你是否应该使用 MySQL?在这种情况下,ElasticSearch 或类似的解决方案可能会好得多。

我怎样才能拥有真正动态的类属性?或者我是否应该在用户更改实体配置时生成新的 Doctrine 实体 PHP 类?

正如我所提到的 - 是的。除非你想覆盖或替换一些 Doctrine 代码,但我想它可能有很多(代理类等)

关于php - Symfony 2 中的可配置动态 Doctrine 数据库实体,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35869096/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com