Commit ca410302 authored by Alvaro Herrera's avatar Alvaro Herrera

Fix tablespace handling for partitioned tables

When partitioned tables were introduced, we failed to realize that by
copying the tablespace handling for other relation kinds with no
physical storage we were causing the secondary effect that their
partitions would not automatically inherit the tablespace setting.  This
is surprising and unhelpful, so change it to adopt the behavior
introduced in pg11 (commit 33e6c34c3267) for partitioned indexes: the
parent relation remembers the tablespace specification, which is then
used for any new partitions that don't declare one.

Because this commit changes behavior of the TABLESPACE clause for
partitioned tables (it's no longer a no-op), it is not backpatched.

Author: David Rowley, Álvaro Herrera
Reviewed-by: Michael Paquier
Discussion: https://postgr.es/m/CAKJS1f9SxVzqDrGD1teosFd6jBMM0UEaa14_8mRvcWE19Tu0hA@mail.gmail.com
parent 1e6240a3
...@@ -1216,7 +1216,11 @@ WITH ( MODULUS <replaceable class="parameter">numeric_literal</replaceable>, REM ...@@ -1216,7 +1216,11 @@ WITH ( MODULUS <replaceable class="parameter">numeric_literal</replaceable>, REM
of the tablespace in which the new table is to be created. of the tablespace in which the new table is to be created.
If not specified, If not specified,
<xref linkend="guc-default-tablespace"/> is consulted, or <xref linkend="guc-default-tablespace"/> is consulted, or
<xref linkend="guc-temp-tablespaces"/> if the table is temporary. <xref linkend="guc-temp-tablespaces"/> if the table is temporary. For
partitioned tables, since no storage is required for the table itself,
the tablespace specified here only serves to mark the default tablespace
for any newly created partitions when no other tablespace is explicitly
specified.
</para> </para>
</listitem> </listitem>
</varlistentry> </varlistentry>
......
...@@ -333,7 +333,6 @@ heap_create(const char *relname, ...@@ -333,7 +333,6 @@ heap_create(const char *relname,
case RELKIND_VIEW: case RELKIND_VIEW:
case RELKIND_COMPOSITE_TYPE: case RELKIND_COMPOSITE_TYPE:
case RELKIND_FOREIGN_TABLE: case RELKIND_FOREIGN_TABLE:
case RELKIND_PARTITIONED_TABLE:
create_storage = false; create_storage = false;
/* /*
...@@ -343,10 +342,11 @@ heap_create(const char *relname, ...@@ -343,10 +342,11 @@ heap_create(const char *relname,
reltablespace = InvalidOid; reltablespace = InvalidOid;
break; break;
case RELKIND_PARTITIONED_TABLE:
case RELKIND_PARTITIONED_INDEX: case RELKIND_PARTITIONED_INDEX:
/* /*
* Preserve tablespace so that it's used as tablespace for indexes * For partitioned tables and indexes, preserve tablespace so that
* on future partitions. * it's used as the tablespace for future partitions.
*/ */
create_storage = false; create_storage = false;
break; break;
......
...@@ -305,7 +305,7 @@ static void truncate_check_activity(Relation rel); ...@@ -305,7 +305,7 @@ static void truncate_check_activity(Relation rel);
static void RangeVarCallbackForTruncate(const RangeVar *relation, static void RangeVarCallbackForTruncate(const RangeVar *relation,
Oid relId, Oid oldRelId, void *arg); Oid relId, Oid oldRelId, void *arg);
static List *MergeAttributes(List *schema, List *supers, char relpersistence, static List *MergeAttributes(List *schema, List *supers, char relpersistence,
bool is_partition, List **supOids, List **supconstr); bool is_partition, List **supconstr);
static bool MergeCheckConstraint(List *constraints, char *name, Node *expr); static bool MergeCheckConstraint(List *constraints, char *name, Node *expr);
static void MergeAttributesIntoExisting(Relation child_rel, Relation parent_rel); static void MergeAttributesIntoExisting(Relation child_rel, Relation parent_rel);
static void MergeConstraintsIntoExisting(Relation child_rel, Relation parent_rel); static void MergeConstraintsIntoExisting(Relation child_rel, Relation parent_rel);
...@@ -446,7 +446,7 @@ static bool ATPrepChangePersistence(Relation rel, bool toLogged); ...@@ -446,7 +446,7 @@ static bool ATPrepChangePersistence(Relation rel, bool toLogged);
static void ATPrepSetTableSpace(AlteredTableInfo *tab, Relation rel, static void ATPrepSetTableSpace(AlteredTableInfo *tab, Relation rel,
const char *tablespacename, LOCKMODE lockmode); const char *tablespacename, LOCKMODE lockmode);
static void ATExecSetTableSpace(Oid tableOid, Oid newTableSpace, LOCKMODE lockmode); static void ATExecSetTableSpace(Oid tableOid, Oid newTableSpace, LOCKMODE lockmode);
static void ATExecPartedIdxSetTableSpace(Relation rel, Oid newTableSpace); static void ATExecSetTableSpaceNoStorage(Relation rel, Oid newTableSpace);
static void ATExecSetRelOptions(Relation rel, List *defList, static void ATExecSetRelOptions(Relation rel, List *defList,
AlterTableType operation, AlterTableType operation,
LOCKMODE lockmode); LOCKMODE lockmode);
...@@ -536,6 +536,7 @@ DefineRelation(CreateStmt *stmt, char relkind, Oid ownerId, ...@@ -536,6 +536,7 @@ DefineRelation(CreateStmt *stmt, char relkind, Oid ownerId,
static char *validnsps[] = HEAP_RELOPT_NAMESPACES; static char *validnsps[] = HEAP_RELOPT_NAMESPACES;
Oid ofTypeId; Oid ofTypeId;
ObjectAddress address; ObjectAddress address;
LOCKMODE parentLockmode;
/* /*
* Truncate relname to appropriate length (probably a waste of time, as * Truncate relname to appropriate length (probably a waste of time, as
...@@ -580,6 +581,46 @@ DefineRelation(CreateStmt *stmt, char relkind, Oid ownerId, ...@@ -580,6 +581,46 @@ DefineRelation(CreateStmt *stmt, char relkind, Oid ownerId,
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE), (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
errmsg("cannot create temporary table within security-restricted operation"))); errmsg("cannot create temporary table within security-restricted operation")));
/*
* Determine the lockmode to use when scanning parents. A self-exclusive
* lock is needed here.
*
* For regular inheritance, if two backends attempt to add children to the
* same parent simultaneously, and that parent has no pre-existing
* children, then both will attempt to update the parent's relhassubclass
* field, leading to a "tuple concurrently updated" error. Also, this
* interlocks against a concurrent ANALYZE on the parent table, which
* might otherwise be attempting to clear the parent's relhassubclass
* field, if its previous children were recently dropped.
*
* If the child table is a partition, then we instead grab an exclusive
* lock on the parent because its partition descriptor will be changed by
* addition of the new partition.
*/
parentLockmode = (stmt->partbound != NULL ? AccessExclusiveLock :
ShareUpdateExclusiveLock);
/* Determine the list of OIDs of the parents. */
inheritOids = NIL;
foreach(listptr, stmt->inhRelations)
{
RangeVar *rv = (RangeVar *) lfirst(listptr);
Oid parentOid;
parentOid = RangeVarGetRelid(rv, parentLockmode, false);
/*
* Reject duplications in the list of parents.
*/
if (list_member_oid(inheritOids, parentOid))
ereport(ERROR,
(errcode(ERRCODE_DUPLICATE_TABLE),
errmsg("relation \"%s\" would be inherited from more than once",
get_rel_name(parentOid))));
inheritOids = lappend_oid(inheritOids, parentOid);
}
/* /*
* Select tablespace to use. If not specified, use default tablespace * Select tablespace to use. If not specified, use default tablespace
* (which may in turn default to database's default). * (which may in turn default to database's default).
...@@ -588,6 +629,25 @@ DefineRelation(CreateStmt *stmt, char relkind, Oid ownerId, ...@@ -588,6 +629,25 @@ DefineRelation(CreateStmt *stmt, char relkind, Oid ownerId,
{ {
tablespaceId = get_tablespace_oid(stmt->tablespacename, false); tablespaceId = get_tablespace_oid(stmt->tablespacename, false);
} }
else if (stmt->partbound)
{
HeapTuple tup;
/*
* For partitions, when no other tablespace is specified, we default
* the tablespace to the parent partitioned table's.
*/
Assert(list_length(inheritOids) == 1);
tup = SearchSysCache1(RELOID,
DatumGetObjectId(linitial_oid(inheritOids)));
tablespaceId = ((Form_pg_class) GETSTRUCT(tup))->reltablespace;
if (!OidIsValid(tablespaceId))
tablespaceId = GetDefaultTablespace(stmt->relation->relpersistence);
ReleaseSysCache(tup);
}
else else
{ {
tablespaceId = GetDefaultTablespace(stmt->relation->relpersistence); tablespaceId = GetDefaultTablespace(stmt->relation->relpersistence);
...@@ -646,10 +706,10 @@ DefineRelation(CreateStmt *stmt, char relkind, Oid ownerId, ...@@ -646,10 +706,10 @@ DefineRelation(CreateStmt *stmt, char relkind, Oid ownerId,
* modified by MergeAttributes.) * modified by MergeAttributes.)
*/ */
stmt->tableElts = stmt->tableElts =
MergeAttributes(stmt->tableElts, stmt->inhRelations, MergeAttributes(stmt->tableElts, inheritOids,
stmt->relation->relpersistence, stmt->relation->relpersistence,
stmt->partbound != NULL, stmt->partbound != NULL,
&inheritOids, &old_constraints); &old_constraints);
/* /*
* Create a tuple descriptor from the relation schema. Note that this * Create a tuple descriptor from the relation schema. Note that this
...@@ -1781,12 +1841,11 @@ storage_name(char c) ...@@ -1781,12 +1841,11 @@ storage_name(char c)
* Input arguments: * Input arguments:
* 'schema' is the column/attribute definition for the table. (It's a list * 'schema' is the column/attribute definition for the table. (It's a list
* of ColumnDef's.) It is destructively changed. * of ColumnDef's.) It is destructively changed.
* 'supers' is a list of names (as RangeVar nodes) of parent relations. * 'supers' is a list of OIDs of parent relations, already locked by caller.
* 'relpersistence' is a persistence type of the table. * 'relpersistence' is a persistence type of the table.
* 'is_partition' tells if the table is a partition * 'is_partition' tells if the table is a partition
* *
* Output arguments: * Output arguments:
* 'supOids' receives a list of the OIDs of the parent relations.
* 'supconstr' receives a list of constraints belonging to the parents, * 'supconstr' receives a list of constraints belonging to the parents,
* updated as necessary to be valid for the child. * updated as necessary to be valid for the child.
* *
...@@ -1834,11 +1893,10 @@ storage_name(char c) ...@@ -1834,11 +1893,10 @@ storage_name(char c)
*/ */
static List * static List *
MergeAttributes(List *schema, List *supers, char relpersistence, MergeAttributes(List *schema, List *supers, char relpersistence,
bool is_partition, List **supOids, List **supconstr) bool is_partition, List **supconstr)
{ {
ListCell *entry; ListCell *entry;
List *inhSchema = NIL; List *inhSchema = NIL;
List *parentOids = NIL;
List *constraints = NIL; List *constraints = NIL;
bool have_bogus_defaults = false; bool have_bogus_defaults = false;
int child_attno; int child_attno;
...@@ -1939,31 +1997,15 @@ MergeAttributes(List *schema, List *supers, char relpersistence, ...@@ -1939,31 +1997,15 @@ MergeAttributes(List *schema, List *supers, char relpersistence,
child_attno = 0; child_attno = 0;
foreach(entry, supers) foreach(entry, supers)
{ {
RangeVar *parent = (RangeVar *) lfirst(entry); Oid parent = lfirst_oid(entry);
Relation relation; Relation relation;
TupleDesc tupleDesc; TupleDesc tupleDesc;
TupleConstr *constr; TupleConstr *constr;
AttrNumber *newattno; AttrNumber *newattno;
AttrNumber parent_attno; AttrNumber parent_attno;
/* /* caller already got lock */
* A self-exclusive lock is needed here. If two backends attempt to relation = heap_open(parent, NoLock);
* add children to the same parent simultaneously, and that parent has
* no pre-existing children, then both will attempt to update the
* parent's relhassubclass field, leading to a "tuple concurrently
* updated" error. Also, this interlocks against a concurrent ANALYZE
* on the parent table, which might otherwise be attempting to clear
* the parent's relhassubclass field, if its previous children were
* recently dropped.
*
* If the child table is a partition, then we instead grab an
* exclusive lock on the parent because its partition descriptor will
* be changed by addition of the new partition.
*/
if (!is_partition)
relation = heap_openrv(parent, ShareUpdateExclusiveLock);
else
relation = heap_openrv(parent, AccessExclusiveLock);
/* /*
* Check for active uses of the parent partitioned table in the * Check for active uses of the parent partitioned table in the
...@@ -1982,12 +2024,12 @@ MergeAttributes(List *schema, List *supers, char relpersistence, ...@@ -1982,12 +2024,12 @@ MergeAttributes(List *schema, List *supers, char relpersistence,
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_WRONG_OBJECT_TYPE), (errcode(ERRCODE_WRONG_OBJECT_TYPE),
errmsg("cannot inherit from partitioned table \"%s\"", errmsg("cannot inherit from partitioned table \"%s\"",
parent->relname))); RelationGetRelationName(relation))));
if (relation->rd_rel->relispartition && !is_partition) if (relation->rd_rel->relispartition && !is_partition)
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_WRONG_OBJECT_TYPE), (errcode(ERRCODE_WRONG_OBJECT_TYPE),
errmsg("cannot inherit from partition \"%s\"", errmsg("cannot inherit from partition \"%s\"",
parent->relname))); RelationGetRelationName(relation))));
if (relation->rd_rel->relkind != RELKIND_RELATION && if (relation->rd_rel->relkind != RELKIND_RELATION &&
relation->rd_rel->relkind != RELKIND_FOREIGN_TABLE && relation->rd_rel->relkind != RELKIND_FOREIGN_TABLE &&
...@@ -1995,7 +2037,7 @@ MergeAttributes(List *schema, List *supers, char relpersistence, ...@@ -1995,7 +2037,7 @@ MergeAttributes(List *schema, List *supers, char relpersistence,
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_WRONG_OBJECT_TYPE), (errcode(ERRCODE_WRONG_OBJECT_TYPE),
errmsg("inherited relation \"%s\" is not a table or foreign table", errmsg("inherited relation \"%s\" is not a table or foreign table",
parent->relname))); RelationGetRelationName(relation))));
/* /*
* If the parent is permanent, so must be all of its partitions. Note * If the parent is permanent, so must be all of its partitions. Note
...@@ -2017,7 +2059,7 @@ MergeAttributes(List *schema, List *supers, char relpersistence, ...@@ -2017,7 +2059,7 @@ MergeAttributes(List *schema, List *supers, char relpersistence,
errmsg(!is_partition errmsg(!is_partition
? "cannot inherit from temporary relation \"%s\"" ? "cannot inherit from temporary relation \"%s\""
: "cannot create a permanent relation as partition of temporary relation \"%s\"", : "cannot create a permanent relation as partition of temporary relation \"%s\"",
parent->relname))); RelationGetRelationName(relation))));
/* If existing rel is temp, it must belong to this session */ /* If existing rel is temp, it must belong to this session */
if (relation->rd_rel->relpersistence == RELPERSISTENCE_TEMP && if (relation->rd_rel->relpersistence == RELPERSISTENCE_TEMP &&
...@@ -2036,17 +2078,6 @@ MergeAttributes(List *schema, List *supers, char relpersistence, ...@@ -2036,17 +2078,6 @@ MergeAttributes(List *schema, List *supers, char relpersistence,
aclcheck_error(ACLCHECK_NOT_OWNER, get_relkind_objtype(relation->rd_rel->relkind), aclcheck_error(ACLCHECK_NOT_OWNER, get_relkind_objtype(relation->rd_rel->relkind),
RelationGetRelationName(relation)); RelationGetRelationName(relation));
/*
* Reject duplications in the list of parents.
*/
if (list_member_oid(parentOids, RelationGetRelid(relation)))
ereport(ERROR,
(errcode(ERRCODE_DUPLICATE_TABLE),
errmsg("relation \"%s\" would be inherited from more than once",
parent->relname)));
parentOids = lappend_oid(parentOids, RelationGetRelid(relation));
tupleDesc = RelationGetDescr(relation); tupleDesc = RelationGetDescr(relation);
constr = tupleDesc->constr; constr = tupleDesc->constr;
...@@ -2463,7 +2494,6 @@ MergeAttributes(List *schema, List *supers, char relpersistence, ...@@ -2463,7 +2494,6 @@ MergeAttributes(List *schema, List *supers, char relpersistence,
} }
} }
*supOids = parentOids;
*supconstr = constraints; *supconstr = constraints;
return schema; return schema;
} }
...@@ -4157,11 +4187,13 @@ ATExecCmd(List **wqueue, AlteredTableInfo *tab, Relation rel, ...@@ -4157,11 +4187,13 @@ ATExecCmd(List **wqueue, AlteredTableInfo *tab, Relation rel,
break; break;
case AT_SetTableSpace: /* SET TABLESPACE */ case AT_SetTableSpace: /* SET TABLESPACE */
/* /*
* Only do this for partitioned indexes, for which this is just * Only do this for partitioned tables and indexes, for which this
* a catalog change. Other relation types are handled by Phase 3. * is just a catalog change. Other relation types which have
* storage are handled by Phase 3.
*/ */
if (rel->rd_rel->relkind == RELKIND_PARTITIONED_INDEX) if (rel->rd_rel->relkind == RELKIND_PARTITIONED_TABLE ||
ATExecPartedIdxSetTableSpace(rel, tab->newTableSpace); rel->rd_rel->relkind == RELKIND_PARTITIONED_INDEX)
ATExecSetTableSpaceNoStorage(rel, tab->newTableSpace);
break; break;
case AT_SetRelOptions: /* SET (...) */ case AT_SetRelOptions: /* SET (...) */
...@@ -10935,19 +10967,26 @@ ATExecSetTableSpace(Oid tableOid, Oid newTableSpace, LOCKMODE lockmode) ...@@ -10935,19 +10967,26 @@ ATExecSetTableSpace(Oid tableOid, Oid newTableSpace, LOCKMODE lockmode)
} }
/* /*
* Special handling of ALTER TABLE SET TABLESPACE for partitioned indexes, * Special handling of ALTER TABLE SET TABLESPACE for relations with no
* which have no storage (so not handled in Phase 3 like other relation types) * storage that have an interest in preserving tablespace.
*
* Since these have no storage the tablespace can be updated with a simple
* metadata only operation to update the tablespace.
*/ */
static void static void
ATExecPartedIdxSetTableSpace(Relation rel, Oid newTableSpace) ATExecSetTableSpaceNoStorage(Relation rel, Oid newTableSpace)
{ {
HeapTuple tuple; HeapTuple tuple;
Oid oldTableSpace; Oid oldTableSpace;
Relation pg_class; Relation pg_class;
Form_pg_class rd_rel; Form_pg_class rd_rel;
Oid indexOid = RelationGetRelid(rel); Oid reloid = RelationGetRelid(rel);
Assert(rel->rd_rel->relkind == RELKIND_PARTITIONED_INDEX); /*
* Shouldn't be called on relations having storage; these are processed
* in phase 3.
*/
Assert(!RELKIND_CAN_HAVE_STORAGE(rel->rd_rel->relkind));
/* Can't allow a non-shared relation in pg_global */ /* Can't allow a non-shared relation in pg_global */
if (newTableSpace == GLOBALTABLESPACE_OID) if (newTableSpace == GLOBALTABLESPACE_OID)
...@@ -10962,24 +11001,23 @@ ATExecPartedIdxSetTableSpace(Relation rel, Oid newTableSpace) ...@@ -10962,24 +11001,23 @@ ATExecPartedIdxSetTableSpace(Relation rel, Oid newTableSpace)
if (newTableSpace == oldTableSpace || if (newTableSpace == oldTableSpace ||
(newTableSpace == MyDatabaseTableSpace && oldTableSpace == 0)) (newTableSpace == MyDatabaseTableSpace && oldTableSpace == 0))
{ {
InvokeObjectPostAlterHook(RelationRelationId, InvokeObjectPostAlterHook(RelationRelationId, reloid, 0);
indexOid, 0);
return; return;
} }
/* Get a modifiable copy of the relation's pg_class row */ /* Get a modifiable copy of the relation's pg_class row */
pg_class = heap_open(RelationRelationId, RowExclusiveLock); pg_class = heap_open(RelationRelationId, RowExclusiveLock);
tuple = SearchSysCacheCopy1(RELOID, ObjectIdGetDatum(indexOid)); tuple = SearchSysCacheCopy1(RELOID, ObjectIdGetDatum(reloid));
if (!HeapTupleIsValid(tuple)) if (!HeapTupleIsValid(tuple))
elog(ERROR, "cache lookup failed for relation %u", indexOid); elog(ERROR, "cache lookup failed for relation %u", reloid);
rd_rel = (Form_pg_class) GETSTRUCT(tuple); rd_rel = (Form_pg_class) GETSTRUCT(tuple);
/* update the pg_class row */ /* update the pg_class row */
rd_rel->reltablespace = (newTableSpace == MyDatabaseTableSpace) ? InvalidOid : newTableSpace; rd_rel->reltablespace = (newTableSpace == MyDatabaseTableSpace) ? InvalidOid : newTableSpace;
CatalogTupleUpdate(pg_class, &tuple->t_self, tuple); CatalogTupleUpdate(pg_class, &tuple->t_self, tuple);
InvokeObjectPostAlterHook(RelationRelationId, indexOid, 0); InvokeObjectPostAlterHook(RelationRelationId, reloid, 0);
heap_freetuple(tuple); heap_freetuple(tuple);
......
...@@ -122,6 +122,19 @@ typedef FormData_pg_class *Form_pg_class; ...@@ -122,6 +122,19 @@ typedef FormData_pg_class *Form_pg_class;
*/ */
#define REPLICA_IDENTITY_INDEX 'i' #define REPLICA_IDENTITY_INDEX 'i'
/*
* Relation kinds that have physical storage. These relations normally have
* relfilenode set to non-zero, but it can also be zero if the relation is
* mapped.
*/
#define RELKIND_CAN_HAVE_STORAGE(relkind) \
((relkind) == RELKIND_RELATION || \
(relkind) == RELKIND_INDEX || \
(relkind) == RELKIND_SEQUENCE || \
(relkind) == RELKIND_TOASTVALUE || \
(relkind) == RELKIND_MATVIEW)
#endif /* EXPOSE_TO_CLIENT_CODE */ #endif /* EXPOSE_TO_CLIENT_CODE */
#endif /* PG_CLASS_H */ #endif /* PG_CLASS_H */
...@@ -44,6 +44,18 @@ CREATE INDEX foo_idx on testschema.foo(i) TABLESPACE regress_tblspace; ...@@ -44,6 +44,18 @@ CREATE INDEX foo_idx on testschema.foo(i) TABLESPACE regress_tblspace;
SELECT relname, spcname FROM pg_catalog.pg_tablespace t, pg_catalog.pg_class c SELECT relname, spcname FROM pg_catalog.pg_tablespace t, pg_catalog.pg_class c
where c.reltablespace = t.oid AND c.relname = 'foo_idx'; where c.reltablespace = t.oid AND c.relname = 'foo_idx';
-- partitioned table
CREATE TABLE testschema.part (a int) PARTITION BY LIST (a);
CREATE TABLE testschema.part12 PARTITION OF testschema.part FOR VALUES IN(1,2) PARTITION BY LIST (a) TABLESPACE regress_tblspace;
CREATE TABLE testschema.part12_1 PARTITION OF testschema.part12 FOR VALUES IN (1);
ALTER TABLE testschema.part12 SET TABLESPACE pg_default;
CREATE TABLE testschema.part12_2 PARTITION OF testschema.part12 FOR VALUES IN (2);
-- Ensure part12_1 defaulted to regress_tblspace and part12_2 defaulted to pg_default.
SELECT relname, spcname FROM pg_catalog.pg_class c
LEFT JOIN pg_catalog.pg_tablespace t ON c.reltablespace = t.oid
where c.relname LIKE 'part%' order by relname;
DROP TABLE testschema.part;
-- partitioned index -- partitioned index
CREATE TABLE testschema.part (a int) PARTITION BY LIST (a); CREATE TABLE testschema.part (a int) PARTITION BY LIST (a);
CREATE TABLE testschema.part1 PARTITION OF testschema.part FOR VALUES IN (1); CREATE TABLE testschema.part1 PARTITION OF testschema.part FOR VALUES IN (1);
......
...@@ -61,6 +61,25 @@ SELECT relname, spcname FROM pg_catalog.pg_tablespace t, pg_catalog.pg_class c ...@@ -61,6 +61,25 @@ SELECT relname, spcname FROM pg_catalog.pg_tablespace t, pg_catalog.pg_class c
foo_idx | regress_tblspace foo_idx | regress_tblspace
(1 row) (1 row)
-- partitioned table
CREATE TABLE testschema.part (a int) PARTITION BY LIST (a);
CREATE TABLE testschema.part12 PARTITION OF testschema.part FOR VALUES IN(1,2) PARTITION BY LIST (a) TABLESPACE regress_tblspace;
CREATE TABLE testschema.part12_1 PARTITION OF testschema.part12 FOR VALUES IN (1);
ALTER TABLE testschema.part12 SET TABLESPACE pg_default;
CREATE TABLE testschema.part12_2 PARTITION OF testschema.part12 FOR VALUES IN (2);
-- Ensure part12_1 defaulted to regress_tblspace and part12_2 defaulted to pg_default.
SELECT relname, spcname FROM pg_catalog.pg_class c
LEFT JOIN pg_catalog.pg_tablespace t ON c.reltablespace = t.oid
where c.relname LIKE 'part%' order by relname;
relname | spcname
----------+------------------
part |
part12 |
part12_1 | regress_tblspace
part12_2 |
(4 rows)
DROP TABLE testschema.part;
-- partitioned index -- partitioned index
CREATE TABLE testschema.part (a int) PARTITION BY LIST (a); CREATE TABLE testschema.part (a int) PARTITION BY LIST (a);
CREATE TABLE testschema.part1 PARTITION OF testschema.part FOR VALUES IN (1); CREATE TABLE testschema.part1 PARTITION OF testschema.part FOR VALUES IN (1);
......
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