Re: EXPLAIN (VERBOSE) fails with for JSON_ARRAYAGG/JSON_OBJECTAGG + window function

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

In response to

Browse pgsql-bugs by date

  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