0

我的参加者表有一个jsonb名为eventfilters.

给定与 具有相同签名(表单)的查询过滤器数组eventfilters,我需要eventfilters通过查询过滤器值选择针对该字段过滤的与会者。


以下是查询过滤器和eventfilters字段的示例:

eventfilters字段如下所示:

[
    {
      "field": "Org Type",
      "selected": ["B2C", "Both", "Nonprofit"]
    },
    {
      "field": "Job Role",
      "selected": ["Customer Experience", "Digital Marketing"]
    },
    {
      "field": "Industry Sector",
      "selected": ["Advertising", "Construction / Development"]
    }
]

查询过滤器可能如下所示:

[
    {
      "field": "Org Type",
      "selected": ["B2C", "Nonprofit"]
    },
    {
      "field": "Industry Sector",
      "selected": ["Advertising"]
    }
]

因此,eventfilters字段和查询过滤器始终具有相同的签名:

Array<{"field": text, "selected": text[]}>

给定查询过滤器和eventfilters上面的过滤逻辑,过滤逻辑如下:

  • 选择具有以下eventfilters字段的所有与会者:
    • selected带有field: "Org Type"参加者 ( )的数组eventfilters包含数组中存在的任何值,该selected数组带有查询过滤器的“组织类型”字段;
    • selected带有field: "Industry Sector"参加者 ( )的数组eventfilters包含数组中存在的任何值,该selected数组带有查询过滤器的“行业部门”字段。

查询过滤器数组可以有不同的长度和不同的元素,但总是具有相同的签名(形式)。


我能想出的是上述逻辑,但不是and查询过滤器中的每个元素,而是or

select distinct attendee.id,
                attendee.email,
                attendee.eventfilters
from attendee cross join lateral jsonb_array_elements(attendee.eventfilters) single_filter
where (
    ((single_filter ->> 'field')::text = 'Org Type' and (single_filter ->> 'selected')::jsonb ?| array ['B2C', 'Nonprofit'])
    or ((single_filter ->> 'field')::text = 'Industry Sector' and (single_filter ->> 'selected')::jsonb ?| array ['Advertising'])
);

基本上我需要将上述查询or中的where子句更改为and,但这显然行不通。

where子句将动态生成。

这是我现在如何生成它的示例(它是 javascript,但我希望你能掌握这个想法):

  function buildEventFiltersWhereSql(eventFilters) {
    return eventFilters.map((filter) => {
      const selectedArray = filter.selected.map((s) => `'${s}'`).join(', ');
      return `((single_filter ->> 'field')::text = '${filter.field}' and (single_filter ->> 'selected')::jsonb ?| array[${selectedArray}])`;
    }).join('\nor ');
  }

orand逻辑的简单交换在实现中似乎非常不同。我想使用 来实现它会更容易jsonpath,但我的 postgres 版本是 11 :(

我怎样才能实现这样的过滤?


PS:create tableinsert复制代码:https ://pastebin.com/1tsHyJV0

4

1 回答 1

0

所以,我稍微摸索了一下,得到了以下查询:

with unnested_filters as (
    select distinct attendee.id,
                    jsonb_agg((single_filter ->> 'selected')::jsonb) filter (where (single_filter ->> 'field')::text = 'Org Type') over (partition by attendee.id) as "Org Type",
                    jsonb_agg((single_filter ->> 'selected')::jsonb) filter (where (single_filter ->> 'field')::text = 'Industry Sector') over (partition by attendee.id) as "Industry Sector",
                    attendee.email,
                    attendee.eventid,
                    attendee.eventfilters
    from attendee
             cross join lateral jsonb_array_elements(attendee.eventfilters) single_filter
    where eventid = 1
) select * from unnested_filters where (
    (unnested_filters."Org Type" #>> '{0}')::jsonb ?| array['B2C', 'Both', 'Nonprofit']
    and (unnested_filters."Industry Sector" #>> '{0}')::jsonb ?| array['Advertising']
);

这有点奇怪,尤其是带有 的部分jsonb_agg,但似乎有效。

于 2021-10-22T16:10:49.083 回答