如何修改新的PostgreSQL JSON数据类型中的字段?

380

使用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...
24个回答

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;

请注意,我曾为以前的雇主编写了一组用于处理JSON数据的C函数,作为文本(而不是 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版本快得多。

3
您还可以像这样在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://dev59.com/sF8e5IYBdhLWcg3wPIV_#39076637


它在Postgres 14.1上运行非常良好。 - Mahdi

2
尽管以下内容无法满足此请求(PostgreSQL 9.3中没有可用的json_object_agg函数),但以下内容对于寻找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 );

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

这个解决方案是 jsonb_set 的替代品,即使 JSONB 列中有 NULL,它也可以正常工作。而 jsonb_set 只在对象存在时才能使用。

在下面的示例中,settings 是 Users 表上的一个 JSONB 列。

 UPDATE public."Users"
 SET settings = coalesce("settings", '{}')::jsonb || '{ "KeyToSet" : "ValueToSet" }'
 WHERE id=35877;

1
如果您想添加新字段,可以尝试以下方法:

typeorm code

let keyName:string = '{key2}'
let valueName:string = '"new_value"'

emailLog: () => "jsonb_set(cast(email_log as jsonb), '" + keyNAme + "','" + valueName + "'," + "true" + ")"

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

很遗憾,Amazon RDS不支持plpython3u! - dbau
2
value 在设置非数字值(如字符串)时也需要 loadsjs[key] = loads(value))- 否则:select json_update('{"a":"a"}', 'a', to_json('b')); -> {"a": "\"b\""} - hooblei
这个答案也可以修改一下,当值被设置为None时删除一个键:if value is None: del data[key] - Joshua Burns

1

UPDATE table_name SET attrs = jsonb_set(cast(attrs as jsonb), '{key}', '"new_value"', true) WHERE id = 'some_id';

这是我用的方法,attrs 是一个json类型的字段。首先将其转换为jsonb然后进行更新。

或者

UPDATE table_name SET attrs = jsonb_set(cast(attrs as jsonb), '{key}', '"new_value"', true) WHERE attrs->>key = 'old_value';

同样是更新操作,这里通过判断 attrs 字段中 key 对应的值是否等于 old_value 来实现。


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

您对这个解决方案有什么看法?

它将添加新值或更新现有值。

编辑:已编辑以使其能够处理 null 和空对象。

编辑2:已编辑以使其能够处理嵌套对象。

create or replace function updateJsonb(object1 json, object2 json)
returns jsonb
language plpgsql
as
$$
declare
    result jsonb;
    tempObj1 text;
    tempObj2 text;

begin
    tempObj1 = substr(object1::text, 2, length(object1::text) - 2); --remove the first { and last }
    tempObj2 = substr(object2::text, 2, length(object2::text) - 2); --remove the first { and last }

    IF object1::text != '{}' and object1::text != 'null' and object1::text != '[]' THEN
        result = ('{' || tempObj1 || ',' || tempObj2 || '}')::jsonb;
    ELSE
        result = ('{' || tempObj2 || '}')::jsonb;
    END IF;
    return result;
end;
$$;

用法:

update table_name
set data = updatejsonb(data, '{"test": "ok"}'::json)

我认为这个实现存在很高的安全漏洞潜在性。如果攻击者能够影响tempObj1tempObj2的内容,包括引号,那么他或她就可以通过在字符串中设置合适的值来修改对象的结构,因为代码会将生成的字符串重新解释为jsonb。 - Mikko Rantalainen

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接