postgresql 中的教义 JSON 列存储相同数据的对象或数组

问题描述 投票:0回答:1

我有一个 Symfony 应用程序,

User
[doctrine] 实体具有属性
roles
,它应该是一个数组或字符串。

    /**
     * @ORM\Column(type="json")
     */
    private $roles = [];

    public function getRoles(): array
    {
        $roles = $this->roles;
        // guarantee every user at least has ROLE_USER
        $roles[] = 'ROLE_USER';

        return array_unique($roles);
    }

    public function setRoles(array $roles): self
    {
        $this->roles = $roles;

        return $this;
    }

那是很久以前的事了,但我很确定这部分(上面的代码)是我根据

文档
使用php bin/console make:user得到的。正如你所看到的,它始终是一个数组。

我的应用程序已经达到了需要根据角色选择

Users
的程度,但我遇到的问题是数据库数据不一致。

["ROLE_TRANSLATOR","ROLE_DATA_ENTRY"]
["ROLE_TRANSLATOR","ROLE_DATA_ENTRY","ROLE_SITE_DIRECTOR"]
["ROLE_DATA_ENTRY"]
[]
{"0":"ROLE_SITE_DIRECTOR","2":"ROLE_TRANSLATOR","1":"ROLE_DATA_ENTRY"}
["ROLE_SPONSOR_REP"]
["ROLE_TRANSLATOR","ROLE_DATA_ENTRY"]
{"0":"ROLE_SPONSOR_REP","2":"ROLE_TRANSLATOR","1":"ROLE_DATA_ENTRY","3":"ROLE_SITE_DIRECTOR"}

因此,如果我正在寻找

ROLE_TRANSLATOR
,我必须签入数组或每个对象键

WHERE u.roles::jsonb ? 'ROLE_TRANSLATOR'
    OR u.roles::json ->> '0' = 'ROLE_TRANSLATOR'
    OR u.roles::json ->> '1' = 'ROLE_TRANSLATOR'
    OR u.roles::json ->> '2' = 'ROLE_TRANSLATOR'
    OR u.roles::json ->> '3' = 'ROLE_TRANSLATOR'

当我添加更多角色时,情况只会变得更糟。 目前只有4个。

我的问题;
为什么教义不一致?而且,我可以修复它吗? 或
如何制作更简单的 WHERE 子句?

就应用程序的其余部分而言,角色和安全性按预期工作。

php postgresql symfony doctrine
1个回答
2
投票

对 Slack 上的 @Dave Redfern 表示疯狂的支持,他指出了我的问题。 当传递非零索引数组时,它将被解释为对象。

dump(json_encode([
    0 => "ROLE_SITE_DIRECTOR", 2 => "ROLE_TRANSLATOR", 1 => "ROLE_DATA_ENTRY",
]));
dump(json_encode(array_values([
    0 => "ROLE_SITE_DIRECTOR", 2 => "ROLE_TRANSLATOR", 1 => "ROLE_DATA_ENTRY",
])));

输出将是:

"{"0":"ROLE_SITE_DIRECTOR","2":"ROLE_TRANSLATOR","1":"ROLE_DATA_ENTRY"}"
"["ROLE_SITE_DIRECTOR","ROLE_TRANSLATOR","ROLE_DATA_ENTRY"]"

所以修复很简单,我是我的设置者:

$this->roles = array_values($roles);

展望未来,使用角色的数据库关系也是一种更好的实践。 通过 JSON 数据搜索并不有趣,但在连接中搜索是标准做法。

© www.soinside.com 2019 - 2024. All rights reserved.