0001-VACUUM-FULL-silently-NULL-out-fast-default-columns.patch
application/octet-stream
Filename: 0001-VACUUM-FULL-silently-NULL-out-fast-default-columns.patch
Type: application/octet-stream
Part: 0
From 1a3e25855452137d6d7c6db89a243ff8089e939f Mon Sep 17 00:00:00 2001
From: Satya Narlapuram <satyanarlapuram@gmail.com>
Date: Mon, 4 May 2026 08:52:50 +0000
Subject: [PATCH] VACUUM FULL silently NULL out fast-default columns
Commit 28d534e2 introduced reform_tuple() in heapam_handler.c with a
fast path that returns the source tuple verbatim when no dropped
columns require fixing up. The check ignores tuples that are short
due to attmissingval (the fast-default mechanism).
After the rewrite, finish_heap_swap() calls RelationClearMissing(),
clearing the catalog metadata that was the only source of those
values. Short tuples then read as NULL (or the type's zero value if
NOT NULL is in effect, also bypassing CHECK constraints). Affects
VACUUM FULL, CLUSTER, REPACK, and REPACK (CONCURRENTLY).
Force reform when the source tuple is shorter than the new tuple
descriptor so heap_deform_tuple() materializes the missing values
before heap_form_tuple() rebuilds a full-width tuple.
---
src/backend/access/heap/heapam_handler.c | 12 +++++++-
src/test/regress/expected/fast_default.out | 36 ++++++++++++++++++++++
src/test/regress/sql/fast_default.sql | 16 ++++++++++
3 files changed, 63 insertions(+), 1 deletion(-)
diff --git a/src/backend/access/heap/heapam_handler.c b/src/backend/access/heap/heapam_handler.c
index 20d3b46e..12b27776 100644
--- a/src/backend/access/heap/heapam_handler.c
+++ b/src/backend/access/heap/heapam_handler.c
@@ -2411,8 +2411,18 @@ reform_tuple(HeapTuple tuple, Relation OldHeap, Relation NewHeap,
TupleDesc newTupDesc = RelationGetDescr(NewHeap);
bool needs_reform = false;
+ /*
+ * If the tuple is short due to missing-value (fast-default) attributes,
+ * we must materialize them now: finish_heap_swap() will subsequently call
+ * RelationClearMissing() on the new relation, dropping the catalog
+ * metadata that is the only source of those values. Without reforming,
+ * those columns would silently become NULL after the rewrite.
+ */
+ if (HeapTupleHeaderGetNatts(tuple->t_data) < newTupDesc->natts)
+ needs_reform = true;
+
/* Skip work if the tuple doesn't need any attributes changed */
- for (int i = 0; i < newTupDesc->natts; i++)
+ for (int i = 0; !needs_reform && i < newTupDesc->natts; i++)
{
if (TupleDescCompactAttr(newTupDesc, i)->attisdropped &&
!heap_attisnull(tuple, i + 1, newTupDesc))
diff --git a/src/test/regress/expected/fast_default.out b/src/test/regress/expected/fast_default.out
index bd142ed4..5813f1c6 100644
--- a/src/test/regress/expected/fast_default.out
+++ b/src/test/regress/expected/fast_default.out
@@ -969,6 +969,42 @@ SELECT count(*)
0
(1 row)
+-- Verify that table-rewriting maintenance commands preserve attmissingval
+-- columns.
+CREATE TABLE t (id int PRIMARY KEY);
+INSERT INTO t SELECT generate_series(1, 3);
+ALTER TABLE t ADD COLUMN a int DEFAULT 42;
+ALTER TABLE t ADD COLUMN b int NOT NULL DEFAULT 7 CHECK (b > 0);
+VACUUM FULL t;
+SELECT * FROM t ORDER BY id;
+ id | a | b
+----+----+---
+ 1 | 42 | 7
+ 2 | 42 | 7
+ 3 | 42 | 7
+(3 rows)
+
+ALTER TABLE t ADD COLUMN c text DEFAULT 'hello';
+CLUSTER t USING t_pkey;
+SELECT * FROM t ORDER BY id;
+ id | a | b | c
+----+----+---+-------
+ 1 | 42 | 7 | hello
+ 2 | 42 | 7 | hello
+ 3 | 42 | 7 | hello
+(3 rows)
+
+ALTER TABLE t ADD COLUMN d int DEFAULT 99;
+REPACK t;
+SELECT * FROM t ORDER BY id;
+ id | a | b | c | d
+----+----+---+-------+----
+ 1 | 42 | 7 | hello | 99
+ 2 | 42 | 7 | hello | 99
+ 3 | 42 | 7 | hello | 99
+(3 rows)
+
+DROP TABLE t;
-- cleanup
DROP FOREIGN TABLE ft1;
DROP SERVER s0;
diff --git a/src/test/regress/sql/fast_default.sql b/src/test/regress/sql/fast_default.sql
index 8b31d317..e5d9a3d2 100644
--- a/src/test/regress/sql/fast_default.sql
+++ b/src/test/regress/sql/fast_default.sql
@@ -653,6 +653,22 @@ SELECT count(*)
WHERE attrelid = 'ft1'::regclass AND
(attmissingval IS NOT NULL OR atthasmissing);
+-- Verify that table-rewriting maintenance commands preserve attmissingval
+-- columns.
+CREATE TABLE t (id int PRIMARY KEY);
+INSERT INTO t SELECT generate_series(1, 3);
+ALTER TABLE t ADD COLUMN a int DEFAULT 42;
+ALTER TABLE t ADD COLUMN b int NOT NULL DEFAULT 7 CHECK (b > 0);
+VACUUM FULL t;
+SELECT * FROM t ORDER BY id;
+ALTER TABLE t ADD COLUMN c text DEFAULT 'hello';
+CLUSTER t USING t_pkey;
+SELECT * FROM t ORDER BY id;
+ALTER TABLE t ADD COLUMN d int DEFAULT 99;
+REPACK t;
+SELECT * FROM t ORDER BY id;
+DROP TABLE t;
+
-- cleanup
DROP FOREIGN TABLE ft1;
DROP SERVER s0;
--
2.43.0