| From: | PG Bug reporting form <noreply(at)postgresql(dot)org> | 
|---|---|
| To: | pgsql-bugs(at)lists(dot)postgresql(dot)org | 
| Cc: | muhlemmer(at)gmail(dot)com | 
| Subject: | BUG #15884: json_object_agg errors on null in field name | 
| Date: | 2019-07-02 15:04:15 | 
| Message-ID: | 15884-c32d848f787fcae3@postgresql.org | 
| Views: | Whole Thread | Raw Message | Download mbox | Resend email | 
| Thread: | |
| Lists: | pgsql-bugs | 
The following bug has been logged on the website:
Bug reference:      15884
Logged by:          Tim Möhlmann
Email address:      muhlemmer(at)gmail(dot)com
PostgreSQL version: 11.3
Operating system:   Gentoo linux (stable)
Description:        
According to the documentation on aggregate expressions:
https://www.postgresql.org/docs/11/sql-expressions.html#SYNTAX-AGGREGATES
"Most aggregate functions ignore null inputs, so that rows in which one or
more of the expression(s) yield null are discarded. This can be assumed to
be true, unless otherwise specified, for all built-in aggregates."
On aggregate function documentation no specific mention is made for
json_object_agg().
https://www.postgresql.org/docs/11/functions-aggregate.html
However, json_object_agg() throws an error in case of null in the
argument(s): "error: field name must not be null". And I get it, JSON keys
need to be unique strings and null is not that. However, this error is also
thrown if both keys and values are an empty CTE result.
In the following example there are pages. Each page has sections and each
sections has text fields (texts). It is a simplified version of my app's
query. When there is a page without sections json_object_agg() gets the
empty result for "s.title" and the nested json_build_object() call.
create table pages (
    page_id serial primary key,
    domain text unique not null
);
create table sections (
    section_id serial primary key,
    title text not null,
    page_id int references pages
);
create table texts (
    section_id int references sections,
    pos int not null,
    content text not null,
    primary key (section_id, pos)
);
-- spanac.com will have 3 sections with texts and images in each, various
amounts
insert into pages (domain) values ('spanac.com');
-- foo.com has 1 empty section
insert into pages (domain) values ('foo.com');
-- bar.com has no sections
insert into pages (domain) values ('bar.com');
-- spanac.com
with s as (
    insert into sections (page_id, title) select page_id, 'first' from pages
where domain = 'spanac.com' returning section_id
),
t1 as (
    insert into texts (section_id, pos, content) select section_id, 1,
'spanac one.one' from s
)
insert into texts (section_id, pos, content) select section_id, 2, 'spanac
one.two' from s;
with s as (
    insert into sections (page_id, title) select page_id, 'second' from
pages where domain = 'spanac.com' returning section_id
),
t1 as (
    insert into texts (section_id, pos, content) select section_id, 1,
'spanac two.one' from s
)
insert into texts (section_id, pos, content) select section_id, 2, 'spanac
two.two' from s;
-- foo.com
insert into sections (page_id, title) select page_id, 'empty' from pages
where domain = 'foo.com';
And this is the query that triggers the error:
with secs as (
	select p.page_id, p.domain, s.section_id as sid, s.title as title
	from pages p
	left join sections s on p.page_id = s.page_id
	where p.domain = 'bar.com' -- 'foo.com' and 'spanac.com' work fine
),
txt as (
	select
		sid,
		json_agg(
			json_build_object(
				'Pos', pos,
				'Text', content
			)
			order by pos asc
		) as txts
	from texts
	join secs on sid = section_id
	group by sid
)
select
	json_build_object(
		'ID', s.page_id,
        'Domain', domain,
		'Sections', json_object_agg ( -- Error occurs here
			s.title,
          	json_build_object(
				'ID', s.sid,
				'Texts', t.txts
			)
			order by s.sid asc
		)
	)
from secs s
left join txt t on s.sid = t.sid
group by s.page_id, domain;
The above is also available in a fiddle, although it does not match the
PostgreSQL version: https://www.db-fiddle.com/f/nzZz7jjrDSAYKtYK53bz7L/1
I've asked a question regarding this on:
https://dba.stackexchange.com/q/241541/150398. The answer I got is that this
might be a bug, hence I'm posting here. Two notes:
1. I cannot reproduce this when I simplify the query into not using a CTE
and nested json_build_object()
2. I "blame" json_object_agg()", because I used a regular json_agg() without
the s.title fields before and it worked fine. As in, the resulting json
document just had "Sections": null.
Best regards, Tim Mohlmann (muhlemmer)
| From | Date | Subject | |
|---|---|---|---|
| Next Message | PG Bug reporting form | 2019-07-02 19:43:15 | BUG #15885: EnterpriseDB Installer upgrade process fails with icacls error | 
| Previous Message | PG Bug reporting form | 2019-07-02 14:57:50 | BUG #15883: Event Trigger Firing Matrix Table is incomplete |