Kana iyo yekubatanidza yakavharwa, ese matafura enguva pfupi anobviswa otomatiki, saka nemaoko DROP TABLE x hapana chinobuda kunze...
Kana uri kushanda kuburikidza pgbouncer mune transaction mode, ipapo dhatabhesi inoramba ichitenda kuti kubatana uku kuchiri kushanda, uye mairi iyi tafura yenguva pfupi ichiripo.
Naizvozvo, kuyedza kuigadzira zvakare, kubva kune yakasiyana yekubatanidza kune pgbouncer, zvinoguma nekukanganisa. Asi izvi zvinogona kudziviswa nekushandisa CREATE TEMPORARY TABLE IF NOT EXISTS x.
Ichokwadi, zviri nani kusaita izvi zvakadaro, nokuti ipapo iwe unogona "kamwe kamwe" kuwana ipapo data yakasara kubva kune "muridzi wekare". Pane kudaro, zviri nani kuverenga bhuku rekushandisa uye ona kuti kana uchigadzira tafura zvinokwanisika kuwedzera ON COMMIT DROP - ndiko kuti, kana kutengeserana kwapera, tafura inobviswa otomatiki.
Kusadzokorora
Nekuti iwo ndeechechete chekubatanidza, matafura enguva pfupi haana kudzokororwa. Asi izvi zvinobvisa kudiwa kwekurekodha kaviri data mumurwi + WAL, saka INSERT/UPDATE/DELETE mairi inokurumidza kukurumidza.
pamusoro ON COMMIT DROP Ndakatonyora pamusoro, inogadzira DROP TABLE, asi ne ON COMMIT DELETE ROWS mamiriro acho anonyanya kufadza - anogadzirwa pano TRUNCATE TABLE.
Asi usimbe ndiyo injini yekufambira mberi! Ndosaka gadzira tafura itsva "yakavakirwa pamuenzaniso" zvinogona kunge zviri nyore:
CREATE TEMPORARY TABLE import_table(
LIKE target_table
);
Sezvo iwe uchizogona kugadzira data rakawanda mutafura iyi, kutsvaga kuburikidza nayo hakuzombofa kwakakurumidza. Asi pane mhinduro yechinyakare kune iyi - indexes! Uye, hongu, tafura yenguva pfupi inogonawo kuva nema indexes.
Sezvo, kazhinji, ma index anodiwa achienderana nema indexes etafura inotarirwa, unogona kungonyora LIKE target_table INCLUDING INDEXES.
Kana uchidawo DEFAULT-values ββ(semuenzaniso, kuzadza iwo ekutanga makiyi kukosha), unogona kushandisa LIKE target_table INCLUDING DEFAULTS. Kana nyore - LIKE target_table INCLUDING ALL - makopi defaults, indexes, constraints, ...
Asi pano unofanira kunzwisisa kuti kana wakasika pinza tafura nekukasira nemaindex, ipapo iyo data inotora nguva yakareba kurodhakupfuura kana iwe ukatanga kuzadza zvese kumusoro, uye chete wozopeta ma indexes - tarisa kuti zvinozviita sei izvi semuenzaniso pg_dump.
-- Π²ΡΠ΅ Π΄Π΅Π»Π°Π΅ΠΌ Π² ΡΠ°ΠΌΠΊΠ°Ρ ΡΡΠ°Π½Π·Π°ΠΊΡΠΈΠΈ, ΡΡΠΎΠ±Ρ Π½ΠΈΠΊΡΠΎ Π½Π΅ Π²ΠΈΠ΄Π΅Π» "ΠΏΡΠΎΠΌΠ΅ΠΆΡΡΠΎΡΠ½ΡΡ " ΡΠΎΡΡΠΎΡΠ½ΠΈΠΉ
BEGIN;
-- ΡΠΎΠ·Π΄Π°Π΅ΠΌ Π²ΡΠ΅ΠΌΠ΅Π½Π½ΡΡ ΡΠ°Π±Π»ΠΈΡΡ Ρ ΠΈΠΌΠΏΠΎΡΡΠΈΡΡΠ΅ΠΌΡΠΌΠΈ Π΄Π°Π½Π½ΡΠΌΠΈ
CREATE TEMPORARY TABLE tmp(
LIKE dst INCLUDING INDEXES -- ΠΏΠΎ ΠΎΠ±ΡΠ°Π·Ρ ΠΈ ΠΏΠΎΠ΄ΠΎΠ±ΠΈΡ, Π²ΠΌΠ΅ΡΡΠ΅ Ρ ΠΈΠ½Π΄Π΅ΠΊΡΠ°ΠΌΠΈ
) ON COMMIT DROP; -- Π·Π° ΡΠ°ΠΌΠΊΠ°ΠΌΠΈ ΡΡΠ°Π½Π·Π°ΠΊΡΠΈΠΈ ΠΎΠ½Π° Π½Π°ΠΌ Π½Π΅ Π½ΡΠΆΠ½Π°
-- Π±ΡΡΡΡΠΎ-Π±ΡΡΡΡΠΎ Π²Π»ΠΈΠ²Π°Π΅ΠΌ Π½ΠΎΠ²ΡΠΉ ΠΎΠ±ΡΠ°Π· ΡΠ΅ΡΠ΅Π· COPY
COPY tmp FROM STDIN;
-- ...
-- .
-- ΡΠ΄Π°Π»ΡΠ΅ΠΌ ΠΎΡΡΡΡΡΡΠ²ΡΡΡΠΈΠ΅
DELETE FROM
dst D
USING
dst X
LEFT JOIN
tmp Y
USING(pk1, pk2) -- ΠΏΠΎΠ»Ρ ΠΏΠ΅ΡΠ²ΠΈΡΠ½ΠΎΠ³ΠΎ ΠΊΠ»ΡΡΠ°
WHERE
(D.pk1, D.pk2) = (X.pk1, X.pk2) AND
Y IS NOT DISTINCT FROM NULL; -- "Π°Π½ΡΠΈΠ΄ΠΆΠΎΠΉΠ½"
-- ΠΎΠ±Π½ΠΎΠ²Π»ΡΠ΅ΠΌ ΠΎΡΡΠ°Π²ΡΠΈΠ΅ΡΡ
UPDATE
dst D
SET
(f1, f2, f3) = (T.f1, T.f2, T.f3)
FROM
tmp T
WHERE
(D.pk1, D.pk2) = (T.pk1, T.pk2) AND
(D.f1, D.f2, D.f3) IS DISTINCT FROM (T.f1, T.f2, T.f3); -- Π½Π΅Π·Π°ΡΠ΅ΠΌ ΠΎΠ±Π½ΠΎΠ²Π»ΡΡΡ ΡΠΎΠ²ΠΏΠ°Π΄Π°ΡΡΠΈΠ΅
-- Π²ΡΡΠ°Π²Π»ΡΠ΅ΠΌ ΠΎΡΡΡΡΡΡΠ²ΡΡΡΠΈΠ΅
INSERT INTO
dst
SELECT
T.*
FROM
tmp T
LEFT JOIN
dst D
USING(pk1, pk2)
WHERE
D IS NOT DISTINCT FROM NULL;
COMMIT;
3.2. Import post-processing
MuKLADR imwe chete, marekodhi ese akachinjwa anofanirwa kufambiswazve kuburikidza nekugadzirisa-kumashure - akajairwa, mazwi akakosha anosimbiswa, uye kuderedzwa kune anodiwa zvimiro. Asi unoziva sei - chii chaizvo chachinjapasina kuomesa kodhi yekubatanidza, zvakanaka pasina kuibata zvachose?
Kana chete maitiro ako aine mukana wekunyora panguva yekuwiriranisa, saka unogona kushandisa chinokonzeresa chinozounganidza zvese shanduko kwatiri:
-- ΡΠ΅Π»Π΅Π²ΡΠ΅ ΡΠ°Π±Π»ΠΈΡΡ
CREATE TABLE kladr(...);
CREATE TABLE kladr_house(...);
-- ΡΠ°Π±Π»ΠΈΡΡ Ρ ΠΈΡΡΠΎΡΠΈΠ΅ΠΉ ΠΈΠ·ΠΌΠ΅Π½Π΅Π½ΠΈΠΉ
CREATE TABLE kladr$log(
ro kladr, -- ΡΡΡ Π»Π΅ΠΆΠ°Ρ ΡΠ΅Π»ΡΠ΅ ΠΎΠ±ΡΠ°Π·Ρ Π·Π°ΠΏΠΈΡΠ΅ΠΉ ΡΡΠ°ΡΠΎΠΉ/Π½ΠΎΠ²ΠΎΠΉ
rn kladr
);
CREATE TABLE kladr_house$log(
ro kladr_house,
rn kladr_house
);
-- ΠΎΠ±ΡΠ°Ρ ΡΡΠ½ΠΊΡΠΈΡ Π»ΠΎΠ³ΠΈΡΠΎΠ²Π°Π½ΠΈΡ ΠΈΠ·ΠΌΠ΅Π½Π΅Π½ΠΈΠΉ
CREATE OR REPLACE FUNCTION diff$log() RETURNS trigger AS $$
DECLARE
dst varchar = TG_TABLE_NAME || '$log';
stmt text = '';
BEGIN
-- ΠΏΡΠΎΠ²Π΅ΡΡΠ΅ΠΌ Π½Π΅ΠΎΠ±Ρ ΠΎΠ΄ΠΈΠΌΠΎΡΡΡ Π»ΠΎΠ³Π³ΠΈΡΠΎΠ²Π°Π½ΠΈΡ ΠΏΡΠΈ ΠΎΠ±Π½ΠΎΠ²Π»Π΅Π½ΠΈΠΈ Π·Π°ΠΏΠΈΡΠΈ
IF TG_OP = 'UPDATE' THEN
IF NEW IS NOT DISTINCT FROM OLD THEN
RETURN NEW;
END IF;
END IF;
-- ΡΠΎΠ·Π΄Π°Π΅ΠΌ Π·Π°ΠΏΠΈΡΡ Π»ΠΎΠ³Π°
stmt = 'INSERT INTO ' || dst::text || '(ro,rn)VALUES(';
CASE TG_OP
WHEN 'INSERT' THEN
EXECUTE stmt || 'NULL,$1)' USING NEW;
WHEN 'UPDATE' THEN
EXECUTE stmt || '$1,$2)' USING OLD, NEW;
WHEN 'DELETE' THEN
EXECUTE stmt || '$1,NULL)' USING OLD;
END CASE;
RETURN NEW;
END;
$$ LANGUAGE plpgsql;
Iye zvino isu tinogona kuisa zvinokonzeresa tisati tatanga kuwiriranisa (kana kuvagonesa kuburikidza ALTER TABLE ... ENABLE TRIGGER ...):
CREATE TRIGGER log
AFTER INSERT OR UPDATE OR DELETE
ON kladr
FOR EACH ROW
EXECUTE PROCEDURE diff$log();
CREATE TRIGGER log
AFTER INSERT OR UPDATE OR DELETE
ON kladr_house
FOR EACH ROW
EXECUTE PROCEDURE diff$log();
Pamusoro takafunga nyaya kana zvimiro zvedata zvekwakabva uye kwekuenda zvakafanana. Asi ko kana iyo yekukwirisa kubva kune yekunze sisitimu ine fomati yakasiyana neyekuchengetera chimiro mune yedu database?
-- ΠΏΡΠΎΡΡΠ°Π²Π»ΡΠ΅ΠΌ Π² ΡΠ°Π±Π»ΠΈΡΠ΅ ΠΈΠΌΠΏΠΎΡΡΠ° ID ΡΠΆΠ΅ ΡΡΡΠ΅ΡΡΠ²ΡΡΡΠΈΡ Π·Π°ΠΏΠΈΡΠ΅ΠΉ
UPDATE
client_import T
SET
client_id = D.client_id
FROM
client D
WHERE
T.inn = D.inn; -- unique key
-- Π²ΡΡΠ°Π²Π»ΡΠ΅ΠΌ ΠΎΡΡΡΡΡΡΠ²ΠΎΠ²Π°Π²ΡΠΈΠ΅ Π·Π°ΠΏΠΈΡΠΈ ΠΈ ΠΏΡΠΎΡΡΠ°Π²Π»ΡΠ΅ΠΌ ΠΈΡ ID
WITH ins AS (
INSERT INTO client(
inn
, name
)
SELECT
inn
, name
FROM
client_import
WHERE
client_id IS NULL -- Π΅ΡΠ»ΠΈ ID Π½Π΅ ΠΏΡΠΎΡΡΠ°Π²ΠΈΠ»ΡΡ
RETURNING *
)
UPDATE
client_import T
SET
client_id = D.client_id
FROM
ins D
WHERE
T.inn = D.inn; -- unique key
-- ΠΏΡΠΎΡΡΠ°Π²Π»ΡΠ΅ΠΌ ID ΠΊΠ»ΠΈΠ΅Π½ΡΠΎΠ² Ρ Π·Π°ΠΏΠΈΡΠ΅ΠΉ ΡΡΠ΅ΡΠΎΠ²
UPDATE
invoice_import T
SET
client_id = D.client_id
FROM
client_import D
WHERE
T.client_inn = D.inn; -- ΠΏΡΠΈΠΊΠ»Π°Π΄Π½ΠΎΠΉ ΠΊΠ»ΡΡ