Редактировать поля массива jsonb в postgresql - PullRequest
2 голосов
/ 04 марта 2020

У меня есть следующий jsonb в дБ:

[
    {
        "state": 2,
        "activity": "EJECUCIÓN",
        "final_date": "2020-02-24",
        "activity_id": 1,
        "current_days": -7,
        "initial_date": "2020-02-24",

    },
    {
        "state": 2,
        "activity": "REVISIÓN",
        "final_date": "2020-02-25",
        "activity_id": 2,
        "current_days": 0,
        "initial_date": "2020-02-25",

    },
    {
        "state": 2,    
        "activity": "RECEPCIÓN",
        "final_date": "2020-02-27",
        "activity_id": 4,
        "current_days": 0,
        "initial_date": "2020-02-27"


    } ]

Я запускаю следующий запрос для обновления поля current_days:

WITH activity_state     as  ( SELECT taex_id,('{'||index-1||',current_days}')::text[] as pathe ,
        ((task_activity->>'final_date')::date - current_date) as current_days,
        task_activity->'state' as state,
        task_activity->>'final_date' as final_date,
        task_activity->>'current_days' as curren   FROM task_executions,jsonb_array_elements(taex_activitygraph) with ordinality arr(task_activity,index)   WHERE task_activity->>'state' = '2'  )  

update task_executions SET  taex_activitygraph = jsonb_set(taex_activitygraph,activity_state.pathe,to_jsonb(current_days),true) FROM activity_state  WHERE task_executions.taex_id = activity_state.taex_id  AND activity_state.state = '2'

Но этот запрос обновляет только первый элемент массив JSON, который существует, остальные не изменяются под go, хотя и в первой части запроса.

( SELECT taex_id,('{'||index-1||',current_days}')::text[] as pathe ,
        ((task_activity->>'final_date')::date - current_date) as current_days,
        task_activity->'state' as state,
        task_activity->>'final_date' as final_date,
        task_activity->>'current_days' as curren   FROM task_executions,jsonb_array_elements(taex_activitygraph) with ordinality arr(task_activity,index)   WHERE task_activity->>'state' = '2'  )

Он приносит мне все элементы массива, которые должны быть обновлены, но второй часть, в которой они должны обновляться:

update task_executions SET  taex_activitygraph = jsonb_set(taex_activitygraph,activity_state.pathe,to_jsonb(current_days),true) FROM activity_state  WHERE task_executions.taex_id = activity_state.taex_id  AND activity_state.state = '2'

Просто обновите мне первый элемент.

1 Ответ

2 голосов
/ 05 марта 2020

Предполагая эту структуру и данные:

postgres=# \d task_executions 
               Table "public.task_executions"
       Column       | Type  | Collation | Nullable | Default 
--------------------+-------+-----------+----------+---------
 task_activitygraph | jsonb |           |          | 

postgres=# SELECT jsonb_pretty(task_activitygraph) FROM task_executions ;
             jsonb_pretty             
--------------------------------------
 [                                   +
     {                               +
         "state": 2,                 +
         "activity": "EJECUCIÓN",    +
         "final_date": "2020-02-24", +
         "activity_id": 1,           +
         "current_days": -7,         +
         "initial_date": "2020-02-24"+
     },                              +
     {                               +
         "state": 2,                 +
         "activity": "REVISIÓN",     +
         "final_date": "2020-02-25", +
         "activity_id": 2,           +
         "current_days": 0,          +
         "initial_date": "2020-02-25"+
     }                               +
 ]
(1 row)

... это ОБНОВЛЕНИЕ должно работать:

postgres=# UPDATE task_executions
SET task_activitygraph = (
  SELECT jsonb_agg(
    CASE WHEN elem->>'state' = '2'
    THEN 
      jsonb_set(
        elem,
        '{current_days}',
        to_jsonb((elem->>'final_date')::date - current_date)
      )
    ELSE
      elem
    END
  )
  FROM jsonb_array_elements(task_activitygraph) AS a(elem)
);
UPDATE 1

Документация: https://www.postgresql.org/docs/9.5/functions-json.html

Примечание: в транзакционных базах данных (где у вас много одновременных клиентов, и скорость обработки и эффективность хранения имеют значение), и если ваши объекты имеют фиксированную структуру, НЕ ХРАНИТЕ ваши данные как JSON. Вместо этого используйте реляционную модель данных.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...