aboutsummaryrefslogtreecommitdiff
path: root/src/backend/replication/pgoutput/pgoutput.c
diff options
context:
space:
mode:
authorPeter Eisentraut <peter_e@gmx.net>2017-09-26 10:03:56 -0400
committerPeter Eisentraut <peter_e@gmx.net>2017-09-26 10:13:43 -0400
commitab28feae2bd3d4629bd73ae3548e671c57d785f0 (patch)
treed9253da0d2621abc4522bafcdbf67ea37cce4ca4 /src/backend/replication/pgoutput/pgoutput.c
parent22c5e73562c53437979efec4c26cd9fff408777c (diff)
downloadpostgresql-ab28feae2bd3d4629bd73ae3548e671c57d785f0.tar.gz
postgresql-ab28feae2bd3d4629bd73ae3548e671c57d785f0.zip
Handle heap rewrites better in logical replication
A FOR ALL TABLES publication naturally considers all base tables to be a candidate for replication. This includes transient heaps that are created during a table rewrite during DDL. This causes failures on the subscriber side because it will not have a table like pg_temp_16386 to receive data (and if it did, it would be the wrong table). The prevent this problem, we filter out any tables that match this naming pattern and match an actual table from FOR ALL TABLES publications. This is only a heuristic, meaning that user tables that match that naming could accidentally be omitted. A more robust solution might require an explicit marking of such tables in pg_class somehow. Reported-by: yxq <yxq@o2.pl> Bug: #14785 Reviewed-by: Andres Freund <andres@anarazel.de> Reviewed-by: Petr Jelinek <petr.jelinek@2ndquadrant.com>
Diffstat (limited to 'src/backend/replication/pgoutput/pgoutput.c')
-rw-r--r--src/backend/replication/pgoutput/pgoutput.c26
1 files changed, 26 insertions, 0 deletions
diff --git a/src/backend/replication/pgoutput/pgoutput.c b/src/backend/replication/pgoutput/pgoutput.c
index 9ab954a6e01..c3126545b48 100644
--- a/src/backend/replication/pgoutput/pgoutput.c
+++ b/src/backend/replication/pgoutput/pgoutput.c
@@ -21,6 +21,7 @@
#include "utils/inval.h"
#include "utils/int8.h"
+#include "utils/lsyscache.h"
#include "utils/memutils.h"
#include "utils/syscache.h"
#include "utils/varlena.h"
@@ -509,6 +510,31 @@ get_rel_sync_entry(PGOutputData *data, Oid relid)
{
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))
{
entry->pubactions.pubinsert |= pub->pubactions.pubinsert;