web-dev-qa-db-ja.com

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

Postgresql 9.3ではJSONデータ型の特定のフィールドをSELECTすることができますが、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...
183
user9645

更新PostgreSQL 9.5 では、PostgreSQL自体にいくつかのjsonb操作機能があります(ただし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配列の末尾から数えます。

だから、f.ex. 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

ただし、path内のtargetがJSONオブジェクトのキーである場合、この関数は(jsonb_set()とは)動作が多少異なります。その場合、キーが使用されていないときにのみ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_object_agg()を使用して直接jsonオブジェクトに集約できます。

元の答え:純粋なSQLでも可能です(plpythonまたはplv8なしで)(ただし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:@ErwinBrandstetter が述べたように 上記のこれらの関数はいわゆるUPSERTのように機能します(存在する場合はfield、存在しない場合は挿入します。これは変種で、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$;

更新:関数は現在圧縮されています。

264
pozs

9.5ではjsonb_set-を使う

UPDATE objects
SET body = jsonb_set(body, '{name}', '"Mary"', true)
WHERE id = 1; 

bodyはjsonb列型です。

64
Teo Choong Ping

Postgresql 9.5では、次のようにして実行できます。

UPDATE test
SET data = data - 'a' || '{"a":5}'
WHERE data->>'b' = '2';

OR

UPDATE test
SET data = jsonb_set(data, '{a}', '5'::jsonb);

誰かが一度にjsonb値の多くのフィールドを更新する方法を尋ねた。テーブルを作成するとします。

CREATE TABLE testjsonb ( id SERIAL PRIMARY KEY, object JSONB );

次に実験行を挿入します。

INSERT INTO testjsonb
VALUES (DEFAULT, '{"a":"one", "b":"two", "c":{"c1":"see1","c2":"see2","c3":"see3"}}');

それから行を更新します。

UPDATE testjsonb SET object = object - 'b' || '{"a":1,"d":4}';

これは次のことを行います。

  1. Aフィールドを更新します
  2. Bフィールドを削除します
  3. Dフィールドを追加

データを選択する:

SELECT jsonb_pretty(object) FROM testjsonb;

結果は次のようになります。

      jsonb_pretty
-------------------------
 {                      +
     "a": 1,            +
     "c": {             +
         "c1": "see1",  +
         "c2": "see2",  +
         "c3": "see3",  +
     },                 +
     "d": 4             +
 }
(1 row)

内部のフィールドを更新するには、連結演算子||を使用しないでください。代わりにjsonb_setを使用してください。単純ではありません。

UPDATE testjsonb SET object =
jsonb_set(jsonb_set(object, '{c,c1}','"seeme"'),'{c,c2}','"seehim"');

たとえば、{c、c1}に連結演算子を使用します。

UPDATE testjsonb SET object = object || '{"c":{"c1":"seedoctor"}}';

{c、c2}と{c、c3}を削除します。

より強力なものについては、 postgresql json関数のドキュメント で力を求めてください。 #-演算子、jsonb_set関数、そしてjsonb_insert関数に興味があるかもしれません。

35
Fandi Susanto
UPDATE test
SET data = data::jsonb - 'a' || '{"a":5}'::jsonb
WHERE data->>'b' = '2'

これはPostgreSQL 9.5に取り組んでいるようです

9
sigod

@ pozsの答えを基に、いくつかの人にとって役に立つかもしれない、さらにいくつかのPostgreSQL関数があります。 (PostgreSQL 9.3以降が必要です)

キーによる削除:キーによってJSON構造から値を削除します。

CREATE OR REPLACE FUNCTION "json_object_del_key"(
  "json"          json,
  "key_to_del"    TEXT
)
  RETURNS json
  LANGUAGE sql
  IMMUTABLE
  STRICT
AS $function$
SELECT CASE
  WHEN ("json" -> "key_to_del") IS NULL THEN "json"
  ELSE (SELECT concat('{', string_agg(to_json("key") || ':' || "value", ','), '}')
          FROM (SELECT *
                  FROM json_each("json")
                 WHERE "key" <> "key_to_del"
               ) AS "fields")::json
END
$function$;

キーによる再帰的削除:キーパスでJSON構造から値を削除します。 (@ pozsのjson_object_set_key関数が必要です)

CREATE OR REPLACE FUNCTION "json_object_del_path"(
  "json"          json,
  "key_path"      TEXT[]
)
  RETURNS json
  LANGUAGE sql
  IMMUTABLE
  STRICT
AS $function$
SELECT CASE
  WHEN ("json" -> "key_path"[l] ) IS NULL THEN "json"
  ELSE
     CASE COALESCE(array_length("key_path", 1), 0)
         WHEN 0 THEN "json"
         WHEN 1 THEN "json_object_del_key"("json", "key_path"[l])
         ELSE "json_object_set_key"(
           "json",
           "key_path"[l],
           "json_object_del_path"(
             COALESCE(NULLIF(("json" -> "key_path"[l])::text, 'null'), '{}')::json,
             "key_path"[l+1:u]
           )
         )
       END
    END
  FROM array_lower("key_path", 1) l,
       array_upper("key_path", 1) u
$function$;

使用例:

s1=# SELECT json_object_del_key ('{"hello":[7,3,1],"foo":{"mofu":"fuwa", "moe":"kyun"}}',
                                 'foo'),
            json_object_del_path('{"hello":[7,3,1],"foo":{"mofu":"fuwa", "moe":"kyun"}}',
                                 '{"foo","moe"}');

 json_object_del_key |          json_object_del_path
---------------------+-----------------------------------------
 {"hello":[7,3,1]}   | {"hello":[7,3,1],"foo":{"mofu":"fuwa"}}
9
shru

PostgreSQL 9.4では、以下のPython関数を実装しました。 PostgreSQL 9.3でも動作します。

create language plpython2u;

create or replace function json_set(jdata jsonb, jpaths jsonb, jvalue jsonb) returns jsonb as $$
import json

a = json.loads(jdata)
b = json.loads(jpaths)

if a.__class__.__!= 'dict' and a.__class__.__!= 'list':
  raise plpy.Error("The json data must be an object or a string.")

if b.__class__.__!= 'list':
   raise plpy.Error("The json path must be an array of paths to traverse.")

c = a
for i in range(0, len(b)):
  p = b[i]
  plpy.notice('p == ' + str(p))

  if i == len(b) - 1:
    c[p] = json.loads(jvalue)

  else:
    if p.__class__.__== 'unicode':
      plpy.notice("Traversing '" + p + "'")
      if c.__class__.__!= 'dict':
        raise plpy.Error("  The value here is not a dictionary.")
      else:
        c = c[p]

    if p.__class__.__== 'int':
      plpy.notice("Traversing " + str(p))
      if c.__class__.__!= 'list':
        raise plpy.Error("  The value here is not a list.")
      else:
        c = c[p]

    if c is None:
      break    

return json.dumps(a)
$$ language plpython2u ;

使用例

create table jsonb_table (jsonb_column jsonb);
insert into jsonb_table values
('{"cars":["Jaguar", {"type":"Unknown","partsList":[12, 34, 56]}, "Atom"]}');

select jsonb_column->'cars'->1->'partsList'->2, jsonb_column from jsonb_table;

update jsonb_table
set jsonb_column = json_set(jsonb_column, '["cars",1,"partsList",2]', '99');

select jsonb_column->'cars'->1->'partsList'->2, jsonb_column from jsonb_table;

以前の雇用者のために、PostgreSQL 7、8、9のJSONデータをテキストとして(json型やjsonb型としてではなく)操作するための一連のC関数を書いたことに注意してください。例えば、json_path('{"obj":[12, 34, {"num":-45.67}]}', '$.obj[2]['num']')json_path_set('{"obj":[12, 34, {"num":-45.67}]}', '$.obj[2]['num']', '99.87')などでデータを設定します。作業には約3日かかりました。そのため、従来のシステムで実行して暇な時間がある場合は、努力する価値があります。私は、CバージョンがPythonバージョンよりはるかに速いと思います。

4
Magnus

あなたのフィールドタイプがjsonの場合、以下があなたのために働きます。

UPDATE 
table_name
SET field_name = field_name::jsonb - 'key' || '{"key":new_val}' 
WHERE field_name->>'key' = 'old_value'.

演算子 ' - '左のオペランドからキー/値ペアまたは文字列要素を削除します。キーと値のペアは、それらのキー値に基づいて照合されます。

演算子 '||' 2つのjsonb値を新しいjsonb値に連結します。

これらはjsonb演算子なので、:: jsonbに型キャストする必要があります

より詳しい情報: JSON関数と演算子

あなたは私のノートをここで読むことができます

3
Neethu

以下はこの要求を満たしませんが(関数json_object_aggはPostgreSQL 9.3では利用できません)、以下は||を探している人には役に立ちます。今後のPostgreSQL 9.5で実装されている、PostgreSQL 9.4のための演算子。

CREATE OR REPLACE FUNCTION jsonb_merge(left JSONB, right JSONB)
RETURNS JSONB
AS $$
SELECT
  CASE WHEN jsonb_typeof($1) = 'object' AND jsonb_typeof($2) = 'object' THEN
       (SELECT json_object_agg(COALESCE(o.key, n.key), CASE WHEN n.key IS NOT NULL THEN n.value ELSE o.value END)::jsonb
        FROM jsonb_each($1) o
        FULL JOIN jsonb_each($2) n ON (n.key = o.key))
   ELSE 
     (CASE WHEN jsonb_typeof($1) = 'array' THEN LEFT($1::text, -1) ELSE '['||$1::text END ||', '||
      CASE WHEN jsonb_typeof($2) = 'array' THEN RIGHT($2::text, -1) ELSE $2::text||']' END)::jsonb
   END     
$$ LANGUAGE sql IMMUTABLE STRICT;
GRANT EXECUTE ON FUNCTION jsonb_merge(jsonb, jsonb) TO public;
CREATE OPERATOR || ( LEFTARG = jsonb, RIGHTARG = jsonb, PROCEDURE = jsonb_merge );

私はPostgres 9.4で再帰的に動作する小さな関数を自分で書いた。これが機能です(私はそれがあなたのためにうまくいくことを願っています):

CREATE OR REPLACE FUNCTION jsonb_update(val1 JSONB,val2 JSONB)
RETURNS JSONB AS $$
DECLARE
    result JSONB;
    v RECORD;
BEGIN
    IF jsonb_typeof(val2) = 'null'
    THEN 
        RETURN val1;
    END IF;

    result = val1;

    FOR v IN SELECT key, value FROM jsonb_each(val2) LOOP

        IF jsonb_typeof(val2->v.key) = 'object'
            THEN
                result = result || jsonb_build_object(v.key, jsonb_update(val1->v.key, val2->v.key));
            ELSE
                result = result || jsonb_build_object(v.key, v.value);
        END IF;
    END LOOP;

    RETURN result;
END;
$$ LANGUAGE plpgsql;

これが使用例です。

select jsonb_update('{"a":{"b":{"c":{"d":5,"dd":6},"cc":1}},"aaa":5}'::jsonb, '{"a":{"b":{"c":{"d":15}}},"aa":9}'::jsonb);
                            jsonb_update                             
---------------------------------------------------------------------
 {"a": {"b": {"c": {"d": 15, "dd": 6}, "cc": 1}}, "aa": 9, "aaa": 5}
(1 row)

ご覧のとおり、詳細に分析し、必要に応じて値を更新/追加します。

2
J. Raczkiewicz

次のplpythonの抜粋が役に立つかもしれません。

CREATE EXTENSION IF NOT EXISTS plpythonu;
CREATE LANGUAGE plpythonu;

CREATE OR REPLACE FUNCTION json_update(data json, key text, value text)
 RETURNS json
 AS $$
    import json
    json_data = json.loads(data)
    json_data[key] = value
    return json.dumps(json_data, indent=4)
 $$ LANGUAGE plpythonu;

-- Check how JSON looks before updating

SELECT json_update(content::json, 'CFRDiagnosis.mod_nbs', '1')
FROM sc_server_centre_document WHERE record_id = 35 AND template = 'CFRDiagnosis';

-- Once satisfied update JSON inplace

UPDATE sc_server_centre_document SET content = json_update(content::json, 'CFRDiagnosis.mod_nbs', '1')
WHERE record_id = 35 AND template = 'CFRDiagnosis';
1
Sandeep

残念ながら、ドキュメントには何も見つかりませんでしたが、いくつかの回避策を使用できます。たとえば、拡張機能を記述することができます。

例えば、Pythonでは:

CREATE or REPLACE FUNCTION json_update(data json, key text, value json)
returns json
as $$
from json import loads, dumps
if key is None: return data
js = loads(data)
js[key] = value
return dumps(js)
$$ language plpython3u

その後

update test set data=json_update(data, 'a', to_json(5)) where data->>'b' = '2';
1
Roman Pekar

mybatisを使用する場合、更新ステートメントの例を次に示します。

<update id="saveAnswer">
    update quiz_execution set answer_data = jsonb_set(answer_data, concat('{', #{qid}, '}')::text[], #{value}::jsonb), updated_at = #{updatedAt}
    where id = #{id}
</update>


パラメーター:

  • qid、フィールドのキー。
  • valueは、フィールド値の有効なJSON文字列です。
    e.g jacksonを介してオブジェクトからjson文字列に変換、
0
Eric Wang

文字列型のフィールドを更新しようとしたとき、これは私のために働いた。

UPDATE table_name 
SET body = jsonb_set(body, '{some_key}', to_json('value'::TEXT)::jsonb);

それが他の誰かに役立つことを願っています!

0
Antonio

このようにjsonb内でキーをアトミックにインクリメントすることもできます。

UPDATE users SET counters = counters || CONCAT('{"bar":', COALESCE(counters->>'bar','0')::int + 1, '}')::jsonb WHERE id = 1;

SELECT * FROM users;

 id |    counters
----+------------
  1 | {"bar": 1}

未定義キー - >は開始値0を仮定します。

より詳細な説明については、ここで私の答えを参照してください。 https://stackoverflow.com/a/39076637

0
joonas.fi

以前の回答は経験豊富なPostgreSQLユーザーに適しているため、私の回答は次のとおりです。

次の値を持つJSONBタイプのテーブル列があると仮定します。

{
    "key0": {
        "key01": "2018-05-06T12:36:11.916761+00:00",
        "key02": "DEFAULT_WEB_CONFIGURATION",

    "key1": {
        "key11": "Data System",
        "key12": "<p>Health,<p>my address<p>USA",
        "key13": "*Please refer to main screen labeling"
    }
}

行に新しい値を設定したいと仮定しましょう:

"key13": "*Please refer to main screen labeling"

代わりに値を配置します。

"key13": "See main screen labeling"

json_set()関数を使用して、key13に新しい値を割り当てます

jsonb_set() へのパラメーター

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

in "target"-jsonb column-nameを配置します(これは変更されるテーブル列です)

path」-は「json keys path」であり、上書きするキーにつながる(そして含む)

new_value」-これは割り当てる新しい値です

この場合、key1の下にあるkey13の値を更新したい(key1-> key13):

したがって、パス構文は次のとおりです: '{key1、key13}'(パスはクラックするのが最も難しい部分でした-チュートリアルはひどいため

jsonb_set(jsonb_column,'{key1,key13}','"See main screen labeling"')
0
Dror