mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Handle heap rewrites even better in logical decoding
Logical decoding should not publish anything about tables created as
part of a heap rewrite during DDL.  Those tables don't exist externally,
so consumers of logical decoding cannot do anything sensible with that
information.  In ab28feae2b, we worked
around this for built-in logical replication, but that was hack.
This is a more proper fix: We mark such transient heaps using the new
field pg_class.relwrite, linking to the original relation OID.  By
default, we ignore them in logical decoding before they get to the
output plugin.  Optionally, a plugin can register their interest in
getting such changes, if they handle DDL specially, in which case the
new field will help them get information about the actual table.
Reviewed-by: Craig Ringer <craig@2ndquadrant.com>
			
			
This commit is contained in:
		| @@ -10,7 +10,7 @@ step s1_insert_tbl1: INSERT INTO tbl1 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl2_float: ALTER TABLE tbl2 ALTER COLUMN val2 TYPE float; | step s2_alter_tbl2_float: ALTER TABLE tbl2 ALTER COLUMN val2 TYPE float; | ||||||
| step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -32,16 +32,13 @@ step s2_alter_tbl1_float: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE float; <waitin | |||||||
| step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl1_float: <... completed> | step s2_alter_tbl1_float: <... completed> | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| COMMIT          | COMMIT          | ||||||
| BEGIN           |  | ||||||
| table public.pg_temp: INSERT: val1[integer]:1 val2[double precision]:1 |  | ||||||
| COMMIT          |  | ||||||
| ?column?        | ?column?        | ||||||
|  |  | ||||||
| stop            | stop            | ||||||
| @@ -56,7 +53,7 @@ step s1_insert_tbl1: INSERT INTO tbl1 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl2_char: ALTER TABLE tbl2 ALTER COLUMN val2 TYPE character varying; | step s2_alter_tbl2_char: ALTER TABLE tbl2 ALTER COLUMN val2 TYPE character varying; | ||||||
| step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -78,16 +75,13 @@ step s2_alter_tbl1_char: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE character varyi | |||||||
| step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl1_char: <... completed> | step s2_alter_tbl1_char: <... completed> | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| COMMIT          | COMMIT          | ||||||
| BEGIN           |  | ||||||
| table public.pg_temp: INSERT: val1[integer]:1 val2[character varying]:'1' |  | ||||||
| COMMIT          |  | ||||||
| ?column?        | ?column?        | ||||||
|  |  | ||||||
| stop            | stop            | ||||||
| @@ -103,16 +97,13 @@ step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl1_float: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE float; <waiting ...> | step s2_alter_tbl1_float: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE float; <waiting ...> | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl1_float: <... completed> | step s2_alter_tbl1_float: <... completed> | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| COMMIT          | COMMIT          | ||||||
| BEGIN           |  | ||||||
| table public.pg_temp: INSERT: val1[integer]:1 val2[double precision]:1 |  | ||||||
| COMMIT          |  | ||||||
| ?column?        | ?column?        | ||||||
|  |  | ||||||
| stop            | stop            | ||||||
| @@ -128,16 +119,13 @@ step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl1_char: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE character varying; <waiting ...> | step s2_alter_tbl1_char: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE character varying; <waiting ...> | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl1_char: <... completed> | step s2_alter_tbl1_char: <... completed> | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| COMMIT          | COMMIT          | ||||||
| BEGIN           |  | ||||||
| table public.pg_temp: INSERT: val1[integer]:1 val2[character varying]:'1' |  | ||||||
| COMMIT          |  | ||||||
| ?column?        | ?column?        | ||||||
|  |  | ||||||
| stop            | stop            | ||||||
| @@ -154,16 +142,13 @@ step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl1_float: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE float; <waiting ...> | step s2_alter_tbl1_float: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE float; <waiting ...> | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl1_float: <... completed> | step s2_alter_tbl1_float: <... completed> | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| table public.tbl2: INSERT: val1[integer]:1 val2[double precision]:1 | table public.tbl2: INSERT: val1[integer]:1 val2[double precision]:1 | ||||||
| COMMIT          | COMMIT          | ||||||
| BEGIN           |  | ||||||
| table public.pg_temp: INSERT: val1[integer]:1 val2[double precision]:1 |  | ||||||
| COMMIT          |  | ||||||
| ?column?        | ?column?        | ||||||
|  |  | ||||||
| stop            | stop            | ||||||
| @@ -180,16 +165,13 @@ step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl1_char: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE character varying; <waiting ...> | step s2_alter_tbl1_char: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE character varying; <waiting ...> | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl1_char: <... completed> | step s2_alter_tbl1_char: <... completed> | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| table public.tbl2: INSERT: val1[integer]:1 val2[character varying]:'1' | table public.tbl2: INSERT: val1[integer]:1 val2[character varying]:'1' | ||||||
| COMMIT          | COMMIT          | ||||||
| BEGIN           |  | ||||||
| table public.pg_temp: INSERT: val1[integer]:1 val2[character varying]:'1' |  | ||||||
| COMMIT          |  | ||||||
| ?column?        | ?column?        | ||||||
|  |  | ||||||
| stop            | stop            | ||||||
| @@ -205,7 +187,7 @@ step s1_insert_tbl1: INSERT INTO tbl1 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl2_text: ALTER TABLE tbl2 ALTER COLUMN val2 TYPE text; | step s2_alter_tbl2_text: ALTER TABLE tbl2 ALTER COLUMN val2 TYPE text; | ||||||
| step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -229,16 +211,13 @@ step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl1_char: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE character varying; <waiting ...> | step s2_alter_tbl1_char: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE character varying; <waiting ...> | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl1_char: <... completed> | step s2_alter_tbl1_char: <... completed> | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | table public.tbl1: INSERT: val1[integer]:1 val2[integer]:1 | ||||||
| table public.tbl2: INSERT: val1[integer]:1 val2[text]:'1' | table public.tbl2: INSERT: val1[integer]:1 val2[text]:'1' | ||||||
| COMMIT          | COMMIT          | ||||||
| BEGIN           |  | ||||||
| table public.pg_temp: INSERT: val1[integer]:1 val2[character varying]:'1' |  | ||||||
| COMMIT          |  | ||||||
| ?column?        | ?column?        | ||||||
|  |  | ||||||
| stop            | stop            | ||||||
| @@ -254,7 +233,7 @@ step s2_alter_tbl2_boolean: ALTER TABLE tbl2 ALTER COLUMN val2 TYPE boolean; | |||||||
| ERROR:  column "val2" cannot be cast automatically to type boolean | ERROR:  column "val2" cannot be cast automatically to type boolean | ||||||
| step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -279,7 +258,7 @@ step s2_alter_tbl1_boolean: ALTER TABLE tbl1 ALTER COLUMN val2 TYPE boolean; <wa | |||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl1_boolean: <... completed> | step s2_alter_tbl1_boolean: <... completed> | ||||||
| error in steps s1_commit s2_alter_tbl1_boolean: ERROR:  column "val2" cannot be cast automatically to type boolean | error in steps s1_commit s2_alter_tbl1_boolean: ERROR:  column "val2" cannot be cast automatically to type boolean | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -300,7 +279,7 @@ step s1_insert_tbl1: INSERT INTO tbl1 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl2_add_int: ALTER TABLE tbl2 ADD COLUMN val3 INTEGER; | step s2_alter_tbl2_add_int: ALTER TABLE tbl2 ADD COLUMN val3 INTEGER; | ||||||
| step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -324,7 +303,7 @@ step s1_begin: BEGIN; | |||||||
| step s2_alter_tbl2_add_int: ALTER TABLE tbl2 ADD COLUMN val3 INTEGER; | step s2_alter_tbl2_add_int: ALTER TABLE tbl2 ADD COLUMN val3 INTEGER; | ||||||
| step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -348,7 +327,7 @@ step s1_insert_tbl1: INSERT INTO tbl1 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl2_add_float: ALTER TABLE tbl2 ADD COLUMN val3 FLOAT; | step s2_alter_tbl2_add_float: ALTER TABLE tbl2 ADD COLUMN val3 FLOAT; | ||||||
| step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -372,7 +351,7 @@ step s1_begin: BEGIN; | |||||||
| step s2_alter_tbl2_add_float: ALTER TABLE tbl2 ADD COLUMN val3 FLOAT; | step s2_alter_tbl2_add_float: ALTER TABLE tbl2 ADD COLUMN val3 FLOAT; | ||||||
| step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -396,7 +375,7 @@ step s1_insert_tbl1: INSERT INTO tbl1 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl2_add_char: ALTER TABLE tbl2 ADD COLUMN val3 character varying; | step s2_alter_tbl2_add_char: ALTER TABLE tbl2 ADD COLUMN val3 character varying; | ||||||
| step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -420,7 +399,7 @@ step s1_begin: BEGIN; | |||||||
| step s2_alter_tbl2_add_char: ALTER TABLE tbl2 ADD COLUMN val3 character varying; | step s2_alter_tbl2_add_char: ALTER TABLE tbl2 ADD COLUMN val3 character varying; | ||||||
| step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -445,7 +424,7 @@ step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | |||||||
| step s2_alter_tbl2_drop_3rd_col: ALTER TABLE tbl2 DROP COLUMN val3; <waiting ...> | step s2_alter_tbl2_drop_3rd_col: ALTER TABLE tbl2 DROP COLUMN val3; <waiting ...> | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl2_drop_3rd_col: <... completed> | step s2_alter_tbl2_drop_3rd_col: <... completed> | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -468,7 +447,7 @@ step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | |||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl2_drop_3rd_col: <... completed> | step s2_alter_tbl2_drop_3rd_col: <... completed> | ||||||
| step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -493,7 +472,7 @@ step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | |||||||
| step s2_alter_tbl2_drop_3rd_col: ALTER TABLE tbl2 DROP COLUMN val3; <waiting ...> | step s2_alter_tbl2_drop_3rd_col: ALTER TABLE tbl2 DROP COLUMN val3; <waiting ...> | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl2_drop_3rd_col: <... completed> | step s2_alter_tbl2_drop_3rd_col: <... completed> | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -506,7 +485,7 @@ step s2_alter_tbl2_3rd_char: ALTER TABLE tbl2 ALTER COLUMN val3 TYPE character v | |||||||
| step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl2_3rd_char: <... completed> | step s2_alter_tbl2_3rd_char: <... completed> | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -515,14 +494,9 @@ table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 val3[text]:'1' | |||||||
| COMMIT          | COMMIT          | ||||||
| step s2_alter_tbl2_3rd_int: ALTER TABLE tbl2 ALTER COLUMN val3 TYPE int USING val3::integer; | step s2_alter_tbl2_3rd_int: ALTER TABLE tbl2 ALTER COLUMN val3 TYPE int USING val3::integer; | ||||||
| step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           |  | ||||||
| table public.pg_temp: INSERT: val1[integer]:1 val2[integer]:1 val3[integer]:null |  | ||||||
| table public.pg_temp: INSERT: val1[integer]:1 val2[integer]:1 val3[integer]:1 |  | ||||||
| table public.pg_temp: INSERT: val1[integer]:1 val2[integer]:1 val3[integer]:1 |  | ||||||
| COMMIT          |  | ||||||
| BEGIN           | BEGIN           | ||||||
| table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 val3[integer]:1 | table public.tbl2: INSERT: val1[integer]:1 val2[integer]:1 val3[integer]:1 | ||||||
| COMMIT          | COMMIT          | ||||||
| @@ -544,7 +518,7 @@ step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | |||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl2_3rd_text: <... completed> | step s2_alter_tbl2_3rd_text: <... completed> | ||||||
| step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -573,7 +547,7 @@ step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | |||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl2_3rd_char: <... completed> | step s2_alter_tbl2_3rd_char: <... completed> | ||||||
| step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -601,7 +575,7 @@ step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | |||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl2_drop_3rd_col: ALTER TABLE tbl2 DROP COLUMN val3; | step s2_alter_tbl2_drop_3rd_col: ALTER TABLE tbl2 DROP COLUMN val3; | ||||||
| step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -628,7 +602,7 @@ step s1_insert_tbl2_3col: INSERT INTO tbl2 (val1, val2, val3) VALUES (1, 1, 1); | |||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_alter_tbl2_drop_3rd_col: ALTER TABLE tbl2 DROP COLUMN val3; | step s2_alter_tbl2_drop_3rd_col: ALTER TABLE tbl2 DROP COLUMN val3; | ||||||
| step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | step s1_insert_tbl2: INSERT INTO tbl2 (val1, val2) VALUES (1, 1); | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
| @@ -653,7 +627,7 @@ step s1_insert_tbl1: INSERT INTO tbl1 (val1, val2) VALUES (1, 1); | |||||||
| step s2_alter_tbl2_drop_3rd_col: ALTER TABLE tbl2 DROP COLUMN val3; | step s2_alter_tbl2_drop_3rd_col: ALTER TABLE tbl2 DROP COLUMN val3; | ||||||
| step s1_insert_tbl1: INSERT INTO tbl1 (val1, val2) VALUES (1, 1); | step s1_insert_tbl1: INSERT INTO tbl1 (val1, val2) VALUES (1, 1); | ||||||
| step s1_commit: COMMIT; | step s1_commit: COMMIT; | ||||||
| step s2_get_changes: SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | step s2_get_changes: SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
| data            | data            | ||||||
|  |  | ||||||
| BEGIN           | BEGIN           | ||||||
|   | |||||||
| @@ -117,11 +117,11 @@ SELECT data FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'inc | |||||||
| (22 rows) | (22 rows) | ||||||
|  |  | ||||||
| ALTER TABLE replication_example ALTER COLUMN somenum TYPE int4 USING (somenum::int4); | ALTER TABLE replication_example ALTER COLUMN somenum TYPE int4 USING (somenum::int4); | ||||||
| -- throw away changes, they contain oids | -- check that this doesn't produce any changes from the heap rewrite | ||||||
| SELECT count(data) FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | SELECT count(data) FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
|  count  |  count  | ||||||
| ------- | ------- | ||||||
|     12 |      0 | ||||||
| (1 row) | (1 row) | ||||||
|  |  | ||||||
| INSERT INTO replication_example(somedata, somenum) VALUES (5, 1); | INSERT INTO replication_example(somedata, somenum) VALUES (5, 1); | ||||||
| @@ -192,16 +192,20 @@ SELECT data FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'inc | |||||||
|  COMMIT |  COMMIT | ||||||
| (33 rows) | (33 rows) | ||||||
|  |  | ||||||
| -- hide changes bc of oid visible in full table rewrites |  | ||||||
| CREATE TABLE tr_unique(id2 serial unique NOT NULL, data int); | CREATE TABLE tr_unique(id2 serial unique NOT NULL, data int); | ||||||
| INSERT INTO tr_unique(data) VALUES(10); | INSERT INTO tr_unique(data) VALUES(10); | ||||||
| ALTER TABLE tr_unique RENAME TO tr_pkey; | ALTER TABLE tr_unique RENAME TO tr_pkey; | ||||||
| ALTER TABLE tr_pkey ADD COLUMN id serial primary key; | ALTER TABLE tr_pkey ADD COLUMN id serial primary key; | ||||||
| SELECT count(data) FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | SELECT data FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1', 'include-rewrites', '1'); | ||||||
|  count  |                                     data                                      | ||||||
| ------- | ----------------------------------------------------------------------------- | ||||||
|      6 |  BEGIN | ||||||
| (1 row) |  table public.tr_unique: INSERT: id2[integer]:1 data[integer]:10 | ||||||
|  |  COMMIT | ||||||
|  |  BEGIN | ||||||
|  |  table public.tr_pkey: INSERT: id2[integer]:1 data[integer]:10 id[integer]:1 | ||||||
|  |  COMMIT | ||||||
|  | (6 rows) | ||||||
|  |  | ||||||
| INSERT INTO tr_pkey(data) VALUES(1); | INSERT INTO tr_pkey(data) VALUES(1); | ||||||
| --show deletion with primary key | --show deletion with primary key | ||||||
|   | |||||||
| @@ -53,7 +53,7 @@ step "s2_alter_tbl2_3rd_char" { ALTER TABLE tbl2 ALTER COLUMN val3 TYPE characte | |||||||
| step "s2_alter_tbl2_3rd_text" { ALTER TABLE tbl2 ALTER COLUMN val3 TYPE text; } | step "s2_alter_tbl2_3rd_text" { ALTER TABLE tbl2 ALTER COLUMN val3 TYPE text; } | ||||||
| step "s2_alter_tbl2_3rd_int" { ALTER TABLE tbl2 ALTER COLUMN val3 TYPE int USING val3::integer; } | step "s2_alter_tbl2_3rd_int" { ALTER TABLE tbl2 ALTER COLUMN val3 TYPE int USING val3::integer; } | ||||||
|  |  | ||||||
| step "s2_get_changes" { SELECT regexp_replace(data, 'temp_\d+', 'temp') AS data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); } | step "s2_get_changes" { SELECT data FROM pg_logical_slot_get_changes('isolation_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -67,7 +67,7 @@ INSERT INTO replication_example(somedata, somenum) VALUES (4, 1); | |||||||
| SELECT data FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | SELECT data FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
|  |  | ||||||
| ALTER TABLE replication_example ALTER COLUMN somenum TYPE int4 USING (somenum::int4); | ALTER TABLE replication_example ALTER COLUMN somenum TYPE int4 USING (somenum::int4); | ||||||
| -- throw away changes, they contain oids | -- check that this doesn't produce any changes from the heap rewrite | ||||||
| SELECT count(data) FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | SELECT count(data) FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
|  |  | ||||||
| INSERT INTO replication_example(somedata, somenum) VALUES (5, 1); | INSERT INTO replication_example(somedata, somenum) VALUES (5, 1); | ||||||
| @@ -93,12 +93,11 @@ COMMIT; | |||||||
| /* display results */ | /* display results */ | ||||||
| SELECT data FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | SELECT data FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | ||||||
|  |  | ||||||
| -- hide changes bc of oid visible in full table rewrites |  | ||||||
| CREATE TABLE tr_unique(id2 serial unique NOT NULL, data int); | CREATE TABLE tr_unique(id2 serial unique NOT NULL, data int); | ||||||
| INSERT INTO tr_unique(data) VALUES(10); | INSERT INTO tr_unique(data) VALUES(10); | ||||||
| ALTER TABLE tr_unique RENAME TO tr_pkey; | ALTER TABLE tr_unique RENAME TO tr_pkey; | ||||||
| ALTER TABLE tr_pkey ADD COLUMN id serial primary key; | ALTER TABLE tr_pkey ADD COLUMN id serial primary key; | ||||||
| SELECT count(data) FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1'); | SELECT data FROM pg_logical_slot_get_changes('regression_slot', NULL, NULL, 'include-xids', '0', 'skip-empty-xacts', '1', 'include-rewrites', '1'); | ||||||
|  |  | ||||||
| INSERT INTO tr_pkey(data) VALUES(1); | INSERT INTO tr_pkey(data) VALUES(1); | ||||||
| --show deletion with primary key | --show deletion with primary key | ||||||
|   | |||||||
| @@ -101,6 +101,7 @@ pg_decode_startup(LogicalDecodingContext *ctx, OutputPluginOptions *opt, | |||||||
| 	ctx->output_plugin_private = data; | 	ctx->output_plugin_private = data; | ||||||
|  |  | ||||||
| 	opt->output_type = OUTPUT_PLUGIN_TEXTUAL_OUTPUT; | 	opt->output_type = OUTPUT_PLUGIN_TEXTUAL_OUTPUT; | ||||||
|  | 	opt->receive_rewrites = false; | ||||||
|  |  | ||||||
| 	foreach(option, ctx->output_plugin_options) | 	foreach(option, ctx->output_plugin_options) | ||||||
| 	{ | 	{ | ||||||
| @@ -166,6 +167,17 @@ pg_decode_startup(LogicalDecodingContext *ctx, OutputPluginOptions *opt, | |||||||
| 						 errmsg("could not parse value \"%s\" for parameter \"%s\"", | 						 errmsg("could not parse value \"%s\" for parameter \"%s\"", | ||||||
| 								strVal(elem->arg), elem->defname))); | 								strVal(elem->arg), elem->defname))); | ||||||
| 		} | 		} | ||||||
|  | 		else if (strcmp(elem->defname, "include-rewrites") == 0) | ||||||
|  | 		{ | ||||||
|  |  | ||||||
|  | 			if (elem->arg == NULL) | ||||||
|  | 				continue; | ||||||
|  | 			else if (!parse_bool(strVal(elem->arg), &opt->receive_rewrites)) | ||||||
|  | 				ereport(ERROR, | ||||||
|  | 						(errcode(ERRCODE_INVALID_PARAMETER_VALUE), | ||||||
|  | 						 errmsg("could not parse value \"%s\" for parameter \"%s\"", | ||||||
|  | 								strVal(elem->arg), elem->defname))); | ||||||
|  | 		} | ||||||
| 		else | 		else | ||||||
| 		{ | 		{ | ||||||
| 			ereport(ERROR, | 			ereport(ERROR, | ||||||
| @@ -412,6 +424,8 @@ pg_decode_change(LogicalDecodingContext *ctx, ReorderBufferTXN *txn, | |||||||
| 						   quote_qualified_identifier( | 						   quote_qualified_identifier( | ||||||
| 													  get_namespace_name( | 													  get_namespace_name( | ||||||
| 																		 get_rel_namespace(RelationGetRelid(relation))), | 																		 get_rel_namespace(RelationGetRelid(relation))), | ||||||
|  | 													  class_form->relrewrite ? | ||||||
|  | 													  get_rel_name(class_form->relrewrite) : | ||||||
| 													  NameStr(class_form->relname))); | 													  NameStr(class_form->relname))); | ||||||
| 	appendStringInfoChar(ctx->out, ':'); | 	appendStringInfoChar(ctx->out, ':'); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1923,6 +1923,18 @@ SCRAM-SHA-256$<replaceable><iteration count></replaceable>:<replaceable>&l | |||||||
|       <entry>True if table is a partition</entry> |       <entry>True if table is a partition</entry> | ||||||
|      </row> |      </row> | ||||||
|  |  | ||||||
|  |      <row> | ||||||
|  |       <entry><structfield>relrewrite</structfield></entry> | ||||||
|  |       <entry><type>oid</type></entry> | ||||||
|  |       <entry><literal><link linkend="catalog-pg-class"><structname>pg_class</structname></link>.oid</literal></entry> | ||||||
|  |       <entry> | ||||||
|  |        For new relations being written during a DDL operation that requires a | ||||||
|  |        table rewrite, this contains the OID of the original relation; | ||||||
|  |        otherwise 0.  That state is only visible internally; this field should | ||||||
|  |        never contain anything other than 0 for a user-visible relation. | ||||||
|  |       </entry> | ||||||
|  |      </row> | ||||||
|  |  | ||||||
|      <row> |      <row> | ||||||
|       <entry><structfield>relfrozenxid</structfield></entry> |       <entry><structfield>relfrozenxid</structfield></entry> | ||||||
|       <entry><type>xid</type></entry> |       <entry><type>xid</type></entry> | ||||||
|   | |||||||
| @@ -486,12 +486,17 @@ typedef void (*LogicalDecodeStartupCB) (struct LogicalDecodingContext *ctx, | |||||||
| typedef struct OutputPluginOptions | typedef struct OutputPluginOptions | ||||||
| { | { | ||||||
|     OutputPluginOutputType output_type; |     OutputPluginOutputType output_type; | ||||||
|  |     bool        receive_rewrites; | ||||||
| } OutputPluginOptions; | } OutputPluginOptions; | ||||||
| </programlisting> | </programlisting> | ||||||
|       <literal>output_type</literal> has to either be set to |       <literal>output_type</literal> has to either be set to | ||||||
|       <literal>OUTPUT_PLUGIN_TEXTUAL_OUTPUT</literal> |       <literal>OUTPUT_PLUGIN_TEXTUAL_OUTPUT</literal> | ||||||
|       or <literal>OUTPUT_PLUGIN_BINARY_OUTPUT</literal>. See also |       or <literal>OUTPUT_PLUGIN_BINARY_OUTPUT</literal>. See also | ||||||
|       <xref linkend="logicaldecoding-output-mode"/>. |       <xref linkend="logicaldecoding-output-mode"/>. | ||||||
|  |       If <literal>receive_rewrites</literal> is true, the output plugin will | ||||||
|  |       also be called for changes made by heap rewrites during certain DDL | ||||||
|  |       operations.  These are of interest to plugins that handle DDL | ||||||
|  |       replication, but they require special handling. | ||||||
|      </para> |      </para> | ||||||
|  |  | ||||||
|      <para> |      <para> | ||||||
|   | |||||||
| @@ -257,6 +257,7 @@ Boot_CreateStmt: | |||||||
| 													  false, | 													  false, | ||||||
| 													  true, | 													  true, | ||||||
| 													  false, | 													  false, | ||||||
|  | 													  InvalidOid, | ||||||
| 													  NULL); | 													  NULL); | ||||||
| 						elog(DEBUG4, "relation created with OID %u", id); | 						elog(DEBUG4, "relation created with OID %u", id); | ||||||
| 					} | 					} | ||||||
|   | |||||||
| @@ -806,6 +806,7 @@ InsertPgClassTuple(Relation pg_class_desc, | |||||||
| 	values[Anum_pg_class_relispopulated - 1] = BoolGetDatum(rd_rel->relispopulated); | 	values[Anum_pg_class_relispopulated - 1] = BoolGetDatum(rd_rel->relispopulated); | ||||||
| 	values[Anum_pg_class_relreplident - 1] = CharGetDatum(rd_rel->relreplident); | 	values[Anum_pg_class_relreplident - 1] = CharGetDatum(rd_rel->relreplident); | ||||||
| 	values[Anum_pg_class_relispartition - 1] = BoolGetDatum(rd_rel->relispartition); | 	values[Anum_pg_class_relispartition - 1] = BoolGetDatum(rd_rel->relispartition); | ||||||
|  | 	values[Anum_pg_class_relrewrite - 1] = ObjectIdGetDatum(rd_rel->relrewrite); | ||||||
| 	values[Anum_pg_class_relfrozenxid - 1] = TransactionIdGetDatum(rd_rel->relfrozenxid); | 	values[Anum_pg_class_relfrozenxid - 1] = TransactionIdGetDatum(rd_rel->relfrozenxid); | ||||||
| 	values[Anum_pg_class_relminmxid - 1] = MultiXactIdGetDatum(rd_rel->relminmxid); | 	values[Anum_pg_class_relminmxid - 1] = MultiXactIdGetDatum(rd_rel->relminmxid); | ||||||
| 	if (relacl != (Datum) 0) | 	if (relacl != (Datum) 0) | ||||||
| @@ -1038,6 +1039,7 @@ heap_create_with_catalog(const char *relname, | |||||||
| 						 bool use_user_acl, | 						 bool use_user_acl, | ||||||
| 						 bool allow_system_table_mods, | 						 bool allow_system_table_mods, | ||||||
| 						 bool is_internal, | 						 bool is_internal, | ||||||
|  | 						 Oid relrewrite, | ||||||
| 						 ObjectAddress *typaddress) | 						 ObjectAddress *typaddress) | ||||||
| { | { | ||||||
| 	Relation	pg_class_desc; | 	Relation	pg_class_desc; | ||||||
| @@ -1176,6 +1178,8 @@ heap_create_with_catalog(const char *relname, | |||||||
|  |  | ||||||
| 	Assert(relid == RelationGetRelid(new_rel_desc)); | 	Assert(relid == RelationGetRelid(new_rel_desc)); | ||||||
|  |  | ||||||
|  | 	new_rel_desc->rd_rel->relrewrite = relrewrite; | ||||||
|  |  | ||||||
| 	/* | 	/* | ||||||
| 	 * Decide whether to create an array type over the relation's rowtype. We | 	 * Decide whether to create an array type over the relation's rowtype. We | ||||||
| 	 * do not create any array types for system catalogs (ie, those made | 	 * do not create any array types for system catalogs (ie, those made | ||||||
|   | |||||||
| @@ -279,6 +279,7 @@ create_toast_table(Relation rel, Oid toastOid, Oid toastIndexOid, | |||||||
| 										   false, | 										   false, | ||||||
| 										   true, | 										   true, | ||||||
| 										   true, | 										   true, | ||||||
|  | 										   InvalidOid, | ||||||
| 										   NULL); | 										   NULL); | ||||||
| 	Assert(toast_relid != InvalidOid); | 	Assert(toast_relid != InvalidOid); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -692,6 +692,7 @@ make_new_heap(Oid OIDOldHeap, Oid NewTableSpace, char relpersistence, | |||||||
| 										  false, | 										  false, | ||||||
| 										  true, | 										  true, | ||||||
| 										  true, | 										  true, | ||||||
|  | 										  OIDOldHeap, | ||||||
| 										  NULL); | 										  NULL); | ||||||
| 	Assert(OIDNewHeap != InvalidOid); | 	Assert(OIDNewHeap != InvalidOid); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -764,6 +764,7 @@ DefineRelation(CreateStmt *stmt, char relkind, Oid ownerId, | |||||||
| 										  true, | 										  true, | ||||||
| 										  allowSystemTableMods, | 										  allowSystemTableMods, | ||||||
| 										  false, | 										  false, | ||||||
|  | 										  InvalidOid, | ||||||
| 										  typaddress); | 										  typaddress); | ||||||
|  |  | ||||||
| 	/* Store inheritance information for new rel. */ | 	/* Store inheritance information for new rel. */ | ||||||
|   | |||||||
| @@ -317,6 +317,8 @@ CreateInitDecodingContext(char *plugin, | |||||||
| 		startup_cb_wrapper(ctx, &ctx->options, true); | 		startup_cb_wrapper(ctx, &ctx->options, true); | ||||||
| 	MemoryContextSwitchTo(old_context); | 	MemoryContextSwitchTo(old_context); | ||||||
|  |  | ||||||
|  | 	ctx->reorder->output_rewrites = ctx->options.receive_rewrites; | ||||||
|  |  | ||||||
| 	return ctx; | 	return ctx; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -410,6 +412,8 @@ CreateDecodingContext(XLogRecPtr start_lsn, | |||||||
| 		startup_cb_wrapper(ctx, &ctx->options, false); | 		startup_cb_wrapper(ctx, &ctx->options, false); | ||||||
| 	MemoryContextSwitchTo(old_context); | 	MemoryContextSwitchTo(old_context); | ||||||
|  |  | ||||||
|  | 	ctx->reorder->output_rewrites = ctx->options.receive_rewrites; | ||||||
|  |  | ||||||
| 	ereport(LOG, | 	ereport(LOG, | ||||||
| 			(errmsg("starting logical decoding for slot \"%s\"", | 			(errmsg("starting logical decoding for slot \"%s\"", | ||||||
| 					NameStr(slot->data.name)), | 					NameStr(slot->data.name)), | ||||||
|   | |||||||
| @@ -1402,6 +1402,13 @@ ReorderBufferCommit(ReorderBuffer *rb, TransactionId xid, | |||||||
| 					if (!RelationIsLogicallyLogged(relation)) | 					if (!RelationIsLogicallyLogged(relation)) | ||||||
| 						goto change_done; | 						goto change_done; | ||||||
|  |  | ||||||
|  | 					/* | ||||||
|  | 					 * Ignore temporary heaps created during DDL unless the | ||||||
|  | 					 * plugin has asked for them. | ||||||
|  | 					 */ | ||||||
|  | 					if (relation->rd_rel->relrewrite && !rb->output_rewrites) | ||||||
|  | 						goto change_done; | ||||||
|  |  | ||||||
| 					/* | 					/* | ||||||
| 					 * For now ignore sequence changes entirely. Most of the | 					 * For now ignore sequence changes entirely. Most of the | ||||||
| 					 * time they don't log changes using records we | 					 * time they don't log changes using records we | ||||||
|   | |||||||
| @@ -21,7 +21,6 @@ | |||||||
|  |  | ||||||
| #include "utils/inval.h" | #include "utils/inval.h" | ||||||
| #include "utils/int8.h" | #include "utils/int8.h" | ||||||
| #include "utils/lsyscache.h" |  | ||||||
| #include "utils/memutils.h" | #include "utils/memutils.h" | ||||||
| #include "utils/syscache.h" | #include "utils/syscache.h" | ||||||
| #include "utils/varlena.h" | #include "utils/varlena.h" | ||||||
| @@ -511,31 +510,6 @@ get_rel_sync_entry(PGOutputData *data, Oid relid) | |||||||
| 		{ | 		{ | ||||||
| 			Publication *pub = lfirst(lc); | 			Publication *pub = lfirst(lc); | ||||||
|  |  | ||||||
| 			/* |  | ||||||
| 			 * Skip tables that look like they are from a heap rewrite (see |  | ||||||
| 			 * make_new_heap()).  We need to skip them because the subscriber |  | ||||||
| 			 * won't have a table by that name to receive the data.  That |  | ||||||
| 			 * means we won't ship the new data in, say, an added column with |  | ||||||
| 			 * a DEFAULT, but if the user applies the same DDL manually on the |  | ||||||
| 			 * subscriber, then this will work out for them. |  | ||||||
| 			 * |  | ||||||
| 			 * We only need to consider the alltables case, because such a |  | ||||||
| 			 * transient heap won't be an explicit member of a publication. |  | ||||||
| 			 */ |  | ||||||
| 			if (pub->alltables) |  | ||||||
| 			{ |  | ||||||
| 				char	   *relname = get_rel_name(relid); |  | ||||||
| 				unsigned int u; |  | ||||||
| 				int			n; |  | ||||||
|  |  | ||||||
| 				if (sscanf(relname, "pg_temp_%u%n", &u, &n) == 1 && |  | ||||||
| 					relname[n] == '\0') |  | ||||||
| 				{ |  | ||||||
| 					if (get_rel_relkind(u) == RELKIND_RELATION) |  | ||||||
| 						break; |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			if (pub->alltables || list_member_oid(pubids, pub->oid)) | 			if (pub->alltables || list_member_oid(pubids, pub->oid)) | ||||||
| 			{ | 			{ | ||||||
| 				entry->pubactions.pubinsert |= pub->pubactions.pubinsert; | 				entry->pubactions.pubinsert |= pub->pubactions.pubinsert; | ||||||
|   | |||||||
| @@ -53,6 +53,6 @@ | |||||||
|  */ |  */ | ||||||
|  |  | ||||||
| /*							yyyymmddN */ | /*							yyyymmddN */ | ||||||
| #define CATALOG_VERSION_NO	201803141 | #define CATALOG_VERSION_NO	201803211 | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|   | |||||||
| @@ -71,6 +71,7 @@ extern Oid heap_create_with_catalog(const char *relname, | |||||||
| 						 bool use_user_acl, | 						 bool use_user_acl, | ||||||
| 						 bool allow_system_table_mods, | 						 bool allow_system_table_mods, | ||||||
| 						 bool is_internal, | 						 bool is_internal, | ||||||
|  | 						 Oid relrewrite, | ||||||
| 						 ObjectAddress *typaddress); | 						 ObjectAddress *typaddress); | ||||||
|  |  | ||||||
| extern void heap_create_init_fork(Relation rel); | extern void heap_create_init_fork(Relation rel); | ||||||
|   | |||||||
| @@ -70,6 +70,7 @@ CATALOG(pg_class,1259) BKI_BOOTSTRAP BKI_ROWTYPE_OID(83) BKI_SCHEMA_MACRO | |||||||
| 	bool		relispopulated; /* matview currently holds query results */ | 	bool		relispopulated; /* matview currently holds query results */ | ||||||
| 	char		relreplident;	/* see REPLICA_IDENTITY_xxx constants  */ | 	char		relreplident;	/* see REPLICA_IDENTITY_xxx constants  */ | ||||||
| 	bool		relispartition; /* is relation a partition? */ | 	bool		relispartition; /* is relation a partition? */ | ||||||
|  | 	Oid			relrewrite;		/* heap for rewrite during DDL, link to original rel */ | ||||||
| 	TransactionId relfrozenxid; /* all Xids < this are frozen in this rel */ | 	TransactionId relfrozenxid; /* all Xids < this are frozen in this rel */ | ||||||
| 	TransactionId relminmxid;	/* all multixacts in this rel are >= this. | 	TransactionId relminmxid;	/* all multixacts in this rel are >= this. | ||||||
| 								 * this is really a MultiXactId */ | 								 * this is really a MultiXactId */ | ||||||
| @@ -98,7 +99,7 @@ typedef FormData_pg_class *Form_pg_class; | |||||||
|  * ---------------- |  * ---------------- | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| #define Natts_pg_class						32 | #define Natts_pg_class						33 | ||||||
| #define Anum_pg_class_relname				1 | #define Anum_pg_class_relname				1 | ||||||
| #define Anum_pg_class_relnamespace			2 | #define Anum_pg_class_relnamespace			2 | ||||||
| #define Anum_pg_class_reltype				3 | #define Anum_pg_class_reltype				3 | ||||||
| @@ -126,11 +127,12 @@ typedef FormData_pg_class *Form_pg_class; | |||||||
| #define Anum_pg_class_relispopulated		25 | #define Anum_pg_class_relispopulated		25 | ||||||
| #define Anum_pg_class_relreplident			26 | #define Anum_pg_class_relreplident			26 | ||||||
| #define Anum_pg_class_relispartition		27 | #define Anum_pg_class_relispartition		27 | ||||||
| #define Anum_pg_class_relfrozenxid			28 | #define Anum_pg_class_relrewrite			28 | ||||||
| #define Anum_pg_class_relminmxid			29 | #define Anum_pg_class_relfrozenxid			29 | ||||||
| #define Anum_pg_class_relacl				30 | #define Anum_pg_class_relminmxid			30 | ||||||
| #define Anum_pg_class_reloptions			31 | #define Anum_pg_class_relacl				31 | ||||||
| #define Anum_pg_class_relpartbound			32 | #define Anum_pg_class_reloptions			32 | ||||||
|  | #define Anum_pg_class_relpartbound			33 | ||||||
|  |  | ||||||
| /* ---------------- | /* ---------------- | ||||||
|  *		initial contents of pg_class |  *		initial contents of pg_class | ||||||
| @@ -145,13 +147,13 @@ typedef FormData_pg_class *Form_pg_class; | |||||||
|  * Note: "3" in the relfrozenxid column stands for FirstNormalTransactionId; |  * Note: "3" in the relfrozenxid column stands for FirstNormalTransactionId; | ||||||
|  * similarly, "1" in relminmxid stands for FirstMultiXactId |  * similarly, "1" in relminmxid stands for FirstMultiXactId | ||||||
|  */ |  */ | ||||||
| DATA(insert OID = 1247 (  pg_type		PGNSP 71 0 PGUID 0 0 0 0 0 0 0 f f p r 30 0 t f f f f f t n f 3 1 _null_ _null_ _null_)); | DATA(insert OID = 1247 (  pg_type		PGNSP 71 0 PGUID 0 0 0 0 0 0 0 f f p r 30 0 t f f f f f t n f 0 3 1 _null_ _null_ _null_)); | ||||||
| DESCR(""); | DESCR(""); | ||||||
| DATA(insert OID = 1249 (  pg_attribute	PGNSP 75 0 PGUID 0 0 0 0 0 0 0 f f p r 22 0 f f f f f f t n f 3 1 _null_ _null_ _null_)); | DATA(insert OID = 1249 (  pg_attribute	PGNSP 75 0 PGUID 0 0 0 0 0 0 0 f f p r 22 0 f f f f f f t n f 0 3 1 _null_ _null_ _null_)); | ||||||
| DESCR(""); | DESCR(""); | ||||||
| DATA(insert OID = 1255 (  pg_proc		PGNSP 81 0 PGUID 0 0 0 0 0 0 0 f f p r 28 0 t f f f f f t n f 3 1 _null_ _null_ _null_)); | DATA(insert OID = 1255 (  pg_proc		PGNSP 81 0 PGUID 0 0 0 0 0 0 0 f f p r 28 0 t f f f f f t n f 0 3 1 _null_ _null_ _null_)); | ||||||
| DESCR(""); | DESCR(""); | ||||||
| DATA(insert OID = 1259 (  pg_class		PGNSP 83 0 PGUID 0 0 0 0 0 0 0 f f p r 32 0 t f f f f f t n f 3 1 _null_ _null_ _null_)); | DATA(insert OID = 1259 (  pg_class		PGNSP 83 0 PGUID 0 0 0 0 0 0 0 f f p r 33 0 t f f f f f t n f 0 3 1 _null_ _null_ _null_)); | ||||||
| DESCR(""); | DESCR(""); | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -26,6 +26,7 @@ typedef enum OutputPluginOutputType | |||||||
| typedef struct OutputPluginOptions | typedef struct OutputPluginOptions | ||||||
| { | { | ||||||
| 	OutputPluginOutputType output_type; | 	OutputPluginOutputType output_type; | ||||||
|  | 	bool		receive_rewrites; | ||||||
| } OutputPluginOptions; | } OutputPluginOptions; | ||||||
|  |  | ||||||
| /* | /* | ||||||
|   | |||||||
| @@ -336,6 +336,11 @@ struct ReorderBuffer | |||||||
| 	 */ | 	 */ | ||||||
| 	void	   *private_data; | 	void	   *private_data; | ||||||
|  |  | ||||||
|  | 	/* | ||||||
|  | 	 * Saved output plugin option | ||||||
|  | 	 */ | ||||||
|  | 	bool		output_rewrites; | ||||||
|  |  | ||||||
| 	/* | 	/* | ||||||
| 	 * Private memory context. | 	 * Private memory context. | ||||||
| 	 */ | 	 */ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user