Update : Mit PostgreSQL 9.5 gibt es einige jsonb
Manipulationsfunktionen in PostgreSQL selbst (aber keine für json
; Casts sind erforderlich, um json
Werte zu manipulieren ).
Zusammenführen von 2 (oder mehr) JSON-Objekten (oder Verketten von Arrays):
SELECT jsonb '{"a":1}' || jsonb '{"b":2}', -- will yield jsonb '{"a":1,"b":2}'
jsonb '["a",1]' || jsonb '["b",2]' -- will yield jsonb '["a",1,"b",2]'
Das Einstellen eines einfachen Schlüssels kann also folgendermaßen erfolgen:
SELECT jsonb '{"a":1}' || jsonb_build_object('<key>', '<value>')
Wo <key>
sollte Zeichenfolge sein, und <value>
kann jeder Typ sein, der to_jsonb()
akzeptiert.
Zum Festlegen eines Werts tief in einer JSON-Hierarchie kann die folgende jsonb_set()
Funktion verwendet werden:
SELECT jsonb_set('{"a":[null,{"b":[]}]}', '{a,1,b,0}', jsonb '{"c":3}')
-- will yield jsonb '{"a":[null,{"b":[{"c":3}]}]}'
Vollständige Parameterliste von jsonb_set()
:
jsonb_set(target jsonb,
path text[],
new_value jsonb,
create_missing boolean default true)
path
kann auch JSON-Array-Indizes enthalten & negative Ganzzahlen, die dort erscheinen, zählen ab dem Ende von JSON-Arrays. Ein nicht vorhandener, aber positiver JSON-Array-Index hängt das Element jedoch an das Ende des Arrays an:
SELECT jsonb_set('{"a":[null,{"b":[1,2]}]}', '{a,1,b,1000}', jsonb '3', true)
-- will yield jsonb '{"a":[null,{"b":[1,2,3]}]}'
Zum Einfügen in ein JSON-Array (unter Beibehaltung aller ursprünglichen Werte) kann die jsonb_insert()
Funktion verwendet werden ( in 9.6+; nur diese Funktion in diesem Abschnitt ):
SELECT jsonb_insert('{"a":[null,{"b":[1]}]}', '{a,1,b,0}', jsonb '2')
-- will yield jsonb '{"a":[null,{"b":[2,1]}]}', and
SELECT jsonb_insert('{"a":[null,{"b":[1]}]}', '{a,1,b,0}', jsonb '2', true)
-- will yield jsonb '{"a":[null,{"b":[1,2]}]}'
Vollständige Parameterliste von jsonb_insert()
:
jsonb_insert(target jsonb,
path text[],
new_value jsonb,
insert_after boolean default false)
Wiederum zählen negative Ganzzahlen, die path
ab dem Ende der JSON-Arrays angezeigt werden.
Also, z. Das Anhängen an ein Ende eines JSON-Arrays kann erfolgen mit:
SELECT jsonb_insert('{"a":[null,{"b":[1,2]}]}', '{a,1,b,-1}', jsonb '3', true)
-- will yield jsonb '{"a":[null,{"b":[1,2,3]}]}', and
Diese Funktion funktioniert jedoch etwas anders (als jsonb_set()
), wenn das path
In target
der Schlüssel eines JSON-Objekts ist. In diesem Fall wird nur dann ein neues Schlüssel-Wert-Paar für das JSON-Objekt hinzugefügt, wenn der Schlüssel nicht verwendet wird. Wenn es verwendet wird, wird ein Fehler ausgegeben:
SELECT jsonb_insert('{"a":[null,{"b":[1]}]}', '{a,1,c}', jsonb '[2]')
-- will yield jsonb '{"a":[null,{"b":[1],"c":[2]}]}', but
SELECT jsonb_insert('{"a":[null,{"b":[1]}]}', '{a,1,b}', jsonb '[2]')
-- will raise SQLSTATE 22023 (invalid_parameter_value): cannot replace existing key
Das Löschen eines Schlüssels (oder eines Index) aus einem JSON-Objekt (oder aus einem Array) kann mit dem -
Operator erfolgen:
SELECT jsonb '{"a":1,"b":2}' - 'a', -- will yield jsonb '{"b":2}'
jsonb '["a",1,"b",2]' - 1 -- will yield jsonb '["a","b",2]'
Das Löschen aus einer JSON-Hierarchie heraus kann mit dem #-
Operator erfolgen:
SELECT '{"a":[null,{"b":[3.14]}]}' #- '{a,1,b,0}'
-- will yield jsonb '{"a":[null,{"b":[]}]}'
Für 9.4 können Sie eine geänderte Version der ursprünglichen Antwort (unten) verwenden. Anstatt jedoch eine JSON-Zeichenfolge zu aggregieren, können Sie direkt mit zu einem JSON-Objekt aggregieren json_object_agg()
.
Ursprüngliche Antwort : Es ist möglich (ohne plpython oder plv8) auch in reinem SQL (benötigt aber 9.3+, funktioniert nicht mit 9.2)
CREATE OR REPLACE FUNCTION "json_object_set_key"(
"json" json,
"key_to_set" TEXT,
"value_to_set" anyelement
)
RETURNS json
LANGUAGE sql
IMMUTABLE
STRICT
AS $function$
SELECT concat('{', string_agg(to_json("key") || ':' || "value", ','), '}')::json
FROM (SELECT *
FROM json_each("json")
WHERE "key" <> "key_to_set"
UNION ALL
SELECT "key_to_set", to_json("value_to_set")) AS "fields"
$function$;
SQLFiddle
Bearbeiten :
Eine Version, die mehrere Schlüssel und Werte festlegt:
CREATE OR REPLACE FUNCTION "json_object_set_keys"(
"json" json,
"keys_to_set" TEXT[],
"values_to_set" anyarray
)
RETURNS json
LANGUAGE sql
IMMUTABLE
STRICT
AS $function$
SELECT concat('{', string_agg(to_json("key") || ':' || "value", ','), '}')::json
FROM (SELECT *
FROM json_each("json")
WHERE "key" <> ALL ("keys_to_set")
UNION ALL
SELECT DISTINCT ON ("keys_to_set"["index"])
"keys_to_set"["index"],
CASE
WHEN "values_to_set"["index"] IS NULL THEN 'null'::json
ELSE to_json("values_to_set"["index"])
END
FROM generate_subscripts("keys_to_set", 1) AS "keys"("index")
JOIN generate_subscripts("values_to_set", 1) AS "values"("index")
USING ("index")) AS "fields"
$function$;
Bearbeiten 2 : Wie @ErwinBrandstetter feststellte, funktionieren diese Funktionen wie ein sogenanntes UPSERT
(aktualisiert ein Feld, falls vorhanden, fügt ein, wenn es nicht vorhanden ist). Hier ist eine Variante, die nur UPDATE
:
CREATE OR REPLACE FUNCTION "json_object_update_key"(
"json" json,
"key_to_set" TEXT,
"value_to_set" anyelement
)
RETURNS json
LANGUAGE sql
IMMUTABLE
STRICT
AS $function$
SELECT CASE
WHEN ("json" -> "key_to_set") IS NULL THEN "json"
ELSE (SELECT concat('{', string_agg(to_json("key") || ':' || "value", ','), '}')
FROM (SELECT *
FROM json_each("json")
WHERE "key" <> "key_to_set"
UNION ALL
SELECT "key_to_set", to_json("value_to_set")) AS "fields")::json
END
$function$;
Bearbeiten 3 : Hier ist eine rekursive Variante, die UPSERT
einen Blattwert setzen ( ) kann (und die erste Funktion aus dieser Antwort verwendet), die sich an einem Schlüsselpfad befindet (wobei Schlüssel nur auf innere Objekte verweisen können, innere Arrays nicht unterstützt werden):
CREATE OR REPLACE FUNCTION "json_object_set_path"(
"json" json,
"key_path" TEXT[],
"value_to_set" anyelement
)
RETURNS json
LANGUAGE sql
IMMUTABLE
STRICT
AS $function$
SELECT CASE COALESCE(array_length("key_path", 1), 0)
WHEN 0 THEN to_json("value_to_set")
WHEN 1 THEN "json_object_set_key"("json", "key_path"[l], "value_to_set")
ELSE "json_object_set_key"(
"json",
"key_path"[l],
"json_object_set_path"(
COALESCE(NULLIF(("json" -> "key_path"[l])::text, 'null'), '{}')::json,
"key_path"[l+1:u],
"value_to_set"
)
)
END
FROM array_lower("key_path", 1) l,
array_upper("key_path", 1) u
$function$;
Update : Funktionen sind jetzt komprimiert.
select json_object_set_key((select data from test where data->>'b' = '2'), 'b', 'two');
Fehlermeldung wird ein FehlerERROR: could not determine polymorphic type because input has type "unknown"