Commit 8e18d04d authored by Robert Haas's avatar Robert Haas

Refine our definition of what constitutes a system relation.

Although user-defined relations can't be directly created in
pg_catalog, it's possible for them to end up there, because you can
create them in some other schema and then use ALTER TABLE .. SET SCHEMA
to move them there.  Previously, such relations couldn't afterwards
be manipulated, because IsSystemRelation()/IsSystemClass() rejected
all attempts to modify objects in the pg_catalog schema, regardless
of their origin.  With this patch, they now reject only those
objects in pg_catalog which were created at initdb-time, allowing
most operations on user-created tables in pg_catalog to proceed
normally.

This patch also adds new functions IsCatalogRelation() and
IsCatalogClass(), which is similar to IsSystemRelation() and
IsSystemClass() but with a slightly narrower definition: only TOAST
tables of system catalogs are included, rather than *all* TOAST tables.
This is currently used only for making decisions about when
invalidation messages need to be sent, but upcoming logical decoding
patches will find other uses for this information.

Andres Freund, with some modifications by me.
parent 2fe69cac
...@@ -2465,7 +2465,7 @@ heap_multi_insert(Relation relation, HeapTuple *tuples, int ntuples, ...@@ -2465,7 +2465,7 @@ heap_multi_insert(Relation relation, HeapTuple *tuples, int ntuples,
* because the heaptuples data structure is all in local memory, not in * because the heaptuples data structure is all in local memory, not in
* the shared buffer. * the shared buffer.
*/ */
if (IsSystemRelation(relation)) if (IsCatalogRelation(relation))
{ {
for (i = 0; i < ntuples; i++) for (i = 0; i < ntuples; i++)
CacheInvalidateHeapTuple(relation, heaptuples[i], NULL); CacheInvalidateHeapTuple(relation, heaptuples[i], NULL);
......
...@@ -3628,7 +3628,7 @@ pg_class_aclmask(Oid table_oid, Oid roleid, ...@@ -3628,7 +3628,7 @@ pg_class_aclmask(Oid table_oid, Oid roleid,
* themselves. ACL_USAGE is if we ever have system sequences. * themselves. ACL_USAGE is if we ever have system sequences.
*/ */
if ((mask & (ACL_INSERT | ACL_UPDATE | ACL_DELETE | ACL_TRUNCATE | ACL_USAGE)) && if ((mask & (ACL_INSERT | ACL_UPDATE | ACL_DELETE | ACL_TRUNCATE | ACL_USAGE)) &&
IsSystemClass(classForm) && IsSystemClass(table_oid, classForm) &&
classForm->relkind != RELKIND_VIEW && classForm->relkind != RELKIND_VIEW &&
!has_rolcatupdate(roleid) && !has_rolcatupdate(roleid) &&
!allowSystemTableMods) !allowSystemTableMods)
......
...@@ -97,22 +97,20 @@ GetDatabasePath(Oid dbNode, Oid spcNode) ...@@ -97,22 +97,20 @@ GetDatabasePath(Oid dbNode, Oid spcNode)
/* /*
* IsSystemRelation * IsSystemRelation
* True iff the relation is a system catalog relation. * True iff the relation is either a system catalog or toast table.
* By a system catalog, we mean one that created in the pg_catalog schema
* during initdb. User-created relations in pg_catalog don't count as
* system catalogs.
* *
* NB: TOAST relations are considered system relations by this test * NB: TOAST relations are considered system relations by this test
* for compatibility with the old IsSystemRelationName function. * for compatibility with the old IsSystemRelationName function.
* This is appropriate in many places but not all. Where it's not, * This is appropriate in many places but not all. Where it's not,
* also check IsToastRelation. * also check IsToastRelation or use IsCatalogRelation().
*
* We now just test if the relation is in the system catalog namespace;
* so it's no longer necessary to forbid user relations from having
* names starting with pg_.
*/ */
bool bool
IsSystemRelation(Relation relation) IsSystemRelation(Relation relation)
{ {
return IsSystemNamespace(RelationGetNamespace(relation)) || return IsSystemClass(RelationGetRelid(relation), relation->rd_rel);
IsToastNamespace(RelationGetNamespace(relation));
} }
/* /*
...@@ -122,12 +120,60 @@ IsSystemRelation(Relation relation) ...@@ -122,12 +120,60 @@ IsSystemRelation(Relation relation)
* search pg_class directly. * search pg_class directly.
*/ */
bool bool
IsSystemClass(Form_pg_class reltuple) IsSystemClass(Oid relid, Form_pg_class reltuple)
{
return IsToastClass(reltuple) || IsCatalogClass(relid, reltuple);
}
/*
* IsCatalogRelation
* True iff the relation is a system catalog, or the toast table for
* a system catalog. By a system catalog, we mean one that created
* in the pg_catalog schema during initdb. As with IsSystemRelation(),
* user-created relations in pg_catalog don't count as system catalogs.
*
* Note that IsSystemRelation() returns true for ALL toast relations,
* but this function returns true only for toast relations of system
* catalogs.
*/
bool
IsCatalogRelation(Relation relation)
{
return IsCatalogClass(RelationGetRelid(relation), relation->rd_rel);
}
/*
* IsCatalogClass
* True iff the relation is a system catalog relation.
*
* Check IsCatalogRelation() for details.
*/
bool
IsCatalogClass(Oid relid, Form_pg_class reltuple)
{ {
Oid relnamespace = reltuple->relnamespace; Oid relnamespace = reltuple->relnamespace;
return IsSystemNamespace(relnamespace) || /*
IsToastNamespace(relnamespace); * Never consider relations outside pg_catalog/pg_toast to be catalog
* relations.
*/
if (!IsSystemNamespace(relnamespace) && !IsToastNamespace(relnamespace))
return false;
/* ----
* Check whether the oid was assigned during initdb, when creating the
* initial template database. Minus the relations in information_schema
* excluded above, these are integral part of the system.
* We could instead check whether the relation is pinned in pg_depend, but
* this is noticeably cheaper and doesn't require catalog access.
*
* This test is safe since even a oid wraparound will preserve this
* property (c.f. GetNewObjectId()) and it has the advantage that it works
* correctly even if a user decides to create a relation in the pg_catalog
* namespace.
* ----
*/
return relid < FirstNormalObjectId;
} }
/* /*
......
...@@ -256,10 +256,17 @@ heap_create(const char *relname, ...@@ -256,10 +256,17 @@ heap_create(const char *relname,
Assert(OidIsValid(relid)); Assert(OidIsValid(relid));
/* /*
* sanity checks * Don't allow creating relations in pg_catalog directly, even though it
* is allowed to move user defined relations there. Semantics with search
* paths including pg_catalog are too confusing for now.
*
* But allow creating indexes on relations in pg_catalog even if
* allow_system_table_mods = off, upper layers already guarantee it's on a
* user defined relation, not a system one.
*/ */
if (!allow_system_table_mods && if (!allow_system_table_mods &&
(IsSystemNamespace(relnamespace) || IsToastNamespace(relnamespace)) && ((IsSystemNamespace(relnamespace) && relkind != RELKIND_INDEX) ||
IsToastNamespace(relnamespace)) &&
IsNormalProcessingMode()) IsNormalProcessingMode())
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE), (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
......
...@@ -1354,7 +1354,7 @@ swap_relation_files(Oid r1, Oid r2, bool target_is_pg_class, ...@@ -1354,7 +1354,7 @@ swap_relation_files(Oid r1, Oid r2, bool target_is_pg_class,
* ones the dependency changes would change. It's too late to be * ones the dependency changes would change. It's too late to be
* making any data changes to the target catalog. * making any data changes to the target catalog.
*/ */
if (IsSystemClass(relform1)) if (IsSystemClass(r1, relform1))
elog(ERROR, "cannot swap toast files by links for system catalogs"); elog(ERROR, "cannot swap toast files by links for system catalogs");
/* Delete old dependencies */ /* Delete old dependencies */
......
...@@ -1824,6 +1824,7 @@ ReindexDatabase(const char *databaseName, bool do_system, bool do_user) ...@@ -1824,6 +1824,7 @@ ReindexDatabase(const char *databaseName, bool do_system, bool do_user)
while ((tuple = heap_getnext(scan, ForwardScanDirection)) != NULL) while ((tuple = heap_getnext(scan, ForwardScanDirection)) != NULL)
{ {
Form_pg_class classtuple = (Form_pg_class) GETSTRUCT(tuple); Form_pg_class classtuple = (Form_pg_class) GETSTRUCT(tuple);
Oid relid = HeapTupleGetOid(tuple);
if (classtuple->relkind != RELKIND_RELATION && if (classtuple->relkind != RELKIND_RELATION &&
classtuple->relkind != RELKIND_MATVIEW) classtuple->relkind != RELKIND_MATVIEW)
...@@ -1835,7 +1836,7 @@ ReindexDatabase(const char *databaseName, bool do_system, bool do_user) ...@@ -1835,7 +1836,7 @@ ReindexDatabase(const char *databaseName, bool do_system, bool do_user)
continue; continue;
/* Check user/system classification, and optionally skip */ /* Check user/system classification, and optionally skip */
if (IsSystemClass(classtuple)) if (IsSystemClass(relid, classtuple))
{ {
if (!do_system) if (!do_system)
continue; continue;
...@@ -1850,7 +1851,7 @@ ReindexDatabase(const char *databaseName, bool do_system, bool do_user) ...@@ -1850,7 +1851,7 @@ ReindexDatabase(const char *databaseName, bool do_system, bool do_user)
continue; /* got it already */ continue; /* got it already */
old = MemoryContextSwitchTo(private_context); old = MemoryContextSwitchTo(private_context);
relids = lappend_oid(relids, HeapTupleGetOid(tuple)); relids = lappend_oid(relids, relid);
MemoryContextSwitchTo(old); MemoryContextSwitchTo(old);
} }
heap_endscan(scan); heap_endscan(scan);
......
...@@ -910,7 +910,7 @@ RangeVarCallbackForDropRelation(const RangeVar *rel, Oid relOid, Oid oldRelOid, ...@@ -910,7 +910,7 @@ RangeVarCallbackForDropRelation(const RangeVar *rel, Oid relOid, Oid oldRelOid,
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_CLASS, aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_CLASS,
rel->relname); rel->relname);
if (!allowSystemTableMods && IsSystemClass(classform)) if (!allowSystemTableMods && IsSystemClass(relOid, classform))
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE), (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
errmsg("permission denied: \"%s\" is a system catalog", errmsg("permission denied: \"%s\" is a system catalog",
...@@ -2105,7 +2105,7 @@ renameatt_check(Oid myrelid, Form_pg_class classform, bool recursing) ...@@ -2105,7 +2105,7 @@ renameatt_check(Oid myrelid, Form_pg_class classform, bool recursing)
if (!pg_class_ownercheck(myrelid, GetUserId())) if (!pg_class_ownercheck(myrelid, GetUserId()))
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_CLASS, aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_CLASS,
NameStr(classform->relname)); NameStr(classform->relname));
if (!allowSystemTableMods && IsSystemClass(classform)) if (!allowSystemTableMods && IsSystemClass(myrelid, classform))
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE), (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
errmsg("permission denied: \"%s\" is a system catalog", errmsg("permission denied: \"%s\" is a system catalog",
...@@ -10872,7 +10872,7 @@ RangeVarCallbackForAlterRelation(const RangeVar *rv, Oid relid, Oid oldrelid, ...@@ -10872,7 +10872,7 @@ RangeVarCallbackForAlterRelation(const RangeVar *rv, Oid relid, Oid oldrelid,
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_CLASS, rv->relname); aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_CLASS, rv->relname);
/* No system table modifications unless explicitly allowed. */ /* No system table modifications unless explicitly allowed. */
if (!allowSystemTableMods && IsSystemClass(classform)) if (!allowSystemTableMods && IsSystemClass(relid, classform))
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE), (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
errmsg("permission denied: \"%s\" is a system catalog", errmsg("permission denied: \"%s\" is a system catalog",
......
...@@ -1174,7 +1174,7 @@ RangeVarCallbackForRenameTrigger(const RangeVar *rv, Oid relid, Oid oldrelid, ...@@ -1174,7 +1174,7 @@ RangeVarCallbackForRenameTrigger(const RangeVar *rv, Oid relid, Oid oldrelid,
/* you must own the table to rename one of its triggers */ /* you must own the table to rename one of its triggers */
if (!pg_class_ownercheck(relid, GetUserId())) if (!pg_class_ownercheck(relid, GetUserId()))
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_CLASS, rv->relname); aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_CLASS, rv->relname);
if (!allowSystemTableMods && IsSystemClass(form)) if (!allowSystemTableMods && IsSystemClass(relid, form))
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE), (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
errmsg("permission denied: \"%s\" is a system catalog", errmsg("permission denied: \"%s\" is a system catalog",
......
...@@ -128,7 +128,7 @@ get_relation_info(PlannerInfo *root, Oid relationObjectId, bool inhparent, ...@@ -128,7 +128,7 @@ get_relation_info(PlannerInfo *root, Oid relationObjectId, bool inhparent,
* Don't bother with indexes for an inheritance parent, either. * Don't bother with indexes for an inheritance parent, either.
*/ */
if (inhparent || if (inhparent ||
(IgnoreSystemIndexes && IsSystemClass(relation->rd_rel))) (IgnoreSystemIndexes && IsSystemRelation(relation)))
hasindex = false; hasindex = false;
else else
hasindex = relation->rd_rel->relhasindex; hasindex = relation->rd_rel->relhasindex;
......
...@@ -858,7 +858,7 @@ RangeVarCallbackForRenameRule(const RangeVar *rv, Oid relid, Oid oldrelid, ...@@ -858,7 +858,7 @@ RangeVarCallbackForRenameRule(const RangeVar *rv, Oid relid, Oid oldrelid,
(errcode(ERRCODE_WRONG_OBJECT_TYPE), (errcode(ERRCODE_WRONG_OBJECT_TYPE),
errmsg("\"%s\" is not a table or view", rv->relname))); errmsg("\"%s\" is not a table or view", rv->relname)));
if (!allowSystemTableMods && IsSystemClass(form)) if (!allowSystemTableMods && IsSystemClass(relid, form))
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE), (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
errmsg("permission denied: \"%s\" is a system catalog", errmsg("permission denied: \"%s\" is a system catalog",
......
...@@ -110,7 +110,7 @@ CheckRelationOwnership(RangeVar *rel, bool noCatalogs) ...@@ -110,7 +110,7 @@ CheckRelationOwnership(RangeVar *rel, bool noCatalogs)
if (noCatalogs) if (noCatalogs)
{ {
if (!allowSystemTableMods && if (!allowSystemTableMods &&
IsSystemClass((Form_pg_class) GETSTRUCT(tuple))) IsSystemClass(relOid, (Form_pg_class) GETSTRUCT(tuple)))
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE), (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
errmsg("permission denied: \"%s\" is a system catalog", errmsg("permission denied: \"%s\" is a system catalog",
......
...@@ -1040,15 +1040,15 @@ CacheInvalidateHeapTuple(Relation relation, ...@@ -1040,15 +1040,15 @@ CacheInvalidateHeapTuple(Relation relation,
/* /*
* We only need to worry about invalidation for tuples that are in system * We only need to worry about invalidation for tuples that are in system
* relations; user-relation tuples are never in catcaches and can't affect * catalogs; user-relation tuples are never in catcaches and can't affect
* the relcache either. * the relcache either.
*/ */
if (!IsSystemRelation(relation)) if (!IsCatalogRelation(relation))
return; return;
/* /*
* TOAST tuples can likewise be ignored here. Note that TOAST tables are * IsCatalogRelation() will return true for TOAST tables of system
* considered system relations so they are not filtered by the above test. * catalogs, but we don't care about those, either.
*/ */
if (IsToastRelation(relation)) if (IsToastRelation(relation))
return; return;
......
...@@ -25,9 +25,11 @@ extern char *GetDatabasePath(Oid dbNode, Oid spcNode); ...@@ -25,9 +25,11 @@ extern char *GetDatabasePath(Oid dbNode, Oid spcNode);
extern bool IsSystemRelation(Relation relation); extern bool IsSystemRelation(Relation relation);
extern bool IsToastRelation(Relation relation); extern bool IsToastRelation(Relation relation);
extern bool IsCatalogRelation(Relation relation);
extern bool IsSystemClass(Form_pg_class reltuple); extern bool IsSystemClass(Oid relid, Form_pg_class reltuple);
extern bool IsToastClass(Form_pg_class reltuple); extern bool IsToastClass(Form_pg_class reltuple);
extern bool IsCatalogClass(Oid relid, Form_pg_class reltuple);
extern bool IsSystemNamespace(Oid namespaceId); extern bool IsSystemNamespace(Oid namespaceId);
extern bool IsToastNamespace(Oid namespaceId); extern bool IsToastNamespace(Oid namespaceId);
......
...@@ -2337,3 +2337,37 @@ FROM ( ...@@ -2337,3 +2337,37 @@ FROM (
0 | t 0 | t
(1 row) (1 row)
-- Checks on creating and manipulation of user defined relations in
-- pg_catalog.
--
-- XXX: It would be useful to add checks around trying to manipulate
-- catalog tables, but that might have ugly consequences when run
-- against an existing server with allow_system_table_mods = on.
SHOW allow_system_table_mods;
allow_system_table_mods
-------------------------
off
(1 row)
-- disallowed because of search_path issues with pg_dump
CREATE TABLE pg_catalog.new_system_table();
ERROR: permission denied to create "pg_catalog.new_system_table"
DETAIL: System catalog modifications are currently disallowed.
-- instead create in public first, move to catalog
CREATE TABLE new_system_table(id serial primary key, othercol text);
ALTER TABLE new_system_table SET SCHEMA pg_catalog;
-- XXX: it's currently impossible to move relations out of pg_catalog
ALTER TABLE new_system_table SET SCHEMA public;
ERROR: cannot remove dependency on schema pg_catalog because it is a system object
-- move back, will currently error out, already there
ALTER TABLE new_system_table SET SCHEMA pg_catalog;
ERROR: table new_system_table is already in schema "pg_catalog"
ALTER TABLE new_system_table RENAME TO old_system_table;
CREATE INDEX old_system_table__othercol ON old_system_table (othercol);
INSERT INTO old_system_table(othercol) VALUES ('somedata'), ('otherdata');
UPDATE old_system_table SET id = -id;
DELETE FROM old_system_table WHERE othercol = 'somedata';
TRUNCATE old_system_table;
ALTER TABLE old_system_table DROP CONSTRAINT new_system_table_pkey;
ALTER TABLE old_system_table DROP COLUMN othercol;
DROP TABLE old_system_table;
...@@ -1567,3 +1567,31 @@ FROM ( ...@@ -1567,3 +1567,31 @@ FROM (
FROM pg_class FROM pg_class
WHERE relkind IN ('r', 'i', 'S', 't', 'm') WHERE relkind IN ('r', 'i', 'S', 't', 'm')
) mapped; ) mapped;
-- Checks on creating and manipulation of user defined relations in
-- pg_catalog.
--
-- XXX: It would be useful to add checks around trying to manipulate
-- catalog tables, but that might have ugly consequences when run
-- against an existing server with allow_system_table_mods = on.
SHOW allow_system_table_mods;
-- disallowed because of search_path issues with pg_dump
CREATE TABLE pg_catalog.new_system_table();
-- instead create in public first, move to catalog
CREATE TABLE new_system_table(id serial primary key, othercol text);
ALTER TABLE new_system_table SET SCHEMA pg_catalog;
-- XXX: it's currently impossible to move relations out of pg_catalog
ALTER TABLE new_system_table SET SCHEMA public;
-- move back, will currently error out, already there
ALTER TABLE new_system_table SET SCHEMA pg_catalog;
ALTER TABLE new_system_table RENAME TO old_system_table;
CREATE INDEX old_system_table__othercol ON old_system_table (othercol);
INSERT INTO old_system_table(othercol) VALUES ('somedata'), ('otherdata');
UPDATE old_system_table SET id = -id;
DELETE FROM old_system_table WHERE othercol = 'somedata';
TRUNCATE old_system_table;
ALTER TABLE old_system_table DROP CONSTRAINT new_system_table_pkey;
ALTER TABLE old_system_table DROP COLUMN othercol;
DROP TABLE old_system_table;
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