mirror of
https://git.postgresql.org/git/postgresql.git
synced 2024-12-15 08:20:16 +08:00
2ed532ee8c
Most error messages about a relkind that was not supported or appropriate for the command was of the pattern "relation \"%s\" is not a table, foreign table, or materialized view" This style can become verbose and tedious to maintain. Moreover, it's not very helpful: If I'm trying to create a comment on a TOAST table, which is not supported, then the information that I could have created a comment on a materialized view is pointless. Instead, write the primary error message shorter and saying more directly that what was attempted is not possible. Then, in the detail message, explain that the operation is not supported for the relkind the object was. To simplify that, add a new function errdetail_relkind_not_supported() that does this. In passing, make use of RELKIND_HAS_STORAGE() where appropriate, instead of listing out the relkinds individually. Reviewed-by: Michael Paquier <michael@paquier.xyz> Reviewed-by: Alvaro Herrera <alvherre@alvh.no-ip.org> Discussion: https://www.postgresql.org/message-id/flat/dc35a398-37d0-75ce-07ea-1dd71d98f8ec@2ndquadrant.com
181 lines
4.5 KiB
Plaintext
181 lines
4.5 KiB
Plaintext
create extension pg_surgery;
|
|
-- create a normal heap table and insert some rows.
|
|
-- use a temp table so that vacuum behavior doesn't depend on global xmin
|
|
create temp table htab (a int);
|
|
insert into htab values (100), (200), (300), (400), (500);
|
|
-- test empty TID array
|
|
select heap_force_freeze('htab'::regclass, ARRAY[]::tid[]);
|
|
heap_force_freeze
|
|
-------------------
|
|
|
|
(1 row)
|
|
|
|
-- nothing should be frozen yet
|
|
select * from htab where xmin = 2;
|
|
a
|
|
---
|
|
(0 rows)
|
|
|
|
-- freeze forcibly
|
|
select heap_force_freeze('htab'::regclass, ARRAY['(0, 4)']::tid[]);
|
|
heap_force_freeze
|
|
-------------------
|
|
|
|
(1 row)
|
|
|
|
-- now we should have one frozen tuple
|
|
select ctid, xmax from htab where xmin = 2;
|
|
ctid | xmax
|
|
-------+------
|
|
(0,4) | 0
|
|
(1 row)
|
|
|
|
-- kill forcibly
|
|
select heap_force_kill('htab'::regclass, ARRAY['(0, 4)']::tid[]);
|
|
heap_force_kill
|
|
-----------------
|
|
|
|
(1 row)
|
|
|
|
-- should be gone now
|
|
select * from htab where ctid = '(0, 4)';
|
|
a
|
|
---
|
|
(0 rows)
|
|
|
|
-- should now be skipped because it's already dead
|
|
select heap_force_kill('htab'::regclass, ARRAY['(0, 4)']::tid[]);
|
|
NOTICE: skipping tid (0, 4) for relation "htab" because it is marked dead
|
|
heap_force_kill
|
|
-----------------
|
|
|
|
(1 row)
|
|
|
|
select heap_force_freeze('htab'::regclass, ARRAY['(0, 4)']::tid[]);
|
|
NOTICE: skipping tid (0, 4) for relation "htab" because it is marked dead
|
|
heap_force_freeze
|
|
-------------------
|
|
|
|
(1 row)
|
|
|
|
-- freeze two TIDs at once while skipping an out-of-range block number
|
|
select heap_force_freeze('htab'::regclass,
|
|
ARRAY['(0, 1)', '(0, 3)', '(1, 1)']::tid[]);
|
|
NOTICE: skipping block 1 for relation "htab" because the block number is out of range
|
|
heap_force_freeze
|
|
-------------------
|
|
|
|
(1 row)
|
|
|
|
-- we should now have two frozen tuples
|
|
select ctid, xmax from htab where xmin = 2;
|
|
ctid | xmax
|
|
-------+------
|
|
(0,1) | 0
|
|
(0,3) | 0
|
|
(2 rows)
|
|
|
|
-- out-of-range TIDs should be skipped
|
|
select heap_force_freeze('htab'::regclass, ARRAY['(0, 0)', '(0, 6)']::tid[]);
|
|
NOTICE: skipping tid (0, 0) for relation "htab" because the item number is out of range
|
|
NOTICE: skipping tid (0, 6) for relation "htab" because the item number is out of range
|
|
heap_force_freeze
|
|
-------------------
|
|
|
|
(1 row)
|
|
|
|
-- set up a new table with a redirected line pointer
|
|
-- use a temp table so that vacuum behavior doesn't depend on global xmin
|
|
create temp table htab2(a int);
|
|
insert into htab2 values (100);
|
|
update htab2 set a = 200;
|
|
vacuum htab2;
|
|
-- redirected TIDs should be skipped
|
|
select heap_force_kill('htab2'::regclass, ARRAY['(0, 1)']::tid[]);
|
|
NOTICE: skipping tid (0, 1) for relation "htab2" because it redirects to item 2
|
|
heap_force_kill
|
|
-----------------
|
|
|
|
(1 row)
|
|
|
|
-- now create an unused line pointer
|
|
select ctid from htab2;
|
|
ctid
|
|
-------
|
|
(0,2)
|
|
(1 row)
|
|
|
|
update htab2 set a = 300;
|
|
select ctid from htab2;
|
|
ctid
|
|
-------
|
|
(0,3)
|
|
(1 row)
|
|
|
|
vacuum freeze htab2;
|
|
-- unused TIDs should be skipped
|
|
select heap_force_kill('htab2'::regclass, ARRAY['(0, 2)']::tid[]);
|
|
NOTICE: skipping tid (0, 2) for relation "htab2" because it is marked unused
|
|
heap_force_kill
|
|
-----------------
|
|
|
|
(1 row)
|
|
|
|
-- multidimensional TID array should be rejected
|
|
select heap_force_kill('htab2'::regclass, ARRAY[['(0, 2)']]::tid[]);
|
|
ERROR: argument must be empty or one-dimensional array
|
|
-- TID array with nulls should be rejected
|
|
select heap_force_kill('htab2'::regclass, ARRAY[NULL]::tid[]);
|
|
ERROR: array must not contain nulls
|
|
-- but we should be able to kill the one tuple we have
|
|
select heap_force_kill('htab2'::regclass, ARRAY['(0, 3)']::tid[]);
|
|
heap_force_kill
|
|
-----------------
|
|
|
|
(1 row)
|
|
|
|
-- materialized view.
|
|
-- note that we don't commit the transaction, so autovacuum can't interfere.
|
|
begin;
|
|
create materialized view mvw as select a from generate_series(1, 3) a;
|
|
select * from mvw where xmin = 2;
|
|
a
|
|
---
|
|
(0 rows)
|
|
|
|
select heap_force_freeze('mvw'::regclass, ARRAY['(0, 3)']::tid[]);
|
|
heap_force_freeze
|
|
-------------------
|
|
|
|
(1 row)
|
|
|
|
select * from mvw where xmin = 2;
|
|
a
|
|
---
|
|
3
|
|
(1 row)
|
|
|
|
select heap_force_kill('mvw'::regclass, ARRAY['(0, 3)']::tid[]);
|
|
heap_force_kill
|
|
-----------------
|
|
|
|
(1 row)
|
|
|
|
select * from mvw where ctid = '(0, 3)';
|
|
a
|
|
---
|
|
(0 rows)
|
|
|
|
rollback;
|
|
-- check that it fails on an unsupported relkind
|
|
create view vw as select 1;
|
|
select heap_force_kill('vw'::regclass, ARRAY['(0, 1)']::tid[]);
|
|
ERROR: cannot operate on relation "vw"
|
|
DETAIL: This operation is not supported for views.
|
|
select heap_force_freeze('vw'::regclass, ARRAY['(0, 1)']::tid[]);
|
|
ERROR: cannot operate on relation "vw"
|
|
DETAIL: This operation is not supported for views.
|
|
-- cleanup.
|
|
drop view vw;
|
|
drop extension pg_surgery;
|