Commit 47167b79 authored by Simon Riggs's avatar Simon Riggs

Reduce lock levels for ALTER TABLE SET autovacuum storage options

Reduce lock levels down to ShareUpdateExclusiveLock for all autovacuum-related
relation options when setting them using ALTER TABLE.

Add infrastructure to allow varying lock levels for relation options in later
patches. Setting multiple options together uses the highest lock level required
for any option. Works for both main and toast tables.

Fabrízio Mello, reviewed by Michael Paquier, mild edit and additional regression
tests from myself
parent f16d5226
...@@ -543,6 +543,10 @@ ALTER TABLE ALL IN TABLESPACE <replaceable class="PARAMETER">name</replaceable> ...@@ -543,6 +543,10 @@ ALTER TABLE ALL IN TABLESPACE <replaceable class="PARAMETER">name</replaceable>
of <command>ALTER TABLE</> that forces a table rewrite. of <command>ALTER TABLE</> that forces a table rewrite.
</para> </para>
<para>
Changing autovacuum storage parameters acquires a <literal>SHARE UPDATE EXCLUSIVE</literal> lock.
</para>
<note> <note>
<para> <para>
While <command>CREATE TABLE</> allows <literal>OIDS</> to be specified While <command>CREATE TABLE</> allows <literal>OIDS</> to be specified
......
...@@ -57,7 +57,8 @@ static relopt_bool boolRelOpts[] = ...@@ -57,7 +57,8 @@ static relopt_bool boolRelOpts[] =
{ {
"autovacuum_enabled", "autovacuum_enabled",
"Enables autovacuum in this relation", "Enables autovacuum in this relation",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, },
true true
}, },
...@@ -65,7 +66,8 @@ static relopt_bool boolRelOpts[] = ...@@ -65,7 +66,8 @@ static relopt_bool boolRelOpts[] =
{ {
"user_catalog_table", "user_catalog_table",
"Declare a table as an additional catalog table, e.g. for the purpose of logical replication", "Declare a table as an additional catalog table, e.g. for the purpose of logical replication",
RELOPT_KIND_HEAP RELOPT_KIND_HEAP,
AccessExclusiveLock
}, },
false false
}, },
...@@ -73,7 +75,8 @@ static relopt_bool boolRelOpts[] = ...@@ -73,7 +75,8 @@ static relopt_bool boolRelOpts[] =
{ {
"fastupdate", "fastupdate",
"Enables \"fast update\" feature for this GIN index", "Enables \"fast update\" feature for this GIN index",
RELOPT_KIND_GIN RELOPT_KIND_GIN,
AccessExclusiveLock
}, },
true true
}, },
...@@ -81,7 +84,8 @@ static relopt_bool boolRelOpts[] = ...@@ -81,7 +84,8 @@ static relopt_bool boolRelOpts[] =
{ {
"security_barrier", "security_barrier",
"View acts as a row security barrier", "View acts as a row security barrier",
RELOPT_KIND_VIEW RELOPT_KIND_VIEW,
AccessExclusiveLock
}, },
false false
}, },
...@@ -95,7 +99,8 @@ static relopt_int intRelOpts[] = ...@@ -95,7 +99,8 @@ static relopt_int intRelOpts[] =
{ {
"fillfactor", "fillfactor",
"Packs table pages only to this percentage", "Packs table pages only to this percentage",
RELOPT_KIND_HEAP RELOPT_KIND_HEAP,
AccessExclusiveLock
}, },
HEAP_DEFAULT_FILLFACTOR, HEAP_MIN_FILLFACTOR, 100 HEAP_DEFAULT_FILLFACTOR, HEAP_MIN_FILLFACTOR, 100
}, },
...@@ -103,7 +108,8 @@ static relopt_int intRelOpts[] = ...@@ -103,7 +108,8 @@ static relopt_int intRelOpts[] =
{ {
"fillfactor", "fillfactor",
"Packs btree index pages only to this percentage", "Packs btree index pages only to this percentage",
RELOPT_KIND_BTREE RELOPT_KIND_BTREE,
AccessExclusiveLock
}, },
BTREE_DEFAULT_FILLFACTOR, BTREE_MIN_FILLFACTOR, 100 BTREE_DEFAULT_FILLFACTOR, BTREE_MIN_FILLFACTOR, 100
}, },
...@@ -111,7 +117,8 @@ static relopt_int intRelOpts[] = ...@@ -111,7 +117,8 @@ static relopt_int intRelOpts[] =
{ {
"fillfactor", "fillfactor",
"Packs hash index pages only to this percentage", "Packs hash index pages only to this percentage",
RELOPT_KIND_HASH RELOPT_KIND_HASH,
AccessExclusiveLock
}, },
HASH_DEFAULT_FILLFACTOR, HASH_MIN_FILLFACTOR, 100 HASH_DEFAULT_FILLFACTOR, HASH_MIN_FILLFACTOR, 100
}, },
...@@ -119,7 +126,8 @@ static relopt_int intRelOpts[] = ...@@ -119,7 +126,8 @@ static relopt_int intRelOpts[] =
{ {
"fillfactor", "fillfactor",
"Packs gist index pages only to this percentage", "Packs gist index pages only to this percentage",
RELOPT_KIND_GIST RELOPT_KIND_GIST,
AccessExclusiveLock
}, },
GIST_DEFAULT_FILLFACTOR, GIST_MIN_FILLFACTOR, 100 GIST_DEFAULT_FILLFACTOR, GIST_MIN_FILLFACTOR, 100
}, },
...@@ -127,7 +135,8 @@ static relopt_int intRelOpts[] = ...@@ -127,7 +135,8 @@ static relopt_int intRelOpts[] =
{ {
"fillfactor", "fillfactor",
"Packs spgist index pages only to this percentage", "Packs spgist index pages only to this percentage",
RELOPT_KIND_SPGIST RELOPT_KIND_SPGIST,
AccessExclusiveLock
}, },
SPGIST_DEFAULT_FILLFACTOR, SPGIST_MIN_FILLFACTOR, 100 SPGIST_DEFAULT_FILLFACTOR, SPGIST_MIN_FILLFACTOR, 100
}, },
...@@ -135,7 +144,8 @@ static relopt_int intRelOpts[] = ...@@ -135,7 +144,8 @@ static relopt_int intRelOpts[] =
{ {
"autovacuum_vacuum_threshold", "autovacuum_vacuum_threshold",
"Minimum number of tuple updates or deletes prior to vacuum", "Minimum number of tuple updates or deletes prior to vacuum",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, },
-1, 0, INT_MAX -1, 0, INT_MAX
}, },
...@@ -143,7 +153,8 @@ static relopt_int intRelOpts[] = ...@@ -143,7 +153,8 @@ static relopt_int intRelOpts[] =
{ {
"autovacuum_analyze_threshold", "autovacuum_analyze_threshold",
"Minimum number of tuple inserts, updates or deletes prior to analyze", "Minimum number of tuple inserts, updates or deletes prior to analyze",
RELOPT_KIND_HEAP RELOPT_KIND_HEAP,
ShareUpdateExclusiveLock
}, },
-1, 0, INT_MAX -1, 0, INT_MAX
}, },
...@@ -151,7 +162,8 @@ static relopt_int intRelOpts[] = ...@@ -151,7 +162,8 @@ static relopt_int intRelOpts[] =
{ {
"autovacuum_vacuum_cost_delay", "autovacuum_vacuum_cost_delay",
"Vacuum cost delay in milliseconds, for autovacuum", "Vacuum cost delay in milliseconds, for autovacuum",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, },
-1, 0, 100 -1, 0, 100
}, },
...@@ -159,7 +171,8 @@ static relopt_int intRelOpts[] = ...@@ -159,7 +171,8 @@ static relopt_int intRelOpts[] =
{ {
"autovacuum_vacuum_cost_limit", "autovacuum_vacuum_cost_limit",
"Vacuum cost amount available before napping, for autovacuum", "Vacuum cost amount available before napping, for autovacuum",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, },
-1, 1, 10000 -1, 1, 10000
}, },
...@@ -167,7 +180,8 @@ static relopt_int intRelOpts[] = ...@@ -167,7 +180,8 @@ static relopt_int intRelOpts[] =
{ {
"autovacuum_freeze_min_age", "autovacuum_freeze_min_age",
"Minimum age at which VACUUM should freeze a table row, for autovacuum", "Minimum age at which VACUUM should freeze a table row, for autovacuum",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, },
-1, 0, 1000000000 -1, 0, 1000000000
}, },
...@@ -175,7 +189,8 @@ static relopt_int intRelOpts[] = ...@@ -175,7 +189,8 @@ static relopt_int intRelOpts[] =
{ {
"autovacuum_multixact_freeze_min_age", "autovacuum_multixact_freeze_min_age",
"Minimum multixact age at which VACUUM should freeze a row multixact's, for autovacuum", "Minimum multixact age at which VACUUM should freeze a row multixact's, for autovacuum",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, },
-1, 0, 1000000000 -1, 0, 1000000000
}, },
...@@ -183,7 +198,8 @@ static relopt_int intRelOpts[] = ...@@ -183,7 +198,8 @@ static relopt_int intRelOpts[] =
{ {
"autovacuum_freeze_max_age", "autovacuum_freeze_max_age",
"Age at which to autovacuum a table to prevent transaction ID wraparound", "Age at which to autovacuum a table to prevent transaction ID wraparound",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, },
-1, 100000000, 2000000000 -1, 100000000, 2000000000
}, },
...@@ -191,7 +207,8 @@ static relopt_int intRelOpts[] = ...@@ -191,7 +207,8 @@ static relopt_int intRelOpts[] =
{ {
"autovacuum_multixact_freeze_max_age", "autovacuum_multixact_freeze_max_age",
"Multixact age at which to autovacuum a table to prevent multixact wraparound", "Multixact age at which to autovacuum a table to prevent multixact wraparound",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, },
-1, 100000000, 2000000000 -1, 100000000, 2000000000
}, },
...@@ -199,21 +216,24 @@ static relopt_int intRelOpts[] = ...@@ -199,21 +216,24 @@ static relopt_int intRelOpts[] =
{ {
"autovacuum_freeze_table_age", "autovacuum_freeze_table_age",
"Age at which VACUUM should perform a full table sweep to freeze row versions", "Age at which VACUUM should perform a full table sweep to freeze row versions",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, -1, 0, 2000000000 }, -1, 0, 2000000000
}, },
{ {
{ {
"autovacuum_multixact_freeze_table_age", "autovacuum_multixact_freeze_table_age",
"Age of multixact at which VACUUM should perform a full table sweep to freeze row versions", "Age of multixact at which VACUUM should perform a full table sweep to freeze row versions",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, -1, 0, 2000000000 }, -1, 0, 2000000000
}, },
{ {
{ {
"log_autovacuum_min_duration", "log_autovacuum_min_duration",
"Sets the minimum execution time above which autovacuum actions will be logged", "Sets the minimum execution time above which autovacuum actions will be logged",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, },
-1, -1, INT_MAX -1, -1, INT_MAX
}, },
...@@ -221,14 +241,16 @@ static relopt_int intRelOpts[] = ...@@ -221,14 +241,16 @@ static relopt_int intRelOpts[] =
{ {
"pages_per_range", "pages_per_range",
"Number of pages that each page range covers in a BRIN index", "Number of pages that each page range covers in a BRIN index",
RELOPT_KIND_BRIN RELOPT_KIND_BRIN,
AccessExclusiveLock
}, 128, 1, 131072 }, 128, 1, 131072
}, },
{ {
{ {
"gin_pending_list_limit", "gin_pending_list_limit",
"Maximum size of the pending list for this GIN index, in kilobytes.", "Maximum size of the pending list for this GIN index, in kilobytes.",
RELOPT_KIND_GIN RELOPT_KIND_GIN,
AccessExclusiveLock
}, },
-1, 64, MAX_KILOBYTES -1, 64, MAX_KILOBYTES
}, },
...@@ -243,7 +265,8 @@ static relopt_real realRelOpts[] = ...@@ -243,7 +265,8 @@ static relopt_real realRelOpts[] =
{ {
"autovacuum_vacuum_scale_factor", "autovacuum_vacuum_scale_factor",
"Number of tuple updates or deletes prior to vacuum as a fraction of reltuples", "Number of tuple updates or deletes prior to vacuum as a fraction of reltuples",
RELOPT_KIND_HEAP | RELOPT_KIND_TOAST RELOPT_KIND_HEAP | RELOPT_KIND_TOAST,
ShareUpdateExclusiveLock
}, },
-1, 0.0, 100.0 -1, 0.0, 100.0
}, },
...@@ -251,7 +274,8 @@ static relopt_real realRelOpts[] = ...@@ -251,7 +274,8 @@ static relopt_real realRelOpts[] =
{ {
"autovacuum_analyze_scale_factor", "autovacuum_analyze_scale_factor",
"Number of tuple inserts, updates or deletes prior to analyze as a fraction of reltuples", "Number of tuple inserts, updates or deletes prior to analyze as a fraction of reltuples",
RELOPT_KIND_HEAP RELOPT_KIND_HEAP,
ShareUpdateExclusiveLock
}, },
-1, 0.0, 100.0 -1, 0.0, 100.0
}, },
...@@ -259,7 +283,8 @@ static relopt_real realRelOpts[] = ...@@ -259,7 +283,8 @@ static relopt_real realRelOpts[] =
{ {
"seq_page_cost", "seq_page_cost",
"Sets the planner's estimate of the cost of a sequentially fetched disk page.", "Sets the planner's estimate of the cost of a sequentially fetched disk page.",
RELOPT_KIND_TABLESPACE RELOPT_KIND_TABLESPACE,
AccessExclusiveLock
}, },
-1, 0.0, DBL_MAX -1, 0.0, DBL_MAX
}, },
...@@ -267,7 +292,8 @@ static relopt_real realRelOpts[] = ...@@ -267,7 +292,8 @@ static relopt_real realRelOpts[] =
{ {
"random_page_cost", "random_page_cost",
"Sets the planner's estimate of the cost of a nonsequentially fetched disk page.", "Sets the planner's estimate of the cost of a nonsequentially fetched disk page.",
RELOPT_KIND_TABLESPACE RELOPT_KIND_TABLESPACE,
AccessExclusiveLock
}, },
-1, 0.0, DBL_MAX -1, 0.0, DBL_MAX
}, },
...@@ -275,7 +301,8 @@ static relopt_real realRelOpts[] = ...@@ -275,7 +301,8 @@ static relopt_real realRelOpts[] =
{ {
"n_distinct", "n_distinct",
"Sets the planner's estimate of the number of distinct values appearing in a column (excluding child relations).", "Sets the planner's estimate of the number of distinct values appearing in a column (excluding child relations).",
RELOPT_KIND_ATTRIBUTE RELOPT_KIND_ATTRIBUTE,
AccessExclusiveLock
}, },
0, -1.0, DBL_MAX 0, -1.0, DBL_MAX
}, },
...@@ -283,7 +310,8 @@ static relopt_real realRelOpts[] = ...@@ -283,7 +310,8 @@ static relopt_real realRelOpts[] =
{ {
"n_distinct_inherited", "n_distinct_inherited",
"Sets the planner's estimate of the number of distinct values appearing in a column (including child relations).", "Sets the planner's estimate of the number of distinct values appearing in a column (including child relations).",
RELOPT_KIND_ATTRIBUTE RELOPT_KIND_ATTRIBUTE,
AccessExclusiveLock
}, },
0, -1.0, DBL_MAX 0, -1.0, DBL_MAX
}, },
...@@ -297,7 +325,8 @@ static relopt_string stringRelOpts[] = ...@@ -297,7 +325,8 @@ static relopt_string stringRelOpts[] =
{ {
"buffering", "buffering",
"Enables buffering build for this GiST index", "Enables buffering build for this GiST index",
RELOPT_KIND_GIST RELOPT_KIND_GIST,
AccessExclusiveLock
}, },
4, 4,
false, false,
...@@ -308,7 +337,8 @@ static relopt_string stringRelOpts[] = ...@@ -308,7 +337,8 @@ static relopt_string stringRelOpts[] =
{ {
"check_option", "check_option",
"View has WITH CHECK OPTION defined (local or cascaded).", "View has WITH CHECK OPTION defined (local or cascaded).",
RELOPT_KIND_VIEW RELOPT_KIND_VIEW,
AccessExclusiveLock
}, },
0, 0,
true, true,
...@@ -344,13 +374,29 @@ initialize_reloptions(void) ...@@ -344,13 +374,29 @@ initialize_reloptions(void)
j = 0; j = 0;
for (i = 0; boolRelOpts[i].gen.name; i++) for (i = 0; boolRelOpts[i].gen.name; i++)
{
Assert(DoLockModesConflict(boolRelOpts[i].gen.lockmode,
boolRelOpts[i].gen.lockmode));
j++; j++;
}
for (i = 0; intRelOpts[i].gen.name; i++) for (i = 0; intRelOpts[i].gen.name; i++)
{
Assert(DoLockModesConflict(intRelOpts[i].gen.lockmode,
intRelOpts[i].gen.lockmode));
j++; j++;
}
for (i = 0; realRelOpts[i].gen.name; i++) for (i = 0; realRelOpts[i].gen.name; i++)
{
Assert(DoLockModesConflict(realRelOpts[i].gen.lockmode,
realRelOpts[i].gen.lockmode));
j++; j++;
}
for (i = 0; stringRelOpts[i].gen.name; i++) for (i = 0; stringRelOpts[i].gen.name; i++)
{
Assert(DoLockModesConflict(stringRelOpts[i].gen.lockmode,
stringRelOpts[i].gen.lockmode));
j++; j++;
}
j += num_custom_options; j += num_custom_options;
if (relOpts) if (relOpts)
...@@ -1411,3 +1457,41 @@ tablespace_reloptions(Datum reloptions, bool validate) ...@@ -1411,3 +1457,41 @@ tablespace_reloptions(Datum reloptions, bool validate)
return (bytea *) tsopts; return (bytea *) tsopts;
} }
/*
* Determine the required LOCKMODE from an option list.
*
* Called from AlterTableGetLockLevel(), see that function
* for a longer explanation of how this works.
*/
LOCKMODE
AlterTableGetRelOptionsLockLevel(List *defList)
{
LOCKMODE lockmode = NoLock;
ListCell *cell;
if (defList == NIL)
return AccessExclusiveLock;
if (need_initialization)
initialize_reloptions();
foreach(cell, defList)
{
DefElem *def = (DefElem *) lfirst(cell);
int i;
for (i = 0; relOpts[i]; i++)
{
if (pg_strncasecmp(relOpts[i]->name,
def->defname,
relOpts[i]->namelen + 1) == 0)
{
if (lockmode < relOpts[i]->lockmode)
lockmode = relOpts[i]->lockmode;
}
}
}
return lockmode;
}
...@@ -3038,16 +3038,12 @@ AlterTableGetLockLevel(List *cmds) ...@@ -3038,16 +3038,12 @@ AlterTableGetLockLevel(List *cmds)
* are set here for tables, views and indexes; for historical * are set here for tables, views and indexes; for historical
* reasons these can all be used with ALTER TABLE, so we can't * reasons these can all be used with ALTER TABLE, so we can't
* decide between them using the basic grammar. * decide between them using the basic grammar.
*
* XXX Look in detail at each option to determine lock level,
* e.g. cmd_lockmode = GetRelOptionsLockLevel((List *)
* cmd->def);
*/ */
case AT_SetRelOptions: /* Uses MVCC in getIndexes() and case AT_SetRelOptions: /* Uses MVCC in getIndexes() and
* getTables() */ * getTables() */
case AT_ResetRelOptions: /* Uses MVCC in getIndexes() and case AT_ResetRelOptions: /* Uses MVCC in getIndexes() and
* getTables() */ * getTables() */
cmd_lockmode = AccessExclusiveLock; cmd_lockmode = AlterTableGetRelOptionsLockLevel((List *) cmd->def);
break; break;
default: /* oops */ default: /* oops */
......
...@@ -22,6 +22,7 @@ ...@@ -22,6 +22,7 @@
#include "access/htup.h" #include "access/htup.h"
#include "access/tupdesc.h" #include "access/tupdesc.h"
#include "nodes/pg_list.h" #include "nodes/pg_list.h"
#include "storage/lock.h"
/* types supported by reloptions */ /* types supported by reloptions */
typedef enum relopt_type typedef enum relopt_type
...@@ -62,6 +63,7 @@ typedef struct relopt_gen ...@@ -62,6 +63,7 @@ typedef struct relopt_gen
* marker) */ * marker) */
const char *desc; const char *desc;
bits32 kinds; bits32 kinds;
LOCKMODE lockmode;
int namelen; int namelen;
relopt_type type; relopt_type type;
} relopt_gen; } relopt_gen;
...@@ -274,5 +276,6 @@ extern bytea *index_reloptions(RegProcedure amoptions, Datum reloptions, ...@@ -274,5 +276,6 @@ extern bytea *index_reloptions(RegProcedure amoptions, Datum reloptions,
bool validate); bool validate);
extern bytea *attribute_reloptions(Datum reloptions, bool validate); extern bytea *attribute_reloptions(Datum reloptions, bool validate);
extern bytea *tablespace_reloptions(Datum reloptions, bool validate); extern bytea *tablespace_reloptions(Datum reloptions, bool validate);
extern LOCKMODE AlterTableGetRelOptionsLockLevel(List *defList);
#endif /* RELOPTIONS_H */ #endif /* RELOPTIONS_H */
...@@ -1915,18 +1915,18 @@ commit; ...@@ -1915,18 +1915,18 @@ commit;
begin; alter table alterlock set (toast.autovacuum_enabled = off); begin; alter table alterlock set (toast.autovacuum_enabled = off);
select * from my_locks order by 1; select * from my_locks order by 1;
relname | max_lockmode relname | max_lockmode
-----------+--------------------- -----------+--------------------------
alterlock | AccessExclusiveLock alterlock | ShareUpdateExclusiveLock
pg_toast | AccessExclusiveLock pg_toast | ShareUpdateExclusiveLock
(2 rows) (2 rows)
commit; commit;
begin; alter table alterlock set (autovacuum_enabled = off); begin; alter table alterlock set (autovacuum_enabled = off);
select * from my_locks order by 1; select * from my_locks order by 1;
relname | max_lockmode relname | max_lockmode
-----------+--------------------- -----------+--------------------------
alterlock | AccessExclusiveLock alterlock | ShareUpdateExclusiveLock
pg_toast | AccessExclusiveLock pg_toast | ShareUpdateExclusiveLock
(2 rows) (2 rows)
commit; commit;
...@@ -1938,6 +1938,16 @@ select * from my_locks order by 1; ...@@ -1938,6 +1938,16 @@ select * from my_locks order by 1;
(1 row) (1 row)
rollback; rollback;
-- test that mixing options with different lock levels works as expected
begin; alter table alterlock set (autovacuum_enabled = off, fillfactor = 80);
select * from my_locks order by 1;
relname | max_lockmode
-----------+---------------------
alterlock | AccessExclusiveLock
pg_toast | AccessExclusiveLock
(2 rows)
commit;
begin; alter table alterlock alter column f2 set storage extended; begin; alter table alterlock alter column f2 set storage extended;
select * from my_locks order by 1; select * from my_locks order by 1;
relname | max_lockmode relname | max_lockmode
...@@ -2006,6 +2016,47 @@ select * from my_locks order by 1; ...@@ -2006,6 +2016,47 @@ select * from my_locks order by 1;
alterlock_pkey | AccessShareLock alterlock_pkey | AccessShareLock
(4 rows) (4 rows)
rollback;
create or replace view my_locks as
select case when c.relname like 'pg_toast%' then 'pg_toast' else c.relname end, max(mode::lockmodes) as max_lockmode
from pg_locks l join pg_class c on l.relation = c.oid
where virtualtransaction = (
select virtualtransaction
from pg_locks
where transactionid = txid_current()::integer)
and locktype = 'relation'
and relnamespace != (select oid from pg_namespace where nspname = 'pg_catalog')
and c.relname = 'my_locks'
group by c.relname;
-- raise exception
alter table my_locks set (autovacuum_enabled = false);
ERROR: unrecognized parameter "autovacuum_enabled"
alter view my_locks set (autovacuum_enabled = false);
ERROR: unrecognized parameter "autovacuum_enabled"
alter table my_locks reset (autovacuum_enabled);
alter view my_locks reset (autovacuum_enabled);
begin;
alter view my_locks set (security_barrier=off);
select * from my_locks order by 1;
relname | max_lockmode
----------+---------------------
my_locks | AccessExclusiveLock
(1 row)
alter view my_locks reset (security_barrier);
rollback;
-- this test intentionally applies the ALTER TABLE command against a view, but
-- uses a view option so we expect this to succeed. This form of SQL is
-- accepted for historical reasons, as shown in the docs for ALTER VIEW
begin;
alter table my_locks set (security_barrier=off);
select * from my_locks order by 1;
relname | max_lockmode
----------+---------------------
my_locks | AccessExclusiveLock
(1 row)
alter table my_locks reset (security_barrier);
rollback; rollback;
-- cleanup -- cleanup
drop table alterlock2; drop table alterlock2;
......
...@@ -1332,6 +1332,11 @@ begin; alter table alterlock alter column f2 set (n_distinct = 1); ...@@ -1332,6 +1332,11 @@ begin; alter table alterlock alter column f2 set (n_distinct = 1);
select * from my_locks order by 1; select * from my_locks order by 1;
rollback; rollback;
-- test that mixing options with different lock levels works as expected
begin; alter table alterlock set (autovacuum_enabled = off, fillfactor = 80);
select * from my_locks order by 1;
commit;
begin; alter table alterlock alter column f2 set storage extended; begin; alter table alterlock alter column f2 set storage extended;
select * from my_locks order by 1; select * from my_locks order by 1;
rollback; rollback;
...@@ -1365,6 +1370,39 @@ alter table alterlock2 validate constraint alterlock2nv; ...@@ -1365,6 +1370,39 @@ alter table alterlock2 validate constraint alterlock2nv;
select * from my_locks order by 1; select * from my_locks order by 1;
rollback; rollback;
create or replace view my_locks as
select case when c.relname like 'pg_toast%' then 'pg_toast' else c.relname end, max(mode::lockmodes) as max_lockmode
from pg_locks l join pg_class c on l.relation = c.oid
where virtualtransaction = (
select virtualtransaction
from pg_locks
where transactionid = txid_current()::integer)
and locktype = 'relation'
and relnamespace != (select oid from pg_namespace where nspname = 'pg_catalog')
and c.relname = 'my_locks'
group by c.relname;
-- raise exception
alter table my_locks set (autovacuum_enabled = false);
alter view my_locks set (autovacuum_enabled = false);
alter table my_locks reset (autovacuum_enabled);
alter view my_locks reset (autovacuum_enabled);
begin;
alter view my_locks set (security_barrier=off);
select * from my_locks order by 1;
alter view my_locks reset (security_barrier);
rollback;
-- this test intentionally applies the ALTER TABLE command against a view, but
-- uses a view option so we expect this to succeed. This form of SQL is
-- accepted for historical reasons, as shown in the docs for ALTER VIEW
begin;
alter table my_locks set (security_barrier=off);
select * from my_locks order by 1;
alter table my_locks reset (security_barrier);
rollback;
-- cleanup -- cleanup
drop table alterlock2; drop table alterlock2;
drop table alterlock; drop table alterlock;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment