Re: Wrong usage of RelationNeedsWAL - Mailing list pgsql-hackers
From | Kyotaro Horiguchi |
---|---|
Subject | Re: Wrong usage of RelationNeedsWAL |
Date | |
Msg-id | 20210119.134831.1956292555969369177.horikyota.ntt@gmail.com Whole thread Raw |
In response to | Re: Wrong usage of RelationNeedsWAL (Kyotaro Horiguchi <horikyota.ntt@gmail.com>) |
Responses |
Re: Wrong usage of RelationNeedsWAL
|
List | pgsql-hackers |
At Mon, 18 Jan 2021 17:30:22 +0900 (JST), Kyotaro Horiguchi <horikyota.ntt@gmail.com> wrote in > At Sun, 17 Jan 2021 23:02:18 -0800, Noah Misch <noah@leadboat.com> wrote in > > On Sun, Jan 17, 2021 at 10:36:31PM -0800, Noah Misch wrote: > > > I wrote the above based on the "PageGetLSN(page) > (snapshot)->lsn" check in > > > TestForOldSnapshot(). If the LSN isn't important, what else explains > > > RelationAllowsEarlyPruning() checking RelationNeedsWAL()? > > > > Thinking about it more, some RelationAllowsEarlyPruning() callers need > > different treatment. Above, I was writing about the case of deciding whether > > to do early pruning. The other RelationAllowsEarlyPruning() call sites are > > deciding whether the relation might be lacking old data. For that case, we > > should check RELPERSISTENCE_PERMANENT, not RelationNeedsWAL(). Otherwise, we > > could fail to report an old-snapshot error in a case like this: > > A> > setup: CREATE TABLE t AS SELECT ...; B> > xact1: BEGIN ISOLATION LEVEL REPEATABLE READ; SELECT 1; -- take snapshot C> > xact2: DELETE FROM t; D> > (plenty of time passes) E> > xact3: SELECT count(*) FROM t; -- early pruning F> > xact1: SAVEPOINT q; SELECT count(*) FROM t; ROLLBACK TO q; -- "snapshot too old" G> > xact1: ALTER TABLE t SET TABLESPACE something; -- start skipping WAL H> > xact1: SELECT count(*) FROM t; -- no error, wanted "snapshot too old" > > > > Is that plausible? > > Thank you for the consideration and yes. But I get "snapshot too old" > from the last query with the patched version. maybe I'm missing > something. I'm going to investigate the case. Ah. I took that in reverse way. (caught by the discussion on page LSN.) Ok, the "current" code works that way. So we need to perform the check the new way in RelationAllowsEarlyPruning. So in a short answer for the last question in regard to the reference side is yes. I understand that you are suggesting that at least TransactionIdLimitedForOldSnapshots should follow not only relation persistence but RelationNeedsWAL, based on the discussion on the check on LSN of TestForOldSnapshot(). I still don't think that LSN in the WAL-skipping-created relfilenode harm. ALTER TABLE SET TABLESPACE always makes a dead copy of every block (except checksum) including page LSN regardless of wal_level. In the scenario above, the last select at H correctly reads page LSN set by E then copied by G, which is larger than the snapshot LSN at B. So doesn't go the fast-return path before actual check performed by RelationAllowsEarlyPruning. As the result still RelationAllowsEarlyPruning is changed to check only for the persistence of the table in this version. (In other words, all the callers of the function works based on the same criteria.) - Removed RelationIsWalLoggeed which was forgotten to remove in the last version. - Enclosed parameter of RelationAllowsEarlyPruning. regards. -- Kyotaro Horiguchi NTT Open Source Software Center From d688abbc04459b11bc2801f3c7f1955a86ef7a51 Mon Sep 17 00:00:00 2001 From: Kyotaro Horiguchi <horikyoga.ntt@gmail.com> Date: Mon, 18 Jan 2021 14:47:21 +0900 Subject: [PATCH v3] Do not use RelationNeedsWAL to identify relation persistence RelationNeedsWAL() may return false for a permanent relation when wal_level=minimal and the relation is created or truncated in the current transaction. Directly examine relpersistence instead of using the function to know relation persistence. --- src/backend/catalog/pg_publication.c | 2 +- src/backend/optimizer/util/plancat.c | 3 ++- src/include/utils/rel.h | 2 +- src/include/utils/snapmgr.h | 2 +- src/test/subscription/t/001_rep_changes.pl | 20 +++++++++++++++++++- 5 files changed, 24 insertions(+), 5 deletions(-) diff --git a/src/backend/catalog/pg_publication.c b/src/backend/catalog/pg_publication.c index 5f8e1c64e1..84d2efcfd2 100644 --- a/src/backend/catalog/pg_publication.c +++ b/src/backend/catalog/pg_publication.c @@ -67,7 +67,7 @@ check_publication_add_relation(Relation targetrel) errdetail("System tables cannot be added to publications."))); /* UNLOGGED and TEMP relations cannot be part of publication. */ - if (!RelationNeedsWAL(targetrel)) + if (targetrel->rd_rel->relpersistence != RELPERSISTENCE_PERMANENT) ereport(ERROR, (errcode(ERRCODE_INVALID_PARAMETER_VALUE), errmsg("table \"%s\" cannot be replicated", diff --git a/src/backend/optimizer/util/plancat.c b/src/backend/optimizer/util/plancat.c index da322b453e..177e6e336a 100644 --- a/src/backend/optimizer/util/plancat.c +++ b/src/backend/optimizer/util/plancat.c @@ -126,7 +126,8 @@ get_relation_info(PlannerInfo *root, Oid relationObjectId, bool inhparent, relation = table_open(relationObjectId, NoLock); /* Temporary and unlogged relations are inaccessible during recovery. */ - if (!RelationNeedsWAL(relation) && RecoveryInProgress()) + if (relation->rd_rel->relpersistence != RELPERSISTENCE_PERMANENT && + RecoveryInProgress()) ereport(ERROR, (errcode(ERRCODE_FEATURE_NOT_SUPPORTED), errmsg("cannot access temporary or unlogged relations during recovery"))); diff --git a/src/include/utils/rel.h b/src/include/utils/rel.h index 10b63982c0..f58d65cf28 100644 --- a/src/include/utils/rel.h +++ b/src/include/utils/rel.h @@ -554,7 +554,7 @@ typedef struct ViewOptions /* * RelationNeedsWAL - * True if relation needs WAL. + * True if relation needs WAL at the time. * * Returns false if wal_level = minimal and this relation is created or * truncated in the current transaction. See "Skipping WAL for New diff --git a/src/include/utils/snapmgr.h b/src/include/utils/snapmgr.h index 579be352c5..c21ee3c289 100644 --- a/src/include/utils/snapmgr.h +++ b/src/include/utils/snapmgr.h @@ -37,7 +37,7 @@ */ #define RelationAllowsEarlyPruning(rel) \ ( \ - RelationNeedsWAL(rel) \ + (rel)->rd_rel->relpersistence == RELPERSISTENCE_PERMANENT \ && !IsCatalogRelation(rel) \ && !RelationIsAccessibleInLogicalDecoding(rel) \ ) diff --git a/src/test/subscription/t/001_rep_changes.pl b/src/test/subscription/t/001_rep_changes.pl index 0680f44a1a..ed9b48e8bc 100644 --- a/src/test/subscription/t/001_rep_changes.pl +++ b/src/test/subscription/t/001_rep_changes.pl @@ -3,7 +3,7 @@ use strict; use warnings; use PostgresNode; use TestLib; -use Test::More tests => 23; +use Test::More tests => 24; # Initialize publisher node my $node_publisher = get_new_node('publisher'); @@ -358,3 +358,21 @@ is($result, qq(0), 'check replication origin was dropped on subscriber'); $node_subscriber->stop('fast'); $node_publisher->stop('fast'); + +# +# CREATE PUBLICATION while wal_level=mimal should succeed, with a WARNING +$node_publisher->append_conf( + 'postgresql.conf', qq( +wal_level=minimal +max_wal_senders=0 +)); +$node_publisher->start; +($result, my $retout, my $reterr) = $node_publisher->psql( + 'postgres', qq{ +BEGIN; +CREATE TABLE skip_wal(); +CREATE PUBLICATION tap_pub2 FOR TABLE skip_wal; +ROLLBACK; +}); +ok($reterr =~ m/WARNING: wal_level is insufficient to publish logical changes/, + 'test CREATE PUBLICATION can be done while wal_leve=minimal'); -- 2.27.0
pgsql-hackers by date: