Добавить узел / объект в существующий элемент jsonarray другого объекта, Postgresql 9.6 - PullRequest
0 голосов
/ 03 сентября 2018

У меня есть существующий JSON формата:

{
    "createdDate": "2017-06-29 07:32:18",
    "name": "Messi",    
    "lastName": "Tester",
    "addressDetails": {},    
    "products": [
        {
            "lasttimeOfResourceScreenVisit": "2017-08-20T20:29:20.681+05:30",
            "readArticles": [],
            "preferences": {
                "hasLoggedIntoProgramForFirstTime": false,
                "webLatestViewedProductLabelVersion": "",
                "progressTrackingStartDate": 1502443894189
            },
            "vcProductId": "a1G0S000000zvVq"            
        },
        {
            "lasttimeOfResourceScreenVisit": "2017-08-20T20:29:20.681+05:30",
            "readArticles": [],
            "preferences": {
                "hasLoggedIntoProgramForFirstTime": false,
                "webLatestViewedProductLabelVersion": "",
                "progressTrackingStartDate": 1502443894189
            },
            "vcProductId": "tala1G0S000000zvVqdkuylfguib9ytz",           
        }
    ],
    "lastModifiedDate": "2017-08-07 06:16:53",
    "patientId" : "1122334"
}

Я должен добавить новый узел в массиве продуктов

"primaryNumber" : {
   "phoneNumber" : "999876997900",
   "consentVersion" : "99999"
}

Этот узел добавляется на основе продукта и пациента из файла CSV. Для этого я создал следующую функцию, но я не могу обновить определенный индекс в массиве products.

что у меня до сих пор

CREATE OR REPLACE FUNCTION add_primaryNumber_product_node() RETURNS
VOID AS $$
DECLARE
patient_profile record;
 exception_message TEXT;
 patient_details json;
 products json;
 product json;
 primaryNumber jsonb;
 count SMALLINT;
 BEGIN
     --copy primary number for product and patient in a temporary table from csv file
    create table primaryNumber_temp(pcpPatientId varchar, vcProductId varchar, primaryNumber jsonb);
    copy primaryNumber_temp(pcpPatientId, vcProductId, primaryNumber) from '\path\PrimaryNumber.csv' delimiter ',' escape '"' csv header;

    FOR patient_profile IN SELECT * from patientprofile
    LOOP
        patient_details:=patient_profile.jdata;
        IF(patient_details->'products' IS NULL) THEN
            EXIT;
        END IF;
        products:=patient_details->>'products';
        count := 0;
        FOR product IN SELECT * FROM json_array_elements(products)
        LOOP
            IF(product->'primaryNumber' IS NULL)
            THEN
            SELECT primaryNumber_temp.primaryNumber into primaryNumber from primaryNumber_temp where vcProductId = product->>'vcProductId' and pcpPatientId = patient_details->>'pcpPatientId';
            raise notice 'Value primarynumber: %', primaryNumber;
                IF(primaryNumber is NOT NULL)
                THEN
                 raise notice 'Value index: %', count;
                product := product || '{"primaryNumber": {}}';
                raise notice 'value of product: %',product;
--                  UPDATE patientprofile SET jdata =jsonb_set(jdata,'{products}',jdata->'products'->count ||'{"primaryNumber": {}}') where id=1;

                    UPDATE patientprofile SET jdata =jsonb_set(jdata,'{products}', jdata->'products' || (jdata->'products'->0 || '{"primaryNumber": {}}')::jsonb , true)  where id=21;
                    UPDATE patientprofile SET jdata = jdata #- ('{products,0} ') where id=21;
--                      where patientprofile.id=patient_profile.id AND patientprofile.jdata->'products'->>'vcProductId' = product->>'vcProductId';

                    UPDATE patientprofile SET jdata =jsonb_set(jdata,'{products,primaryNumber}', ('{"phoneNumber":"917842672492"}')::jsonb , false)  where id = 21 and jdata->'products'->1->'primaryNumber' IS NOT NULL;
--                      where patientprofile.id=patient_profile.id AND patientprofile.jdata->'products'->>'vcProductId' = product->>'vcProductId';
                END IF;
            END IF;
            count := count + 1;
        END LOOP;
    END LOOP;
    DROP TABLE primaryNumber_temp;
    EXCEPTION WHEN OTHERS THEN
    GET STACKED DIAGNOSTICS exception_message := MESSAGE_TEXT;
    RAISE NOTICE 'Exception occurred in database function while adding primary number node patient profile json, exception: %', exception_message;
    END;
$$ LANGUAGE PLPGSQL;
select add_primaryNumber_product_node();

Копирование из csv в таблицу работает нормально, это просто обновление, с которым я застрял Есть мысли ..

...