Как изменить поля внутри нового типа данных PostgreSQL JSON?

159

С postgresql 9.3 Я могу ВЫБРАТЬ определенные поля типа данных 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.3
postgresql-json

14 ответов

221
Лучший ответ

Обновить: С 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

Однако эта функция работает несколько иначе (чем jsonb_set()), когда path in target является объектным ключом 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

Edit

Версия, которая устанавливает несколько ключей и значений:

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 (обновляет поле, если оно существует, вставляет, если он не существует). Вот вариант, который только 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$;

Обновить: теперь функции уплотняются.

  • 5
    Я попробовал вашу функцию plpgsql, но не уверен, как ее использовать - я вижу ошибку, когда пытаюсь select json_object_set_key((select data from test where data->>'b' = '2'), 'b', 'two'); сообщение об ошибке ERROR: could not determine polymorphic type because input has type "unknown"
  • 0
    @ user9645 Должен использоваться как select json_object_set_key('{"b":2}', 'b', 'two'::text) . Ваша проблема в том, что тип 'two' имеет тип unknown, если вы называете его литералом. То же самое с вызовом select to_json('two') . Эта функция принимает любой тип в качестве значения и преобразует его в json с помощью to_json() , но это стоит того, что этот тип не может быть неизвестен.
Показать ещё 25 комментариев
61

С 9.5 используйте jsonb_set -

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

где body - тип столбца jsonb.

  • 0
    Привет, почему я не могу использовать upper таким образом: update objects set body=jsonb_set(body, '{name}', upper('"Mary"'), true) where id=1; он не распознает, или как я могу добиться того же поведения? Спасибо
  • 0
    Если бы значение, которое я хочу установить, это подстрока из другого столбца, а не "Мэри", как бы я это сделал?
24

С Postgresql 9.5 это можно сделать с помощью following-

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

ИЛИ ЖЕ

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. Обновляет поле
  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)

Чтобы обновить поле внутри, не используйте оператор concat || , Вместо этого используйте jsonb_set. Что не просто:

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

Используя оператор concat для {c, c1}, например:

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

Удаляем {c, c2} и {c, c3}.

Для получения дополнительной мощности обратитесь к документации по функциям postgresql json. Можно было бы интересоваться #- -оператором, функцией jsonb_set а также функцией jsonb_insert.

  • 0
    и если мне нужно обновить два поля, то каков синтаксис?
  • 0
    если у меня есть столбец json с именем поля, как мне добавить поле фамилии в этот столбец
Показать ещё 1 комментарий
9

Чтобы основываться на ответах @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"}}
  • 0
    Очень полезно! Спасибо.
7
UPDATE test
SET data = data::jsonb - 'a' || '{"a":5}'::jsonb
WHERE data->>'b' = '2'

Это похоже на работу с PostgreSQL 9.5

  • 0
    У меня работает, насколько я понял, это удалить поле «а» из данных, а затем добавить поле «а» с новым значением. В моем случае значение «а» было основано на столбце. ОБНОВЛЕНИЕ теста SET data = data :: jsonb - 'a' || ('{"a": "' || myColumn || '"}') :: jsonb;
4

С 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__.__name__ != 'dict' and a.__class__.__name__ != 'list':
  raise plpy.Error("The json data must be an object or a string.")

if b.__class__.__name__ != '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__.__name__ == 'unicode':
      plpy.notice("Traversing '" + p + "'")
      if c.__class__.__name__ != 'dict':
        raise plpy.Error("  The value here is not a dictionary.")
      else:
        c = c[p]

    if p.__class__.__name__ == 'int':
      plpy.notice("Traversing " + str(p))
      if c.__class__.__name__ != '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;

Обратите внимание, что для предыдущего работодателя я написал набор функций C для управления данными JSON в виде текста (а не типа json или jsonb) для PostgreSQL 7, 8 и 9. Например, извлечение данных с 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.

2

Я написал для себя небольшую функцию, которая рекурсивно работает в 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

Несмотря на то, что следующее требование не удовлетворит этот запрос (функция json_object_agg недоступна в PostgreSQL 9.3), следующее может быть полезно для любого, кто ищет || оператор PostgreSQL 9.4, реализованный в предстоящем PostgreSQL 9.5:

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 );
1

Если ваш тип поля json, вы будете работать для вас.

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

Operator '-' удалить пару ключ/значение или элемент строки из левого операнда. Пара ключей/значений сопоставляется на основе их ключевого значения.

Оператор '||' объединить два значения jsonb в новое значение jsonb.

Поскольку это jsonb-операторы, вам просто нужно привести тип к :: jsonb

Дополнительная информация: Функции и операторы JSON

Вы можете прочитать мою заметку здесь

1

Следующий фрагмент 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

К сожалению, я ничего не нашел в документации, но вы можете использовать некоторое обходное решение, например, вы могли бы написать некоторую расширенную функцию.

Например, в 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';
  • 0
    Обидно, что Amazon RDS не поддерживает plpython3u!
  • 2
    Это value также потребует loads при установке нечисловых значений, таких как строки ( js[key] = loads(value) select json_update('{"a":"a"}', 'a', to_json('b')); -> {"a": "\"b\""} js[key] = loads(value) ) - В противном случае: select json_update('{"a":"a"}', 'a', to_json('b')); -> {"a": "\"b\""}
Показать ещё 1 комментарий
0

Если вы делаете этот запрос с помощью клиента языка программирования, например, из python pycopg2 или Node Postgres, убедитесь, что сначала вы разбираете новые данные в JSON.

Может показаться, что словарь python - это тот же, что и объект JSON, но он сначала не делает json.dumps в словаре.

Простой фрагмент питона:

def change_destination(self,parcel_id,destlatlng): query="UPDATE parcels SET destlatlng = '{}' WHERE parcel_id ={};".format(json.dumps(destlatlng), parcel_id) self.cursor.execute(query2) self.connection.commit()

0

Это работало для меня при попытке обновить поле типа строки.

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

Надеюсь, это поможет кому-то еще!

  • 0
    к сожалению, это переформатирует JSON так же, как манипуляции с помощью JSON-специфических функций
0

Вы можете также увеличивать клавиши атомарно внутри 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}

Undefined key → принимает начальное значение 0.

Для более подробного объяснения см. мой ответ здесь: https://stackoverflow.com/questions/25957937/how-to-increment-value-in-postgres-update-statement-on-json-key

Ещё вопросы

Сообщество Overcoder
Наверх
Меню