PostgreSQL
 sql >> база данни >  >> RDS >> PostgreSQL

Как да променя полета в новия PostgreSQL JSON тип данни?

Актуализиране :С PostgreSQL 9.5 има някои jsonb функционалност за манипулиране в самия PostgreSQL (но няма такава за json; за манипулиране на json са необходими cast стойности).

Обединяване на 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() ), когато path в 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

Редактиране :

Версия, която задава множество ключове и стойности:

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

Актуализирано:Добавена е функция за замяна на ключа на съществуващо 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;

Актуализиране :функциите са уплътнени сега.



  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. PostgreSQL - клауза GROUP BY

  2. psql - запазване на резултатите от командата във файл

  3. Как да актуализирам избрани редове със стойности от CSV файл в Postgres?

  4. Конфигурация на Puma Cluster на Heroku

  5. Как да избегнем PostgreSQL Cloud Vendor Lock-in