1. ホーム

[解決済み】新しいPostgreSQL JSONデータ型内のフィールドを変更するにはどうすればよいですか?

2022-03-27 19:42:38

質問

postgresql 9.3では、以下のことが可能です。 SELECT JSONデータ型の特定のフィールドを変更することはできますが、その変更は UPDATE ? postgresqlのドキュメントにも、ネット上のどこにも、このような例は見当たりません。私は当たり前のことをやってみました。

postgres=# create table test (data json);
CREATE TABLE
postgres=# insert into test (data) values ('{"a":1,"b":2}');
INSERT 0 1
postgres=# select data->'a' from test where data->>'b' = '2';
 ?column?
----------
 1
(1 row)
postgres=# update test set data->'a' = to_json(5) where data->>'b' = '2';
ERROR:  syntax error at or near "->"
LINE 1: update test set data->'a' = to_json(5) where data->>'b' = '2...

解決方法は?

更新情報 : PostgreSQL 9.5で は、いくつかの jsonb を操作する機能は、PostgreSQL 自身にはありません。 json を操作するには、キャストが必要です。 json の値)。

2つ(またはそれ以上)のJSONオブジェクトをマージする(または配列を連結する)。

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]'

だから 単純なキーの設定 を使って行うことができます。

SELECT jsonb '{"a":1}' || jsonb_build_object('<key>', '<value>')

ここで <key> は文字列でなければならず <value> はどんな型でも to_jsonb() を受け入れます。

について JSONの深い階層に値を設定する。 を使用することで jsonb_set() 関数が使用できる。

SELECT jsonb_set('{"a":[null,{"b":[]}]}', '{a,1,b,0}', jsonb '{"c":3}')
-- will yield jsonb '{"a":[null,{"b":[{"c":3}]}]}'

の全パラメータリスト jsonb_set() :

jsonb_set(target         jsonb,
          path           text[],
          new_value      jsonb,
          create_missing boolean default true)

path はJSON配列のインデックスを含むことができます & そこに現れる負の整数は、JSON配列の最後から数えます。しかし、存在しないが正の JSON 配列インデックスは、その要素を配列の末尾に追加します。

SELECT jsonb_set('{"a":[null,{"b":[1,2]}]}', '{a,1,b,1000}', jsonb '3', true)
-- will yield jsonb '{"a":[null,{"b":[1,2,3]}]}'

について JSON配列に挿入する(元の値をすべて保持する)。 を使用すると jsonb_insert() 関数を使用することができます ( 9.6+ではこの機能のみ、この項では ):

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]}]}'

の全パラメータリスト jsonb_insert() :

jsonb_insert(target       jsonb,
             path         text[],
             new_value    jsonb,
             insert_after boolean default false)

ここでも、負の整数が path は、JSON配列の末尾から数えます。

そのため、例えばJSON配列の末尾への追加などは

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

しかし、この関数は少し違った働きをしています( jsonb_set() を使用する場合 pathtarget はJSONオブジェクトのキーです。その場合、そのキーが使われていないときだけ、JSONオブジェクトの新しいキーと値のペアを追加します。もし使用されていれば、エラーを発生させます。

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

キー(またはインデックス)を削除する は、JSON オブジェクトから (または配列から) 処理することができます。 - 演算子を使用します。

SELECT jsonb '{"a":1,"b":2}' - 'a', -- will yield jsonb '{"b":2}'
       jsonb '["a",1,"b",2]' - 1    -- will yield jsonb '["a","b",2]'

JSON階層の深いところからの削除 #- 演算子を使用します。

SELECT '{"a":[null,{"b":[3.14]}]}' #- '{a,1,b,0}'
-- will yield jsonb '{"a":[null,{"b":[]}]}'

9.4用 しかし、JSON文字列を集約する代わりに、以下のように直接jsonオブジェクトに集約することができます。 json_object_agg() .

オリジナルの回答 : plpythonやplv8を使わずに)純粋なSQLでも可能です(ただし、9.3以上が必要で、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

編集 :

複数のキーと値を設定するバージョンです。

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$;

編集2 として。 これらの関数は、いわゆる UPSERT (フィールドが存在する場合は更新し、存在しない場合は挿入する)。以下はその変形版で 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$;

3を編集する : これは再帰的なバリエーションで、( UPSERT ) にあるリーフ値 (そしてこの答えの最初の関数を使用します) を、キーパス (ここでキーは内部のオブジェクトのみを参照でき、内部の配列はサポートされていません) に配置します。

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$;

更新しました。既存のjsonフィールドのキーを別のキーで置き換える機能を追加しました。マイグレーションでデータ型を更新するときや、データ構造を修正するようなシナリオで便利です。

CREATE OR REPLACE FUNCTION json_object_replace_key(
    json_value json,
    existing_key text,
    desired_key text)
  RETURNS json AS
$BODY$
SELECT COALESCE(
(
    SELECT ('{' || string_agg(to_json(key) || ':' || value, ',') || '}')
    FROM (
        SELECT *
        FROM json_each(json_value)
        WHERE key <> existing_key
        UNION ALL
        SELECT desired_key, json_value -> existing_key
    ) AS "fields"
    -- WHERE value IS NOT NULL (Actually not required as the string_agg with value's being null will "discard" that entry)

),
    '{}'
)::json
$BODY$
  LANGUAGE sql IMMUTABLE STRICT
  COST 100;

更新情報 の機能をコンパクトにまとめました。