Sie sollten eine hierarchische Abfrage erstellen, um als Ergebnis eine hierarchische Struktur zu erhalten.
Sie möchten viele Personen in einem einzigen JSON-Objekt haben, also verwenden Sie json_agg()
um Personen in einem json-Array zu sammeln. Analog kann eine Person mehrere Autos haben und Sie sollten Autos, die einer einzelnen Person gehören, in einem json-Array platzieren. Gleiches gilt für Autos und Räder.
select
json_build_object(
'persons', json_agg(
json_build_object(
'person_name', p.name,
'cars', cars
)
)
) persons
from person p
left join (
select
personid,
json_agg(
json_build_object(
'carid', c.id,
'type', c.type,
'comment', 'nice car', -- this is constant
'wheels', wheels
)
) cars
from
car c
left join (
select
carid,
json_agg(
json_build_object(
'which', w.whichone,
'serial number', w.serialnumber
)
) wheels
from wheel w
group by 1
) w on c.id = w.carid
group by personid
) c on p.id = c.personid;
Das (formatierte) Ergebnis:
{
"persons": [
{
"person_name": "Johny",
"cars": [
{
"carid": 1,
"type": "Toyota",
"comment": "nice car",
"wheels": [
{
"which": "front",
"serial number": 11
},
{
"which": "back",
"serial number": 12
}
]
},
{
"carid": 2,
"type": "Fiat",
"comment": "nice car",
"wheels": [
{
"which": "front",
"serial number": 21
},
{
"which": "back",
"serial number": 22
}
]
}
]
},
{
"person_name": "Freddy",
"cars": [
{
"carid": 3,
"type": "Opel",
"comment": "nice car",
"wheels": [
{
"which": "front",
"serial number": 3
}
]
}
]
}
]
}
Wenn Sie mit verschachtelten abgeleiteten Tabellen nicht vertraut sind, können Sie gängige Tabellenausdrücke verwenden. Diese Variante veranschaulicht, dass die Abfrage beginnend mit dem am stärksten verschachtelten Objekt zur höchsten Ebene aufgebaut werden sollte:
with wheels as (
select
carid,
json_agg(
json_build_object(
'which', w.whichone,
'serial number', w.serialnumber
)
) wheels
from wheel w
group by 1
),
cars as (
select
personid,
json_agg(
json_build_object(
'carid', c.id,
'type', c.type,
'comment', 'nice car', -- this is constant
'wheels', wheels
)
) cars
from car c
left join wheels w on c.id = w.carid
group by c.personid
)
select
json_build_object(
'persons', json_agg(
json_build_object(
'person_name', p.name,
'cars', cars
)
)
) persons
from person p
left join cars c on p.id = c.personid;