Commit af4de814 authored by Tom Lane's avatar Tom Lane

ALTER TABLE SET TABLESPACE. Gavin Sherry, some rework by Tom Lane.

parent 08d89db3
<!-- <!--
$PostgreSQL: pgsql/doc/src/sgml/ref/alter_table.sgml,v 1.72 2004/06/02 21:04:40 momjian Exp $ $PostgreSQL: pgsql/doc/src/sgml/ref/alter_table.sgml,v 1.73 2004/07/11 23:13:51 tgl Exp $
PostgreSQL documentation PostgreSQL documentation
--> -->
...@@ -39,10 +39,11 @@ where <replaceable class="PARAMETER">action</replaceable> is one of: ...@@ -39,10 +39,11 @@ where <replaceable class="PARAMETER">action</replaceable> is one of:
ALTER [ COLUMN ] <replaceable class="PARAMETER">column</replaceable> SET STORAGE { PLAIN | EXTERNAL | EXTENDED | MAIN } ALTER [ COLUMN ] <replaceable class="PARAMETER">column</replaceable> SET STORAGE { PLAIN | EXTERNAL | EXTENDED | MAIN }
ADD <replaceable class="PARAMETER">table_constraint</replaceable> ADD <replaceable class="PARAMETER">table_constraint</replaceable>
DROP CONSTRAINT <replaceable class="PARAMETER">constraint_name</replaceable> [ RESTRICT | CASCADE ] DROP CONSTRAINT <replaceable class="PARAMETER">constraint_name</replaceable> [ RESTRICT | CASCADE ]
SET WITHOUT OIDS
OWNER TO <replaceable class="PARAMETER">new_owner</replaceable>
CLUSTER ON <replaceable class="PARAMETER">index_name</replaceable> CLUSTER ON <replaceable class="PARAMETER">index_name</replaceable>
SET WITHOUT CLUSTER SET WITHOUT CLUSTER
SET WITHOUT OIDS
OWNER TO <replaceable class="PARAMETER">new_owner</replaceable>
SET TABLESPACE <replaceable class="PARAMETER">tablespace_name</replaceable>
</synopsis> </synopsis>
</refsynopsisdiv> </refsynopsisdiv>
...@@ -181,6 +182,29 @@ where <replaceable class="PARAMETER">action</replaceable> is one of: ...@@ -181,6 +182,29 @@ where <replaceable class="PARAMETER">action</replaceable> is one of:
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry>
<term><literal>CLUSTER</literal></term>
<listitem>
<para>
This form selects the default index for future
<xref linkend="SQL-CLUSTER" endterm="sql-cluster-title">
operations. It does not actually re-cluster the table.
</para>
</listitem>
</varlistentry>
<varlistentry>
<term><literal>SET WITHOUT CLUSTER</literal></term>
<listitem>
<para>
This form removes the most recently used
<xref linkend="SQL-CLUSTER" endterm="sql-cluster-title">
index specification from the table. This affects
future cluster operations that don't specify an index.
</para>
</listitem>
</varlistentry>
<varlistentry> <varlistentry>
<term><literal>SET WITHOUT OIDS</literal></term> <term><literal>SET WITHOUT OIDS</literal></term>
<listitem> <listitem>
...@@ -211,28 +235,19 @@ where <replaceable class="PARAMETER">action</replaceable> is one of: ...@@ -211,28 +235,19 @@ where <replaceable class="PARAMETER">action</replaceable> is one of:
</varlistentry> </varlistentry>
<varlistentry> <varlistentry>
<term><literal>CLUSTER</literal></term> <term><literal>SET TABLESPACE</literal></term>
<listitem> <listitem>
<para> <para>
This form selects the default index for future This form changes the table's tablespace to the specified tablespace and
<xref linkend="SQL-CLUSTER" endterm="sql-cluster-title"> moves the data file(s) associated with the table to the new tablespace.
operations. Indexes on the table, if any, are not moved; but they can be moved
separately with additional <literal>SET TABLESPACE</literal> commands.
See also
<xref linkend="SQL-CREATETABLESPACE" endterm="sql-createtablespace-title">.
</para> </para>
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry>
<term><literal>SET WITHOUT CLUSTER</literal></term>
<listitem>
<para>
This form removes the most recently used
<xref linkend="SQL-CLUSTER" endterm="sql-cluster-title">
index specification from the table. This affects
future cluster operations that don't specify an index.
</para>
</listitem>
</varlistentry>
<varlistentry> <varlistentry>
<term><literal>RENAME</literal></term> <term><literal>RENAME</literal></term>
<listitem> <listitem>
...@@ -293,29 +308,29 @@ where <replaceable class="PARAMETER">action</replaceable> is one of: ...@@ -293,29 +308,29 @@ where <replaceable class="PARAMETER">action</replaceable> is one of:
</varlistentry> </varlistentry>
<varlistentry> <varlistentry>
<term><replaceable class="PARAMETER">type</replaceable></term> <term><replaceable class="PARAMETER">new_column</replaceable></term>
<listitem> <listitem>
<para> <para>
Data type of the new column, or new data type for an existing New name for an existing column.
column.
</para> </para>
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry> <varlistentry>
<term><replaceable class="PARAMETER">new_column</replaceable></term> <term><replaceable class="PARAMETER">new_name</replaceable></term>
<listitem> <listitem>
<para> <para>
New name for an existing column. New name for the table.
</para> </para>
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry> <varlistentry>
<term><replaceable class="PARAMETER">new_name</replaceable></term> <term><replaceable class="PARAMETER">type</replaceable></term>
<listitem> <listitem>
<para> <para>
New name for the table. Data type of the new column, or new data type for an existing
column.
</para> </para>
</listitem> </listitem>
</varlistentry> </varlistentry>
...@@ -339,10 +354,21 @@ where <replaceable class="PARAMETER">action</replaceable> is one of: ...@@ -339,10 +354,21 @@ where <replaceable class="PARAMETER">action</replaceable> is one of:
</varlistentry> </varlistentry>
<varlistentry> <varlistentry>
<term><replaceable class="PARAMETER">new_owner</replaceable></term> <term><literal>CASCADE</literal></term>
<listitem> <listitem>
<para> <para>
The user name of the new owner of the table. Automatically drop objects that depend on the dropped column
or constraint (for example, views referencing the column).
</para>
</listitem>
</varlistentry>
<varlistentry>
<term><literal>RESTRICT</literal></term>
<listitem>
<para>
Refuse to drop the column or constraint if there are any dependent
objects. This is the default behavior.
</para> </para>
</listitem> </listitem>
</varlistentry> </varlistentry>
...@@ -357,21 +383,19 @@ where <replaceable class="PARAMETER">action</replaceable> is one of: ...@@ -357,21 +383,19 @@ where <replaceable class="PARAMETER">action</replaceable> is one of:
</varlistentry> </varlistentry>
<varlistentry> <varlistentry>
<term><literal>CASCADE</literal></term> <term><replaceable class="PARAMETER">new_owner</replaceable></term>
<listitem> <listitem>
<para> <para>
Automatically drop objects that depend on the dropped column The user name of the new owner of the table.
or constraint (for example, views referencing the column).
</para> </para>
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry> <varlistentry>
<term><literal>RESTRICT</literal></term> <term><replaceable class="PARAMETER">tablespace_name</replaceable></term>
<listitem> <listitem>
<para> <para>
Refuse to drop the column or constraint if there are any dependent The tablespace name to which the table will be moved.
objects. This is the default behavior.
</para> </para>
</listitem> </listitem>
</varlistentry> </varlistentry>
...@@ -551,6 +575,14 @@ ALTER TABLE distributors ADD CONSTRAINT dist_id_zipcode_key UNIQUE (dist_id, zip ...@@ -551,6 +575,14 @@ ALTER TABLE distributors ADD CONSTRAINT dist_id_zipcode_key UNIQUE (dist_id, zip
ALTER TABLE distributors ADD PRIMARY KEY (dist_id); ALTER TABLE distributors ADD PRIMARY KEY (dist_id);
</programlisting> </programlisting>
</para> </para>
<para>
To move a table to a different tablespace:
<programlisting>
ALTER TABLE distributors SET TABLESPACE fasttablespace;
</programlisting>
</para>
</refsect1> </refsect1>
<refsect1> <refsect1>
......
...@@ -11,7 +11,7 @@ ...@@ -11,7 +11,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/commands/cluster.c,v 1.126 2004/06/18 06:13:22 tgl Exp $ * $PostgreSQL: pgsql/src/backend/commands/cluster.c,v 1.127 2004/07/11 23:13:53 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -485,6 +485,7 @@ static void ...@@ -485,6 +485,7 @@ static void
rebuild_relation(Relation OldHeap, Oid indexOid) rebuild_relation(Relation OldHeap, Oid indexOid)
{ {
Oid tableOid = RelationGetRelid(OldHeap); Oid tableOid = RelationGetRelid(OldHeap);
Oid tableSpace = OldHeap->rd_rel->reltablespace;
Oid OIDNewHeap; Oid OIDNewHeap;
char NewHeapName[NAMEDATALEN]; char NewHeapName[NAMEDATALEN];
ObjectAddress object; ObjectAddress object;
...@@ -505,7 +506,7 @@ rebuild_relation(Relation OldHeap, Oid indexOid) ...@@ -505,7 +506,7 @@ rebuild_relation(Relation OldHeap, Oid indexOid)
*/ */
snprintf(NewHeapName, sizeof(NewHeapName), "pg_temp_%u", tableOid); snprintf(NewHeapName, sizeof(NewHeapName), "pg_temp_%u", tableOid);
OIDNewHeap = make_new_heap(tableOid, NewHeapName); OIDNewHeap = make_new_heap(tableOid, NewHeapName, tableSpace);
/* /*
* We don't need CommandCounterIncrement() because make_new_heap did * We don't need CommandCounterIncrement() because make_new_heap did
...@@ -520,8 +521,8 @@ rebuild_relation(Relation OldHeap, Oid indexOid) ...@@ -520,8 +521,8 @@ rebuild_relation(Relation OldHeap, Oid indexOid)
/* To make the new heap's data visible (probably not needed?). */ /* To make the new heap's data visible (probably not needed?). */
CommandCounterIncrement(); CommandCounterIncrement();
/* Swap the relfilenodes of the old and new heaps. */ /* Swap the physical files of the old and new heaps. */
swap_relfilenodes(tableOid, OIDNewHeap); swap_relation_files(tableOid, OIDNewHeap);
CommandCounterIncrement(); CommandCounterIncrement();
...@@ -550,7 +551,7 @@ rebuild_relation(Relation OldHeap, Oid indexOid) ...@@ -550,7 +551,7 @@ rebuild_relation(Relation OldHeap, Oid indexOid)
* Create the new table that we will fill with correctly-ordered data. * Create the new table that we will fill with correctly-ordered data.
*/ */
Oid Oid
make_new_heap(Oid OIDOldHeap, const char *NewName) make_new_heap(Oid OIDOldHeap, const char *NewName, Oid NewTableSpace)
{ {
TupleDesc OldHeapDesc, TupleDesc OldHeapDesc,
tupdesc; tupdesc;
...@@ -568,7 +569,7 @@ make_new_heap(Oid OIDOldHeap, const char *NewName) ...@@ -568,7 +569,7 @@ make_new_heap(Oid OIDOldHeap, const char *NewName)
OIDNewHeap = heap_create_with_catalog(NewName, OIDNewHeap = heap_create_with_catalog(NewName,
RelationGetNamespace(OldHeap), RelationGetNamespace(OldHeap),
OldHeap->rd_rel->reltablespace, NewTableSpace,
tupdesc, tupdesc,
OldHeap->rd_rel->relkind, OldHeap->rd_rel->relkind,
OldHeap->rd_rel->relisshared, OldHeap->rd_rel->relisshared,
...@@ -646,13 +647,16 @@ copy_heap_data(Oid OIDNewHeap, Oid OIDOldHeap, Oid OIDOldIndex) ...@@ -646,13 +647,16 @@ copy_heap_data(Oid OIDNewHeap, Oid OIDOldHeap, Oid OIDOldIndex)
} }
/* /*
* Swap the relfilenodes for two given relations. * Swap the physical files of two given relations.
*
* We swap the physical identity (reltablespace and relfilenode) while
* keeping the same logical identities of the two relations.
* *
* Also swap any TOAST links, so that the toast data moves along with * Also swap any TOAST links, so that the toast data moves along with
* the main-table data. * the main-table data.
*/ */
void void
swap_relfilenodes(Oid r1, Oid r2) swap_relation_files(Oid r1, Oid r2)
{ {
Relation relRelation, Relation relRelation,
rel; rel;
...@@ -695,12 +699,16 @@ swap_relfilenodes(Oid r1, Oid r2) ...@@ -695,12 +699,16 @@ swap_relfilenodes(Oid r1, Oid r2)
relation_close(rel, NoLock); relation_close(rel, NoLock);
/* /*
* Actually swap the filenode and TOAST fields in the two tuples * Actually swap the fields in the two tuples
*/ */
swaptemp = relform1->relfilenode; swaptemp = relform1->relfilenode;
relform1->relfilenode = relform2->relfilenode; relform1->relfilenode = relform2->relfilenode;
relform2->relfilenode = swaptemp; relform2->relfilenode = swaptemp;
swaptemp = relform1->reltablespace;
relform1->reltablespace = relform2->reltablespace;
relform2->reltablespace = swaptemp;
swaptemp = relform1->reltoastrelid; swaptemp = relform1->reltoastrelid;
relform1->reltoastrelid = relform2->reltoastrelid; relform1->reltoastrelid = relform2->reltoastrelid;
relform2->reltoastrelid = swaptemp; relform2->reltoastrelid = swaptemp;
...@@ -793,13 +801,16 @@ swap_relfilenodes(Oid r1, Oid r2) ...@@ -793,13 +801,16 @@ swap_relfilenodes(Oid r1, Oid r2)
/* /*
* Blow away the old relcache entries now. We need this kluge because * Blow away the old relcache entries now. We need this kluge because
* relcache.c indexes relcache entries by rd_node as well as OID. It * relcache.c keeps a link to the smgr relation for the physical file,
* will get confused if it is asked to (re)build an entry with a new * and that will be out of date as soon as we do CommandCounterIncrement.
* rd_node value when there is still another entry laying about with * Whichever of the rels is the second to be cleared during cache
* that same rd_node value. (Fortunately, since one of the entries is * invalidation will have a dangling reference to an already-deleted smgr
* local in our transaction, it's sufficient to clear out our own * relation. Rather than trying to avoid this by ordering operations
* relcache this way; the problem cannot arise for other backends when * just so, it's easiest to not have the relcache entries there at all.
* they see our update on the non-local relation.) * (Fortunately, since one of the entries is local in our transaction,
* it's sufficient to clear out our own relcache this way; the problem
* cannot arise for other backends when they see our update on the
* non-local relation.)
*/ */
RelationForgetRelation(r1); RelationForgetRelation(r1);
RelationForgetRelation(r2); RelationForgetRelation(r2);
......
This diff is collapsed.
...@@ -11,7 +11,7 @@ ...@@ -11,7 +11,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/parser/gram.y,v 2.465 2004/06/28 01:19:11 tgl Exp $ * $PostgreSQL: pgsql/src/backend/parser/gram.y,v 2.466 2004/07/11 23:13:54 tgl Exp $
* *
* HISTORY * HISTORY
* AUTHOR DATE MAJOR EVENT * AUTHOR DATE MAJOR EVENT
...@@ -1286,6 +1286,14 @@ alter_table_cmd: ...@@ -1286,6 +1286,14 @@ alter_table_cmd:
n->name = NULL; n->name = NULL;
$$ = (Node *)n; $$ = (Node *)n;
} }
/* ALTER TABLE <name> SET TABLESPACE <tablespacename> */
| SET TABLESPACE name
{
AlterTableCmd *n = makeNode(AlterTableCmd);
n->subtype = AT_SetTableSpace;
n->name = $3;
$$ = (Node *)n;
}
; ;
alter_column_default: alter_column_default:
......
...@@ -6,7 +6,7 @@ ...@@ -6,7 +6,7 @@
* Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group
* Portions Copyright (c) 1994-5, Regents of the University of California * Portions Copyright (c) 1994-5, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/commands/cluster.h,v 1.23 2004/05/08 00:34:49 tgl Exp $ * $PostgreSQL: pgsql/src/include/commands/cluster.h,v 1.24 2004/07/11 23:13:56 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -21,7 +21,8 @@ extern void cluster(ClusterStmt *stmt); ...@@ -21,7 +21,8 @@ extern void cluster(ClusterStmt *stmt);
extern void check_index_is_clusterable(Relation OldHeap, Oid indexOid); extern void check_index_is_clusterable(Relation OldHeap, Oid indexOid);
extern void mark_index_clustered(Relation rel, Oid indexOid); extern void mark_index_clustered(Relation rel, Oid indexOid);
extern Oid make_new_heap(Oid OIDOldHeap, const char *NewName); extern Oid make_new_heap(Oid OIDOldHeap, const char *NewName,
extern void swap_relfilenodes(Oid r1, Oid r2); Oid NewTableSpace);
extern void swap_relation_files(Oid r1, Oid r2);
#endif /* CLUSTER_H */ #endif /* CLUSTER_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/nodes/parsenodes.h,v 1.260 2004/06/25 21:55:59 tgl Exp $ * $PostgreSQL: pgsql/src/include/nodes/parsenodes.h,v 1.261 2004/07/11 23:13:58 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -806,7 +806,8 @@ typedef enum AlterTableType ...@@ -806,7 +806,8 @@ typedef enum AlterTableType
AT_ChangeOwner, /* change owner */ AT_ChangeOwner, /* change owner */
AT_ClusterOn, /* CLUSTER ON */ AT_ClusterOn, /* CLUSTER ON */
AT_DropCluster, /* SET WITHOUT CLUSTER */ AT_DropCluster, /* SET WITHOUT CLUSTER */
AT_DropOids /* SET WITHOUT OIDS */ AT_DropOids, /* SET WITHOUT OIDS */
AT_SetTableSpace /* SET TABLESPACE */
} AlterTableType; } AlterTableType;
typedef struct AlterTableCmd /* one subcommand of an ALTER TABLE */ typedef struct AlterTableCmd /* one subcommand of an ALTER TABLE */
...@@ -814,7 +815,7 @@ typedef struct AlterTableCmd /* one subcommand of an ALTER TABLE */ ...@@ -814,7 +815,7 @@ typedef struct AlterTableCmd /* one subcommand of an ALTER TABLE */
NodeTag type; NodeTag type;
AlterTableType subtype; /* Type of table alteration to apply */ AlterTableType subtype; /* Type of table alteration to apply */
char *name; /* column or constraint name to act on, or char *name; /* column or constraint name to act on, or
* new owner */ * new owner or tablespace */
Node *def; /* definition of new column, column type, Node *def; /* definition of new column, column type,
* index, or constraint */ * index, or constraint */
Node *transform; /* transformation expr for ALTER TYPE */ Node *transform; /* transformation expr for ALTER TYPE */
......
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