| From: | Thom Brown <thom(at)linux(dot)com> |
|---|---|
| To: | Richard Guo <guofenglinux(at)gmail(dot)com> |
| Cc: | pgsql-bugs <pgsql-bugs(at)postgresql(dot)org> |
| Subject: | Re: EXPLAIN (VERBOSE) fails with for JSON_ARRAYAGG/JSON_OBJECTAGG + window function |
| Date: | 2026-07-03 08:36:09 |
| Message-ID: | CAA-aLv6p3tMdY4KJ9wG_DxL1_3MTMCeKj8zUo+1-MctSBmvfdw@mail.gmail.com |
| Views: | Whole Thread | Raw Message | Download mbox | Resend email |
| Thread: | |
| Lists: | pgsql-bugs |
On Fri, 3 Jul 2026 at 04:06, Richard Guo <guofenglinux(at)gmail(dot)com> wrote:
>
> On Fri, Jul 3, 2026 at 7:10 AM Richard Guo <guofenglinux(at)gmail(dot)com> wrote:
> > Reproduced here. get_json_agg_constructor() expects that ctor->func
> > is Aggref or WindowFunc, but what it gets here is a Var.
> >
> > This is because the query has both window functions and grouped
> > aggregates. make_window_input_target() flattens the final tlist using
> > pull_var_clause, which pulls the Aggref out of its JsonConstructorExpr
> > wrapper. Then fix_upper_expr() matches that inner Aggref against the
> > Agg subplan's tlist and replaces it with an OUTER Var.
>
> Here is the patch. It's a bit annoying that the original JSON agg
> syntax then appears nowhere in the EXPLAIN output. All we get is the
> bare jsonb_agg_strict Aggref.
>
> WindowAgg
> Output: g, (jsonb_agg_strict(name)), row_number() OVER w1
> Window: w1 AS (ROWS UNBOUNDED PRECEDING)
> -> HashAggregate
> Output: g, jsonb_agg_strict(name)
>
> The reason is that the JsonConstructorExpr wrapper and the Aggref it
> wraps end up in different plan nodes, and neither alone suffices to
> reconstruct the syntax.
>
> I had an attempt to reconstruct the JSON syntax for the WindowAgg node
> by leveraging resolve_special_varno(), and that works. But I don't
> know how to do that for the HashAggregate node, because the
> JsonConstructorExpr wrapper simply doesn't exist at that plan level.
> Maybe we can hack the planner to make make_window_input_target() keep
> the JsonConstructorExpr together with its Aggref. But I think that is
> too invasive and it changes which node evaluates the wrapper.
>
> So that attempt ended up with:
>
> WindowAgg
> Output: g, JSON_ARRAYAGG(name RETURNING jsonb), row_number() OVER w1
> Window: w1 AS (ROWS UNBOUNDED PRECEDING)
> -> HashAggregate
> Output: g, jsonb_agg_strict(name)
>
> But I don't think this is good. It fails to state the fact that the
> WindowAgg doesn't compute a JSON aggregate; it passes through a value
> that the HashAggregate computed. So I gave up this idea.
Thanks for taking a look at this. It's unfortunate that reconstructing
the syntax is problematic, because in the case of the original query
that tripped on this bug, I lose the following from my original query:
JSON_ARRAYAGG(i RETURNING jsonb)
is instead:
jsonb_agg_strict(i)
And the same with: JSON_ARRAYAGG(…)
JSON_ARRAYAGG(i RETURNING json)
is instead:
json_agg_strict(i)
And the same with: JSON_ARRAYAGG(i RETURNING text)
WITH UNIQUE KEYS
is instead:
json_object_agg_unique(i,i)
NULL ON NULL
This isn't reconstructed
I guess the relevant information can be gleaned from this, but not for
JSON_ARRAYAGG(i RETURNING text) because we get the same output whether
we're returning text or json.
Thom
| From | Date | Subject | |
|---|---|---|---|
| Next Message | Ayush Tiwari | 2026-07-03 08:40:44 | Re: Fw:Re: Fw: gbt_var_consistent in contrib/btree_gist/btree_utils_var.c has internal-node type confusion on the <> strategy, bypassing exclusion constraints |
| Previous Message | Imran Zaheer | 2026-07-03 06:45:42 | Re: BUG #19519: REPACK can fail due to missing chunk for toast value |