0

我有以下数据架构

-- table
master 
 - id (int)
 - meta (jsonb)
 - key (string)

-- table
 details 
  - master_id (int)
  - timestamp (timestamp)
  - value     (float)

(简化)

以下查询因以下错误而终止:

select json_build_object(
    'name',m.meta->>'name',
    'keys',json_agg(
        json_build_object(
        'key',m.key,
        'checkpoints',json_agg(
            json_build_object(
            'time',d.timestamp,
            'value',d.value)
        )
        )
    )
)
from master m, details d 
GROUP by m.meta->>'name',m.key,d.timestamp,d.value


ERROR:  aggregate function calls cannot be nested
LINE 6:   'checkpoints',json_agg(

我想得到以下结构。

[
    {"name":"name",
     "keys":[
        {"key":"mykey",
            "checkpoints":[
                {"time":"timestamp","value":"1.2"},...]
        },...]
    }...
]

我知道我不能使用json_agg嵌套的错误是可以理解的。但我不知道如何解决它。

此答案建议使用内联查询:PostgreSQL 聚合函数调用不能嵌套使用 jsonb_agg 函数

但这对我也不起作用,或者我错过了一些东西。

查询帮助将不胜感激,

编辑// 提供了一个带有示例数据的数据库小提琴和一个工作结果,其中不包括每个传感器的检查点。

https://www.db-fiddle.com/f/2XaUFeTcTfFFUrPYsVnYkt/0

4

1 回答 1

0

在进入查询之前,请注意我更改了连接条件。我认为加入detailsid/master_id 是准确的,因为您在 dbfiddle 中的原始查询似乎没有返回正确的结果;{"name":"x32Z72","keys":[{"key":"sensor1"},{"key":"sensor1"},{"key":"sensor1"},{"key":"sensor1"},{"key":"sensor1"},{"key":"sensor1"}]}当您只有两条detailsid 为 1的记录时,您真的想要吗?


这里有两个选项:

1:不要加入主查询,而是做一个子查询details来获取每个键的检查点。子查询独立存在,因此没有嵌套的 agg 调用问题。

SELECT json_build_object(
    'name', meta->>'name',
    'keys', json_agg(
        json_build_object(
            'key', key,
            'checkpoints', (
                SELECT json_agg(
                    json_build_object(
                        'time', d.timestamp,
                        'value', d.value
                    )
                )
                FROM details d
                WHERE master_id = m.id
                GROUP BY master_id
            )
        )
    )
)
FROM masters m
GROUP by m.meta->>'name', m.key

https://www.db-fiddle.com/f/2XaUFeTcTfFFUrPYsVnYkt/1

2:做join,但是做不同级别的aggs。所以首先在最低/最内层(检查点)上做一个 agg,然后你有一个外部选择来做更高级别的 agg(键):

SELECT json_build_object(
    'name', name,
    'keys', json_agg(
        json_build_object(
            'key', key,
            'checkpoints', checkpoints
        )
    )
)
FROM (
    SELECT
        m.meta->>'name' AS name,
        m.key,
        json_agg(
            json_build_object(
                'time', d.timestamp,
                'value', d.value
            )
        ) AS checkpoints
    FROM masters m
    INNER JOIN details d ON m.id = d.master_id
    GROUP by m.meta->>'name', m.key, d.master_id
) s
GROUP BY name, key

https://www.db-fiddle.com/f/2XaUFeTcTfFFUrPYsVnYkt/2

在任何一种情况下,结果都是相同的:

{"name":"x32Z72","keys":[{"key":"sensor1","checkpoints":[{"time":"2018-10-03T00:00:00+00:00","value":1},{"time":"2018-10-03T01:01:12+00:00","value":1.5},{"time":"2018-10-03T00:10:00+00:00","value":2.1}]}]}
{"name":"x32Z72","keys":[{"key":"sensor2","checkpoints":[{"time":"2018-10-03T02:12:01+00:00","value":0.8},{"time":"2018-10-03T06:10:00+00:00","value":1.8}]}]}
{"name":"x32Z72","keys":[{"key":"sensor3","checkpoints":[{"time":"2018-10-03T01:01:12+00:00","value":0.95}]}]}
于 2018-10-20T10:49:51.953 回答