mirror of
https://github.com/postgres/postgres.git
synced 2025-04-22 23:02:54 +03:00
Refactor ALTER some-obj RENAME implementation
Remove duplicate implementations of catalog munging and miscellaneous privilege checks. Instead rely on already existing data in objectaddress.c to do the work. Author: KaiGai Kohei, changes by me Reviewed by: Robert Haas, Álvaro Herrera, Dimitri Fontaine
This commit is contained in:
parent
8f0d8f481e
commit
765cbfdc92
@ -80,7 +80,9 @@ typedef struct
|
|||||||
Oid class_oid; /* oid of catalog */
|
Oid class_oid; /* oid of catalog */
|
||||||
Oid oid_index_oid; /* oid of index on system oid column */
|
Oid oid_index_oid; /* oid of index on system oid column */
|
||||||
int oid_catcache_id; /* id of catcache on system oid column */
|
int oid_catcache_id; /* id of catcache on system oid column */
|
||||||
int name_catcache_id; /* id of catcache on (name,namespace) */
|
int name_catcache_id; /* id of catcache on (name,namespace),
|
||||||
|
* or (name) if the object does not
|
||||||
|
* live in a namespace */
|
||||||
AttrNumber attnum_name; /* attnum of name field */
|
AttrNumber attnum_name; /* attnum of name field */
|
||||||
AttrNumber attnum_namespace; /* attnum of namespace field */
|
AttrNumber attnum_namespace; /* attnum of namespace field */
|
||||||
AttrNumber attnum_owner; /* attnum of owner field */
|
AttrNumber attnum_owner; /* attnum of owner field */
|
||||||
|
@ -226,66 +226,3 @@ DefineAggregate(List *name, List *args, bool oldstyle, List *parameters)
|
|||||||
transTypeId, /* transition data type */
|
transTypeId, /* transition data type */
|
||||||
initval); /* initial condition */
|
initval); /* initial condition */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* RenameAggregate
|
|
||||||
* Rename an aggregate.
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameAggregate(List *name, List *args, const char *newname)
|
|
||||||
{
|
|
||||||
Oid procOid;
|
|
||||||
Oid namespaceOid;
|
|
||||||
HeapTuple tup;
|
|
||||||
Form_pg_proc procForm;
|
|
||||||
Relation rel;
|
|
||||||
AclResult aclresult;
|
|
||||||
|
|
||||||
rel = heap_open(ProcedureRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
/* Look up function and make sure it's an aggregate */
|
|
||||||
procOid = LookupAggNameTypeNames(name, args, false);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(PROCOID, ObjectIdGetDatum(procOid));
|
|
||||||
if (!HeapTupleIsValid(tup)) /* should not happen */
|
|
||||||
elog(ERROR, "cache lookup failed for function %u", procOid);
|
|
||||||
procForm = (Form_pg_proc) GETSTRUCT(tup);
|
|
||||||
|
|
||||||
namespaceOid = procForm->pronamespace;
|
|
||||||
|
|
||||||
/* make sure the new name doesn't exist */
|
|
||||||
if (SearchSysCacheExists3(PROCNAMEARGSNSP,
|
|
||||||
CStringGetDatum(newname),
|
|
||||||
PointerGetDatum(&procForm->proargtypes),
|
|
||||||
ObjectIdGetDatum(namespaceOid)))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_FUNCTION),
|
|
||||||
errmsg("function %s already exists in schema \"%s\"",
|
|
||||||
funcname_signature_string(newname,
|
|
||||||
procForm->pronargs,
|
|
||||||
NIL,
|
|
||||||
procForm->proargtypes.values),
|
|
||||||
get_namespace_name(namespaceOid))));
|
|
||||||
|
|
||||||
/* must be owner */
|
|
||||||
if (!pg_proc_ownercheck(procOid, GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_PROC,
|
|
||||||
NameListToString(name));
|
|
||||||
|
|
||||||
/* must have CREATE privilege on namespace */
|
|
||||||
aclresult = pg_namespace_aclcheck(namespaceOid, GetUserId(), ACL_CREATE);
|
|
||||||
if (aclresult != ACLCHECK_OK)
|
|
||||||
aclcheck_error(aclresult, ACL_KIND_NAMESPACE,
|
|
||||||
get_namespace_name(namespaceOid));
|
|
||||||
|
|
||||||
/* rename */
|
|
||||||
namestrcpy(&(((Form_pg_proc) GETSTRUCT(tup))->proname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return procOid;
|
|
||||||
}
|
|
||||||
|
@ -21,9 +21,15 @@
|
|||||||
#include "catalog/namespace.h"
|
#include "catalog/namespace.h"
|
||||||
#include "catalog/pg_collation.h"
|
#include "catalog/pg_collation.h"
|
||||||
#include "catalog/pg_conversion.h"
|
#include "catalog/pg_conversion.h"
|
||||||
|
#include "catalog/pg_event_trigger.h"
|
||||||
|
#include "catalog/pg_foreign_data_wrapper.h"
|
||||||
|
#include "catalog/pg_foreign_server.h"
|
||||||
|
#include "catalog/pg_language.h"
|
||||||
#include "catalog/pg_largeobject.h"
|
#include "catalog/pg_largeobject.h"
|
||||||
#include "catalog/pg_largeobject_metadata.h"
|
#include "catalog/pg_largeobject_metadata.h"
|
||||||
#include "catalog/pg_namespace.h"
|
#include "catalog/pg_namespace.h"
|
||||||
|
#include "catalog/pg_opclass.h"
|
||||||
|
#include "catalog/pg_opfamily.h"
|
||||||
#include "catalog/pg_proc.h"
|
#include "catalog/pg_proc.h"
|
||||||
#include "catalog/pg_ts_config.h"
|
#include "catalog/pg_ts_config.h"
|
||||||
#include "catalog/pg_ts_dict.h"
|
#include "catalog/pg_ts_dict.h"
|
||||||
@ -43,6 +49,7 @@
|
|||||||
#include "commands/trigger.h"
|
#include "commands/trigger.h"
|
||||||
#include "commands/typecmds.h"
|
#include "commands/typecmds.h"
|
||||||
#include "commands/user.h"
|
#include "commands/user.h"
|
||||||
|
#include "parser/parse_func.h"
|
||||||
#include "miscadmin.h"
|
#include "miscadmin.h"
|
||||||
#include "tcop/utility.h"
|
#include "tcop/utility.h"
|
||||||
#include "utils/builtins.h"
|
#include "utils/builtins.h"
|
||||||
@ -55,6 +62,233 @@
|
|||||||
|
|
||||||
static Oid AlterObjectNamespace_internal(Relation rel, Oid objid, Oid nspOid);
|
static Oid AlterObjectNamespace_internal(Relation rel, Oid objid, Oid nspOid);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Raise an error to the effect that an object of the given name is already
|
||||||
|
* present in the given namespace.
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
report_name_conflict(Oid classId, const char *name)
|
||||||
|
{
|
||||||
|
char *msgfmt;
|
||||||
|
|
||||||
|
switch (classId)
|
||||||
|
{
|
||||||
|
case EventTriggerRelationId:
|
||||||
|
msgfmt = gettext_noop("event trigger \"%s\" already exists");
|
||||||
|
break;
|
||||||
|
case ForeignDataWrapperRelationId:
|
||||||
|
msgfmt = gettext_noop("foreign-data wrapper \"%s\" already exists");
|
||||||
|
break;
|
||||||
|
case ForeignServerRelationId:
|
||||||
|
msgfmt = gettext_noop("server \"%s\" already exists");
|
||||||
|
break;
|
||||||
|
case LanguageRelationId:
|
||||||
|
msgfmt = gettext_noop("language \"%s\" already exists");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
elog(ERROR, "unsupported object class %u", classId);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
ereport(ERROR,
|
||||||
|
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
||||||
|
errmsg(msgfmt, name)));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
report_namespace_conflict(Oid classId, const char *name, Oid nspOid)
|
||||||
|
{
|
||||||
|
char *msgfmt;
|
||||||
|
|
||||||
|
Assert(OidIsValid(nspOid));
|
||||||
|
|
||||||
|
switch (classId)
|
||||||
|
{
|
||||||
|
case ConversionRelationId:
|
||||||
|
Assert(OidIsValid(nspOid));
|
||||||
|
msgfmt = gettext_noop("conversion \"%s\" already exists in schema \"%s\"");
|
||||||
|
break;
|
||||||
|
case TSParserRelationId:
|
||||||
|
Assert(OidIsValid(nspOid));
|
||||||
|
msgfmt = gettext_noop("text search parser \"%s\" already exists in schema \"%s\"");
|
||||||
|
break;
|
||||||
|
case TSDictionaryRelationId:
|
||||||
|
Assert(OidIsValid(nspOid));
|
||||||
|
msgfmt = gettext_noop("text search dictionary \"%s\" already exists in schema \"%s\"");
|
||||||
|
break;
|
||||||
|
case TSTemplateRelationId:
|
||||||
|
Assert(OidIsValid(nspOid));
|
||||||
|
msgfmt = gettext_noop("text search template \"%s\" already exists in schema \"%s\"");
|
||||||
|
break;
|
||||||
|
case TSConfigRelationId:
|
||||||
|
Assert(OidIsValid(nspOid));
|
||||||
|
msgfmt = gettext_noop("text search configuration \"%s\" already exists in schema \"%s\"");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
elog(ERROR, "unsupported object class %u", classId);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
ereport(ERROR,
|
||||||
|
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
||||||
|
errmsg(msgfmt, name, get_namespace_name(nspOid))));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* AlterObjectRename_internal
|
||||||
|
*
|
||||||
|
* Generic function to rename the given object, for simple cases (won't
|
||||||
|
* work for tables, nor other cases where we need to do more than change
|
||||||
|
* the name column of a single catalog entry).
|
||||||
|
*
|
||||||
|
* rel: catalog relation containing object (RowExclusiveLock'd by caller)
|
||||||
|
* objectId: OID of object to be renamed
|
||||||
|
* new_name: CString representation of new name
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
AlterObjectRename_internal(Relation rel, Oid objectId, const char *new_name)
|
||||||
|
{
|
||||||
|
Oid classId = RelationGetRelid(rel);
|
||||||
|
int oidCacheId = get_object_catcache_oid(classId);
|
||||||
|
int nameCacheId = get_object_catcache_name(classId);
|
||||||
|
AttrNumber Anum_name = get_object_attnum_name(classId);
|
||||||
|
AttrNumber Anum_namespace = get_object_attnum_namespace(classId);
|
||||||
|
AttrNumber Anum_owner = get_object_attnum_owner(classId);
|
||||||
|
AclObjectKind acl_kind = get_object_aclkind(classId);
|
||||||
|
HeapTuple oldtup;
|
||||||
|
HeapTuple newtup;
|
||||||
|
Datum datum;
|
||||||
|
bool isnull;
|
||||||
|
Oid namespaceId;
|
||||||
|
Oid ownerId;
|
||||||
|
char *old_name;
|
||||||
|
AclResult aclresult;
|
||||||
|
Datum *values;
|
||||||
|
bool *nulls;
|
||||||
|
bool *replaces;
|
||||||
|
|
||||||
|
oldtup = SearchSysCache1(oidCacheId, ObjectIdGetDatum(objectId));
|
||||||
|
if (!HeapTupleIsValid(oldtup))
|
||||||
|
elog(ERROR, "cache lookup failed for object %u of catalog \"%s\"",
|
||||||
|
objectId, RelationGetRelationName(rel));
|
||||||
|
|
||||||
|
datum = heap_getattr(oldtup, Anum_name,
|
||||||
|
RelationGetDescr(rel), &isnull);
|
||||||
|
Assert(!isnull);
|
||||||
|
old_name = NameStr(*(DatumGetName(datum)));
|
||||||
|
|
||||||
|
/* Get OID of namespace */
|
||||||
|
if (Anum_namespace > 0)
|
||||||
|
{
|
||||||
|
datum = heap_getattr(oldtup, Anum_namespace,
|
||||||
|
RelationGetDescr(rel), &isnull);
|
||||||
|
Assert(!isnull);
|
||||||
|
namespaceId = DatumGetObjectId(datum);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
namespaceId = InvalidOid;
|
||||||
|
|
||||||
|
/* Permission checks ... superusers can always do it */
|
||||||
|
if (!superuser())
|
||||||
|
{
|
||||||
|
/* Fail if object does not have an explicit owner */
|
||||||
|
if (Anum_owner <= 0)
|
||||||
|
ereport(ERROR,
|
||||||
|
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
|
||||||
|
(errmsg("must be superuser to rename %s",
|
||||||
|
getObjectDescriptionOids(classId, objectId)))));
|
||||||
|
|
||||||
|
/* Otherwise, must be owner of the existing object */
|
||||||
|
datum = heap_getattr(oldtup, Anum_owner,
|
||||||
|
RelationGetDescr(rel), &isnull);
|
||||||
|
Assert(!isnull);
|
||||||
|
ownerId = DatumGetObjectId(datum);
|
||||||
|
|
||||||
|
if (!has_privs_of_role(GetUserId(), DatumGetObjectId(ownerId)))
|
||||||
|
aclcheck_error(ACLCHECK_NOT_OWNER, acl_kind, old_name);
|
||||||
|
|
||||||
|
/* User must have CREATE privilege on the namespace */
|
||||||
|
if (OidIsValid(namespaceId))
|
||||||
|
{
|
||||||
|
aclresult = pg_namespace_aclcheck(namespaceId, GetUserId(),
|
||||||
|
ACL_CREATE);
|
||||||
|
if (aclresult != ACLCHECK_OK)
|
||||||
|
aclcheck_error(aclresult, ACL_KIND_NAMESPACE,
|
||||||
|
get_namespace_name(namespaceId));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check for duplicate name (more friendly than unique-index failure).
|
||||||
|
* Since this is just a friendliness check, we can just skip it in cases
|
||||||
|
* where there isn't suitable support.
|
||||||
|
*/
|
||||||
|
if (classId == ProcedureRelationId)
|
||||||
|
{
|
||||||
|
Form_pg_proc proc = (Form_pg_proc) GETSTRUCT(oldtup);
|
||||||
|
|
||||||
|
IsThereFunctionInNamespace(new_name, proc->pronargs,
|
||||||
|
proc->proargtypes, proc->pronamespace);
|
||||||
|
}
|
||||||
|
else if (classId == CollationRelationId)
|
||||||
|
{
|
||||||
|
Form_pg_collation coll = (Form_pg_collation) GETSTRUCT(oldtup);
|
||||||
|
|
||||||
|
IsThereCollationInNamespace(new_name, coll->collnamespace);
|
||||||
|
}
|
||||||
|
else if (classId == OperatorClassRelationId)
|
||||||
|
{
|
||||||
|
Form_pg_opclass opc = (Form_pg_opclass) GETSTRUCT(oldtup);
|
||||||
|
|
||||||
|
IsThereOpClassInNamespace(new_name, opc->opcmethod,
|
||||||
|
opc->opcnamespace);
|
||||||
|
}
|
||||||
|
else if (classId == OperatorFamilyRelationId)
|
||||||
|
{
|
||||||
|
Form_pg_opfamily opf = (Form_pg_opfamily) GETSTRUCT(oldtup);
|
||||||
|
|
||||||
|
IsThereOpFamilyInNamespace(new_name, opf->opfmethod,
|
||||||
|
opf->opfnamespace);
|
||||||
|
}
|
||||||
|
else if (nameCacheId >= 0)
|
||||||
|
{
|
||||||
|
if (OidIsValid(namespaceId))
|
||||||
|
{
|
||||||
|
if (SearchSysCacheExists2(nameCacheId,
|
||||||
|
CStringGetDatum(new_name),
|
||||||
|
ObjectIdGetDatum(namespaceId)))
|
||||||
|
report_namespace_conflict(classId, new_name, namespaceId);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (SearchSysCacheExists1(nameCacheId,
|
||||||
|
CStringGetDatum(new_name)))
|
||||||
|
report_name_conflict(classId, new_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Build modified tuple */
|
||||||
|
values = palloc0(RelationGetNumberOfAttributes(rel) * sizeof(Datum));
|
||||||
|
nulls = palloc0(RelationGetNumberOfAttributes(rel) * sizeof(bool));
|
||||||
|
replaces = palloc0(RelationGetNumberOfAttributes(rel) * sizeof(bool));
|
||||||
|
values[Anum_name - 1] = PointerGetDatum(new_name);
|
||||||
|
replaces[Anum_name - 1] = true;
|
||||||
|
newtup = heap_modify_tuple(oldtup, RelationGetDescr(rel),
|
||||||
|
values, nulls, replaces);
|
||||||
|
|
||||||
|
/* Perform actual update */
|
||||||
|
simple_heap_update(rel, &oldtup->t_self, newtup);
|
||||||
|
CatalogUpdateIndexes(rel, newtup);
|
||||||
|
|
||||||
|
/* Release memory */
|
||||||
|
pfree(values);
|
||||||
|
pfree(nulls);
|
||||||
|
pfree(replaces);
|
||||||
|
heap_freetuple(newtup);
|
||||||
|
|
||||||
|
ReleaseSysCache(oldtup);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Executes an ALTER OBJECT / RENAME TO statement. Based on the object
|
* Executes an ALTER OBJECT / RENAME TO statement. Based on the object
|
||||||
* type, the function appropriate to that type is executed.
|
* type, the function appropriate to that type is executed.
|
||||||
@ -64,42 +298,12 @@ ExecRenameStmt(RenameStmt *stmt)
|
|||||||
{
|
{
|
||||||
switch (stmt->renameType)
|
switch (stmt->renameType)
|
||||||
{
|
{
|
||||||
case OBJECT_AGGREGATE:
|
|
||||||
return RenameAggregate(stmt->object, stmt->objarg, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_COLLATION:
|
|
||||||
return RenameCollation(stmt->object, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_CONSTRAINT:
|
case OBJECT_CONSTRAINT:
|
||||||
return RenameConstraint(stmt);
|
return RenameConstraint(stmt);
|
||||||
|
|
||||||
case OBJECT_CONVERSION:
|
|
||||||
return RenameConversion(stmt->object, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_DATABASE:
|
case OBJECT_DATABASE:
|
||||||
return RenameDatabase(stmt->subname, stmt->newname);
|
return RenameDatabase(stmt->subname, stmt->newname);
|
||||||
|
|
||||||
case OBJECT_FDW:
|
|
||||||
return RenameForeignDataWrapper(stmt->subname, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_FOREIGN_SERVER:
|
|
||||||
return RenameForeignServer(stmt->subname, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_EVENT_TRIGGER:
|
|
||||||
return RenameEventTrigger(stmt->subname, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_FUNCTION:
|
|
||||||
return RenameFunction(stmt->object, stmt->objarg, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_LANGUAGE:
|
|
||||||
return RenameLanguage(stmt->subname, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_OPCLASS:
|
|
||||||
return RenameOpClass(stmt->object, stmt->subname, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_OPFAMILY:
|
|
||||||
return RenameOpFamily(stmt->object, stmt->subname, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_ROLE:
|
case OBJECT_ROLE:
|
||||||
return RenameRole(stmt->subname, stmt->newname);
|
return RenameRole(stmt->subname, stmt->newname);
|
||||||
|
|
||||||
@ -123,22 +327,44 @@ ExecRenameStmt(RenameStmt *stmt)
|
|||||||
case OBJECT_TRIGGER:
|
case OBJECT_TRIGGER:
|
||||||
return renametrig(stmt);
|
return renametrig(stmt);
|
||||||
|
|
||||||
case OBJECT_TSPARSER:
|
|
||||||
return RenameTSParser(stmt->object, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_TSDICTIONARY:
|
|
||||||
return RenameTSDictionary(stmt->object, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_TSTEMPLATE:
|
|
||||||
return RenameTSTemplate(stmt->object, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_TSCONFIGURATION:
|
|
||||||
return RenameTSConfiguration(stmt->object, stmt->newname);
|
|
||||||
|
|
||||||
case OBJECT_DOMAIN:
|
case OBJECT_DOMAIN:
|
||||||
case OBJECT_TYPE:
|
case OBJECT_TYPE:
|
||||||
return RenameType(stmt);
|
return RenameType(stmt);
|
||||||
|
|
||||||
|
case OBJECT_AGGREGATE:
|
||||||
|
case OBJECT_COLLATION:
|
||||||
|
case OBJECT_CONVERSION:
|
||||||
|
case OBJECT_EVENT_TRIGGER:
|
||||||
|
case OBJECT_FDW:
|
||||||
|
case OBJECT_FOREIGN_SERVER:
|
||||||
|
case OBJECT_FUNCTION:
|
||||||
|
case OBJECT_OPCLASS:
|
||||||
|
case OBJECT_OPFAMILY:
|
||||||
|
case OBJECT_LANGUAGE:
|
||||||
|
case OBJECT_TSCONFIGURATION:
|
||||||
|
case OBJECT_TSDICTIONARY:
|
||||||
|
case OBJECT_TSPARSER:
|
||||||
|
case OBJECT_TSTEMPLATE:
|
||||||
|
{
|
||||||
|
ObjectAddress address;
|
||||||
|
Relation catalog;
|
||||||
|
Relation relation;
|
||||||
|
|
||||||
|
address = get_object_address(stmt->renameType,
|
||||||
|
stmt->object, stmt->objarg,
|
||||||
|
&relation,
|
||||||
|
AccessExclusiveLock, false);
|
||||||
|
Assert(relation == NULL);
|
||||||
|
|
||||||
|
catalog = heap_open(address.classId, RowExclusiveLock);
|
||||||
|
AlterObjectRename_internal(catalog,
|
||||||
|
address.objectId,
|
||||||
|
stmt->newname);
|
||||||
|
heap_close(catalog, RowExclusiveLock);
|
||||||
|
|
||||||
|
return address.objectId;
|
||||||
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
elog(ERROR, "unrecognized rename stmt type: %d",
|
elog(ERROR, "unrecognized rename stmt type: %d",
|
||||||
(int) stmt->renameType);
|
(int) stmt->renameType);
|
||||||
@ -287,43 +513,6 @@ AlterObjectNamespace_oid(Oid classId, Oid objid, Oid nspOid,
|
|||||||
return oldNspOid;
|
return oldNspOid;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Raise an error to the effect that an object of the given name is already
|
|
||||||
* present in the given namespace.
|
|
||||||
*/
|
|
||||||
static void
|
|
||||||
report_namespace_conflict(Oid classId, const char *name, Oid nspOid)
|
|
||||||
{
|
|
||||||
char *msgfmt;
|
|
||||||
|
|
||||||
Assert(OidIsValid(nspOid));
|
|
||||||
switch (classId)
|
|
||||||
{
|
|
||||||
case ConversionRelationId:
|
|
||||||
msgfmt = gettext_noop("conversion \"%s\" already exists in schema \"%s\"");
|
|
||||||
break;
|
|
||||||
case TSParserRelationId:
|
|
||||||
msgfmt = gettext_noop("text search parser \"%s\" already exists in schema \"%s\"");
|
|
||||||
break;
|
|
||||||
case TSDictionaryRelationId:
|
|
||||||
msgfmt = gettext_noop("text search dictionary \"%s\" already exists in schema \"%s\"");
|
|
||||||
break;
|
|
||||||
case TSTemplateRelationId:
|
|
||||||
msgfmt = gettext_noop("text search template \"%s\" already exists in schema \"%s\"");
|
|
||||||
break;
|
|
||||||
case TSConfigRelationId:
|
|
||||||
msgfmt = gettext_noop("text search configuration \"%s\" already exists in schema \"%s\"");
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
elog(ERROR, "unsupported object class %u", classId);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
|
||||||
errmsg(msgfmt, name, get_namespace_name(nspOid))));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Generic function to change the namespace of a given object, for simple
|
* Generic function to change the namespace of a given object, for simple
|
||||||
* cases (won't work for tables, nor other cases where we need to do more
|
* cases (won't work for tables, nor other cases where we need to do more
|
||||||
@ -403,31 +592,34 @@ AlterObjectNamespace_internal(Relation rel, Oid objid, Oid nspOid)
|
|||||||
/*
|
/*
|
||||||
* Check for duplicate name (more friendly than unique-index failure).
|
* Check for duplicate name (more friendly than unique-index failure).
|
||||||
* Since this is just a friendliness check, we can just skip it in cases
|
* Since this is just a friendliness check, we can just skip it in cases
|
||||||
* where there isn't a suitable syscache available.
|
* where there isn't suitable support.
|
||||||
*/
|
*/
|
||||||
if (classId == ProcedureRelationId)
|
if (classId == ProcedureRelationId)
|
||||||
{
|
{
|
||||||
HeapTuple tup;
|
Form_pg_proc proc = (Form_pg_proc) GETSTRUCT(tup);
|
||||||
Form_pg_proc proc;
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(PROCOID, ObjectIdGetDatum(objid));
|
|
||||||
if (!HeapTupleIsValid(tup))
|
|
||||||
elog(ERROR, "cache lookup failed for function %u", objid);
|
|
||||||
proc = (Form_pg_proc) GETSTRUCT(tup);
|
|
||||||
|
|
||||||
IsThereFunctionInNamespace(NameStr(proc->proname), proc->pronargs,
|
IsThereFunctionInNamespace(NameStr(proc->proname), proc->pronargs,
|
||||||
proc->proargtypes, nspOid);
|
proc->proargtypes, nspOid);
|
||||||
heap_freetuple(tup);
|
|
||||||
}
|
}
|
||||||
else if (classId == CollationRelationId)
|
else if (classId == CollationRelationId)
|
||||||
{
|
{
|
||||||
char *collname;
|
Form_pg_collation coll = (Form_pg_collation) GETSTRUCT(tup);
|
||||||
|
|
||||||
collname = get_collation_name(objid);
|
IsThereCollationInNamespace(NameStr(coll->collname), nspOid);
|
||||||
if (!collname)
|
}
|
||||||
elog(ERROR, "cache lookup failed for collation %u", objid);
|
else if (classId == OperatorClassRelationId)
|
||||||
IsThereCollationInNamespace(collname, nspOid);
|
{
|
||||||
pfree(collname);
|
Form_pg_opclass opc = (Form_pg_opclass) GETSTRUCT(tup);
|
||||||
|
|
||||||
|
IsThereOpClassInNamespace(NameStr(opc->opcname),
|
||||||
|
opc->opcmethod, nspOid);
|
||||||
|
}
|
||||||
|
else if (classId == OperatorFamilyRelationId)
|
||||||
|
{
|
||||||
|
Form_pg_opfamily opf = (Form_pg_opfamily) GETSTRUCT(tup);
|
||||||
|
|
||||||
|
IsThereOpFamilyInNamespace(NameStr(opf->opfname),
|
||||||
|
opf->opfmethod, nspOid);
|
||||||
}
|
}
|
||||||
else if (nameCacheId >= 0 &&
|
else if (nameCacheId >= 0 &&
|
||||||
SearchSysCacheExists2(nameCacheId, name,
|
SearchSysCacheExists2(nameCacheId, name,
|
||||||
|
@ -144,54 +144,6 @@ DefineCollation(List *names, List *parameters)
|
|||||||
return newoid;
|
return newoid;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Rename collation
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameCollation(List *name, const char *newname)
|
|
||||||
{
|
|
||||||
Oid collationOid;
|
|
||||||
Oid namespaceOid;
|
|
||||||
HeapTuple tup;
|
|
||||||
Relation rel;
|
|
||||||
AclResult aclresult;
|
|
||||||
|
|
||||||
rel = heap_open(CollationRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
collationOid = get_collation_oid(name, false);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(COLLOID, ObjectIdGetDatum(collationOid));
|
|
||||||
if (!HeapTupleIsValid(tup)) /* should not happen */
|
|
||||||
elog(ERROR, "cache lookup failed for collation %u", collationOid);
|
|
||||||
|
|
||||||
namespaceOid = ((Form_pg_collation) GETSTRUCT(tup))->collnamespace;
|
|
||||||
|
|
||||||
/* make sure the new name doesn't exist */
|
|
||||||
IsThereCollationInNamespace(newname, namespaceOid);
|
|
||||||
|
|
||||||
/* must be owner */
|
|
||||||
if (!pg_collation_ownercheck(collationOid, GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_COLLATION,
|
|
||||||
NameListToString(name));
|
|
||||||
|
|
||||||
/* must have CREATE privilege on namespace */
|
|
||||||
aclresult = pg_namespace_aclcheck(namespaceOid, GetUserId(), ACL_CREATE);
|
|
||||||
if (aclresult != ACLCHECK_OK)
|
|
||||||
aclcheck_error(aclresult, ACL_KIND_NAMESPACE,
|
|
||||||
get_namespace_name(namespaceOid));
|
|
||||||
|
|
||||||
/* rename */
|
|
||||||
namestrcpy(&(((Form_pg_collation) GETSTRUCT(tup))->collname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
heap_close(rel, RowExclusiveLock);
|
|
||||||
|
|
||||||
return collationOid;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Subroutine for ALTER COLLATION SET SCHEMA and RENAME
|
* Subroutine for ALTER COLLATION SET SCHEMA and RENAME
|
||||||
*
|
*
|
||||||
|
@ -114,56 +114,3 @@ CreateConversionCommand(CreateConversionStmt *stmt)
|
|||||||
return ConversionCreate(conversion_name, namespaceId, GetUserId(),
|
return ConversionCreate(conversion_name, namespaceId, GetUserId(),
|
||||||
from_encoding, to_encoding, funcoid, stmt->def);
|
from_encoding, to_encoding, funcoid, stmt->def);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Rename conversion
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameConversion(List *name, const char *newname)
|
|
||||||
{
|
|
||||||
Oid conversionOid;
|
|
||||||
Oid namespaceOid;
|
|
||||||
HeapTuple tup;
|
|
||||||
Relation rel;
|
|
||||||
AclResult aclresult;
|
|
||||||
|
|
||||||
rel = heap_open(ConversionRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
conversionOid = get_conversion_oid(name, false);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(CONVOID, ObjectIdGetDatum(conversionOid));
|
|
||||||
if (!HeapTupleIsValid(tup)) /* should not happen */
|
|
||||||
elog(ERROR, "cache lookup failed for conversion %u", conversionOid);
|
|
||||||
|
|
||||||
namespaceOid = ((Form_pg_conversion) GETSTRUCT(tup))->connamespace;
|
|
||||||
|
|
||||||
/* make sure the new name doesn't exist */
|
|
||||||
if (SearchSysCacheExists2(CONNAMENSP,
|
|
||||||
CStringGetDatum(newname),
|
|
||||||
ObjectIdGetDatum(namespaceOid)))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
|
||||||
errmsg("conversion \"%s\" already exists in schema \"%s\"",
|
|
||||||
newname, get_namespace_name(namespaceOid))));
|
|
||||||
|
|
||||||
/* must be owner */
|
|
||||||
if (!pg_conversion_ownercheck(conversionOid, GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_CONVERSION,
|
|
||||||
NameListToString(name));
|
|
||||||
|
|
||||||
/* must have CREATE privilege on namespace */
|
|
||||||
aclresult = pg_namespace_aclcheck(namespaceOid, GetUserId(), ACL_CREATE);
|
|
||||||
if (aclresult != ACLCHECK_OK)
|
|
||||||
aclcheck_error(aclresult, ACL_KIND_NAMESPACE,
|
|
||||||
get_namespace_name(namespaceOid));
|
|
||||||
|
|
||||||
/* rename */
|
|
||||||
namestrcpy(&(((Form_pg_conversion) GETSTRUCT(tup))->conname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return conversionOid;
|
|
||||||
}
|
|
||||||
|
@ -417,52 +417,6 @@ AlterEventTrigger(AlterEventTrigStmt *stmt)
|
|||||||
return trigoid;
|
return trigoid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Rename event trigger
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameEventTrigger(const char *trigname, const char *newname)
|
|
||||||
{
|
|
||||||
Oid evtId;
|
|
||||||
HeapTuple tup;
|
|
||||||
Relation rel;
|
|
||||||
Form_pg_event_trigger evtForm;
|
|
||||||
|
|
||||||
rel = heap_open(EventTriggerRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
/* newname must be available */
|
|
||||||
if (SearchSysCacheExists1(EVENTTRIGGERNAME, CStringGetDatum(newname)))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
|
||||||
errmsg("event trigger \"%s\" already exists", newname)));
|
|
||||||
|
|
||||||
/* trigname must exists */
|
|
||||||
tup = SearchSysCacheCopy1(EVENTTRIGGERNAME, CStringGetDatum(trigname));
|
|
||||||
if (!HeapTupleIsValid(tup))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_UNDEFINED_OBJECT),
|
|
||||||
errmsg("event trigger \"%s\" does not exist", trigname)));
|
|
||||||
if (!pg_event_trigger_ownercheck(HeapTupleGetOid(tup), GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_EVENT_TRIGGER,
|
|
||||||
trigname);
|
|
||||||
|
|
||||||
evtId = HeapTupleGetOid(tup);
|
|
||||||
|
|
||||||
evtForm = (Form_pg_event_trigger) GETSTRUCT(tup);
|
|
||||||
|
|
||||||
/* tuple is a copy, so we can rename it now */
|
|
||||||
namestrcpy(&(evtForm->evtname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_freetuple(tup);
|
|
||||||
heap_close(rel, RowExclusiveLock);
|
|
||||||
|
|
||||||
return evtId;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Change event trigger's owner -- by name
|
* Change event trigger's owner -- by name
|
||||||
*/
|
*/
|
||||||
|
@ -200,93 +200,6 @@ GetUserOidFromMapping(const char *username, bool missing_ok)
|
|||||||
return get_role_oid(username, missing_ok);
|
return get_role_oid(username, missing_ok);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Rename foreign-data wrapper
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameForeignDataWrapper(const char *oldname, const char *newname)
|
|
||||||
{
|
|
||||||
Oid fdwId;
|
|
||||||
HeapTuple tup;
|
|
||||||
Relation rel;
|
|
||||||
|
|
||||||
rel = heap_open(ForeignDataWrapperRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(FOREIGNDATAWRAPPERNAME, CStringGetDatum(oldname));
|
|
||||||
if (!HeapTupleIsValid(tup))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_UNDEFINED_OBJECT),
|
|
||||||
errmsg("foreign-data wrapper \"%s\" does not exist", oldname)));
|
|
||||||
|
|
||||||
fdwId = HeapTupleGetOid(tup);
|
|
||||||
|
|
||||||
/* make sure the new name doesn't exist */
|
|
||||||
if (SearchSysCacheExists1(FOREIGNDATAWRAPPERNAME, CStringGetDatum(newname)))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
|
||||||
errmsg("foreign-data wrapper \"%s\" already exists", newname)));
|
|
||||||
|
|
||||||
/* must be owner of FDW */
|
|
||||||
if (!pg_foreign_data_wrapper_ownercheck(HeapTupleGetOid(tup), GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_FDW,
|
|
||||||
oldname);
|
|
||||||
|
|
||||||
/* rename */
|
|
||||||
namestrcpy(&(((Form_pg_foreign_data_wrapper) GETSTRUCT(tup))->fdwname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return fdwId;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Rename foreign server
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameForeignServer(const char *oldname, const char *newname)
|
|
||||||
{
|
|
||||||
Oid srvId;
|
|
||||||
HeapTuple tup;
|
|
||||||
Relation rel;
|
|
||||||
|
|
||||||
rel = heap_open(ForeignServerRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(FOREIGNSERVERNAME, CStringGetDatum(oldname));
|
|
||||||
if (!HeapTupleIsValid(tup))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_UNDEFINED_OBJECT),
|
|
||||||
errmsg("server \"%s\" does not exist", oldname)));
|
|
||||||
|
|
||||||
srvId = HeapTupleGetOid(tup);
|
|
||||||
|
|
||||||
/* make sure the new name doesn't exist */
|
|
||||||
if (SearchSysCacheExists1(FOREIGNSERVERNAME, CStringGetDatum(newname)))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
|
||||||
errmsg("server \"%s\" already exists", newname)));
|
|
||||||
|
|
||||||
/* must be owner of server */
|
|
||||||
if (!pg_foreign_server_ownercheck(HeapTupleGetOid(tup), GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_FOREIGN_SERVER,
|
|
||||||
oldname);
|
|
||||||
|
|
||||||
/* rename */
|
|
||||||
namestrcpy(&(((Form_pg_foreign_server) GETSTRUCT(tup))->srvname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return srvId;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Internal workhorse for changing a data wrapper's owner.
|
* Internal workhorse for changing a data wrapper's owner.
|
||||||
*
|
*
|
||||||
|
@ -1036,65 +1036,6 @@ RemoveFunctionById(Oid funcOid)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Rename function
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameFunction(List *name, List *argtypes, const char *newname)
|
|
||||||
{
|
|
||||||
Oid procOid;
|
|
||||||
Oid namespaceOid;
|
|
||||||
HeapTuple tup;
|
|
||||||
Form_pg_proc procForm;
|
|
||||||
Relation rel;
|
|
||||||
AclResult aclresult;
|
|
||||||
|
|
||||||
rel = heap_open(ProcedureRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
procOid = LookupFuncNameTypeNames(name, argtypes, false);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(PROCOID, ObjectIdGetDatum(procOid));
|
|
||||||
if (!HeapTupleIsValid(tup)) /* should not happen */
|
|
||||||
elog(ERROR, "cache lookup failed for function %u", procOid);
|
|
||||||
procForm = (Form_pg_proc) GETSTRUCT(tup);
|
|
||||||
|
|
||||||
if (procForm->proisagg)
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_WRONG_OBJECT_TYPE),
|
|
||||||
errmsg("\"%s\" is an aggregate function",
|
|
||||||
NameListToString(name)),
|
|
||||||
errhint("Use ALTER AGGREGATE to rename aggregate functions.")));
|
|
||||||
|
|
||||||
namespaceOid = procForm->pronamespace;
|
|
||||||
|
|
||||||
/* make sure the new name doesn't exist */
|
|
||||||
IsThereFunctionInNamespace(newname, procForm->pronargs,
|
|
||||||
procForm->proargtypes,
|
|
||||||
namespaceOid);
|
|
||||||
|
|
||||||
/* must be owner */
|
|
||||||
if (!pg_proc_ownercheck(procOid, GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_PROC,
|
|
||||||
NameListToString(name));
|
|
||||||
|
|
||||||
/* must have CREATE privilege on namespace */
|
|
||||||
aclresult = pg_namespace_aclcheck(namespaceOid, GetUserId(), ACL_CREATE);
|
|
||||||
if (aclresult != ACLCHECK_OK)
|
|
||||||
aclcheck_error(aclresult, ACL_KIND_NAMESPACE,
|
|
||||||
get_namespace_name(namespaceOid));
|
|
||||||
|
|
||||||
/* rename */
|
|
||||||
namestrcpy(&(procForm->proname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return procOid;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Implements the ALTER FUNCTION utility command (except for the
|
* Implements the ALTER FUNCTION utility command (except for the
|
||||||
* RENAME and OWNER clauses, which are handled as part of the generic
|
* RENAME and OWNER clauses, which are handled as part of the generic
|
||||||
@ -1677,7 +1618,7 @@ DropCastById(Oid castOid)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Subroutine for ALTER FUNCTION/AGGREGATE SET SCHEMA
|
* Subroutine for ALTER FUNCTION/AGGREGATE SET SCHEMA/RENAME
|
||||||
*
|
*
|
||||||
* Is there a function with the given name and signature already in the given
|
* Is there a function with the given name and signature already in the given
|
||||||
* namespace? If so, raise an appropriate error message.
|
* namespace? If so, raise an appropriate error message.
|
||||||
|
@ -1660,156 +1660,65 @@ RemoveAmProcEntryById(Oid entryOid)
|
|||||||
heap_close(rel, RowExclusiveLock);
|
heap_close(rel, RowExclusiveLock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static char *
|
||||||
/*
|
get_am_name(Oid amOid)
|
||||||
* Rename opclass
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameOpClass(List *name, const char *access_method, const char *newname)
|
|
||||||
{
|
{
|
||||||
Oid opcOid;
|
|
||||||
Oid amOid;
|
|
||||||
Oid namespaceOid;
|
|
||||||
HeapTuple origtup;
|
|
||||||
HeapTuple tup;
|
HeapTuple tup;
|
||||||
Relation rel;
|
char *result = NULL;
|
||||||
AclResult aclresult;
|
|
||||||
|
|
||||||
amOid = get_am_oid(access_method, false);
|
tup = SearchSysCache1(AMOID, ObjectIdGetDatum(amOid));
|
||||||
|
if (HeapTupleIsValid(tup))
|
||||||
rel = heap_open(OperatorClassRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
/* Look up the opclass. */
|
|
||||||
origtup = OpClassCacheLookup(amOid, name, false);
|
|
||||||
tup = heap_copytuple(origtup);
|
|
||||||
ReleaseSysCache(origtup);
|
|
||||||
opcOid = HeapTupleGetOid(tup);
|
|
||||||
namespaceOid = ((Form_pg_opclass) GETSTRUCT(tup))->opcnamespace;
|
|
||||||
|
|
||||||
/* make sure the new name doesn't exist */
|
|
||||||
if (SearchSysCacheExists3(CLAAMNAMENSP,
|
|
||||||
ObjectIdGetDatum(amOid),
|
|
||||||
CStringGetDatum(newname),
|
|
||||||
ObjectIdGetDatum(namespaceOid)))
|
|
||||||
{
|
{
|
||||||
ereport(ERROR,
|
result = pstrdup(NameStr(((Form_pg_am) GETSTRUCT(tup))->amname));
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
ReleaseSysCache(tup);
|
||||||
errmsg("operator class \"%s\" for access method \"%s\" already exists in schema \"%s\"",
|
|
||||||
newname, access_method,
|
|
||||||
get_namespace_name(namespaceOid))));
|
|
||||||
}
|
}
|
||||||
|
return result;
|
||||||
/* must be owner */
|
|
||||||
if (!pg_opclass_ownercheck(opcOid, GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_OPCLASS,
|
|
||||||
NameListToString(name));
|
|
||||||
|
|
||||||
/* must have CREATE privilege on namespace */
|
|
||||||
aclresult = pg_namespace_aclcheck(namespaceOid, GetUserId(), ACL_CREATE);
|
|
||||||
if (aclresult != ACLCHECK_OK)
|
|
||||||
aclcheck_error(aclresult, ACL_KIND_NAMESPACE,
|
|
||||||
get_namespace_name(namespaceOid));
|
|
||||||
|
|
||||||
/* rename */
|
|
||||||
namestrcpy(&(((Form_pg_opclass) GETSTRUCT(tup))->opcname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return opcOid;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Rename opfamily
|
* Subroutine for ALTER OPERATOR CLASS SET SCHEMA/RENAME
|
||||||
|
*
|
||||||
|
* Is there an operator class with the given name and signature already
|
||||||
|
* in the given namespace? If so, raise an appropriate error message.
|
||||||
*/
|
*/
|
||||||
Oid
|
void
|
||||||
RenameOpFamily(List *name, const char *access_method, const char *newname)
|
IsThereOpClassInNamespace(const char *opcname, Oid opcmethod,
|
||||||
|
Oid opcnamespace)
|
||||||
{
|
{
|
||||||
Oid opfOid;
|
/* make sure the new name doesn't exist */
|
||||||
Oid amOid;
|
if (SearchSysCacheExists3(CLAAMNAMENSP,
|
||||||
Oid namespaceOid;
|
ObjectIdGetDatum(opcmethod),
|
||||||
char *schemaname;
|
CStringGetDatum(opcname),
|
||||||
char *opfname;
|
ObjectIdGetDatum(opcnamespace)))
|
||||||
HeapTuple tup;
|
ereport(ERROR,
|
||||||
Relation rel;
|
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
||||||
AclResult aclresult;
|
errmsg("operator class \"%s\" for access method \"%s\" already exists in schema \"%s\"",
|
||||||
|
opcname,
|
||||||
amOid = get_am_oid(access_method, false);
|
get_am_name(opcmethod),
|
||||||
|
get_namespace_name(opcnamespace))));
|
||||||
rel = heap_open(OperatorFamilyRelationId, RowExclusiveLock);
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Look up the opfamily
|
|
||||||
*/
|
|
||||||
DeconstructQualifiedName(name, &schemaname, &opfname);
|
|
||||||
|
|
||||||
if (schemaname)
|
|
||||||
{
|
|
||||||
namespaceOid = LookupExplicitNamespace(schemaname);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy3(OPFAMILYAMNAMENSP,
|
|
||||||
ObjectIdGetDatum(amOid),
|
|
||||||
PointerGetDatum(opfname),
|
|
||||||
ObjectIdGetDatum(namespaceOid));
|
|
||||||
if (!HeapTupleIsValid(tup))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_UNDEFINED_OBJECT),
|
|
||||||
errmsg("operator family \"%s\" does not exist for access method \"%s\"",
|
|
||||||
opfname, access_method)));
|
|
||||||
|
|
||||||
opfOid = HeapTupleGetOid(tup);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
opfOid = OpfamilynameGetOpfid(amOid, opfname);
|
|
||||||
if (!OidIsValid(opfOid))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_UNDEFINED_OBJECT),
|
|
||||||
errmsg("operator family \"%s\" does not exist for access method \"%s\"",
|
|
||||||
opfname, access_method)));
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(OPFAMILYOID, ObjectIdGetDatum(opfOid));
|
|
||||||
if (!HeapTupleIsValid(tup)) /* should not happen */
|
|
||||||
elog(ERROR, "cache lookup failed for opfamily %u", opfOid);
|
|
||||||
|
|
||||||
namespaceOid = ((Form_pg_opfamily) GETSTRUCT(tup))->opfnamespace;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Subroutine for ALTER OPERATOR FAMILY SET SCHEMA/RENAME
|
||||||
|
*
|
||||||
|
* Is there an operator family with the given name and signature already
|
||||||
|
* in the given namespace? If so, raise an appropriate error message.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
IsThereOpFamilyInNamespace(const char *opfname, Oid opfmethod,
|
||||||
|
Oid opfnamespace)
|
||||||
|
{
|
||||||
/* make sure the new name doesn't exist */
|
/* make sure the new name doesn't exist */
|
||||||
if (SearchSysCacheExists3(OPFAMILYAMNAMENSP,
|
if (SearchSysCacheExists3(OPFAMILYAMNAMENSP,
|
||||||
ObjectIdGetDatum(amOid),
|
ObjectIdGetDatum(opfmethod),
|
||||||
CStringGetDatum(newname),
|
CStringGetDatum(opfname),
|
||||||
ObjectIdGetDatum(namespaceOid)))
|
ObjectIdGetDatum(opfnamespace)))
|
||||||
{
|
|
||||||
ereport(ERROR,
|
ereport(ERROR,
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
||||||
errmsg("operator family \"%s\" for access method \"%s\" already exists in schema \"%s\"",
|
errmsg("operator family \"%s\" for access method \"%s\" already exists in schema \"%s\"",
|
||||||
newname, access_method,
|
opfname,
|
||||||
get_namespace_name(namespaceOid))));
|
get_am_name(opfmethod),
|
||||||
}
|
get_namespace_name(opfnamespace))));
|
||||||
|
|
||||||
/* must be owner */
|
|
||||||
if (!pg_opfamily_ownercheck(opfOid, GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_OPFAMILY,
|
|
||||||
NameListToString(name));
|
|
||||||
|
|
||||||
/* must have CREATE privilege on namespace */
|
|
||||||
aclresult = pg_namespace_aclcheck(namespaceOid, GetUserId(), ACL_CREATE);
|
|
||||||
if (aclresult != ACLCHECK_OK)
|
|
||||||
aclcheck_error(aclresult, ACL_KIND_NAMESPACE,
|
|
||||||
get_namespace_name(namespaceOid));
|
|
||||||
|
|
||||||
/* rename */
|
|
||||||
namestrcpy(&(((Form_pg_opfamily) GETSTRUCT(tup))->opfname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return opfOid;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -536,48 +536,6 @@ DropProceduralLanguageById(Oid langOid)
|
|||||||
heap_close(rel, RowExclusiveLock);
|
heap_close(rel, RowExclusiveLock);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Rename language
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameLanguage(const char *oldname, const char *newname)
|
|
||||||
{
|
|
||||||
Oid lanId;
|
|
||||||
HeapTuple tup;
|
|
||||||
Relation rel;
|
|
||||||
|
|
||||||
rel = heap_open(LanguageRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(LANGNAME, CStringGetDatum(oldname));
|
|
||||||
if (!HeapTupleIsValid(tup))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_UNDEFINED_OBJECT),
|
|
||||||
errmsg("language \"%s\" does not exist", oldname)));
|
|
||||||
|
|
||||||
lanId = HeapTupleGetOid(tup);
|
|
||||||
|
|
||||||
/* make sure the new name doesn't exist */
|
|
||||||
if (SearchSysCacheExists1(LANGNAME, CStringGetDatum(newname)))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
|
||||||
errmsg("language \"%s\" already exists", newname)));
|
|
||||||
|
|
||||||
/* must be owner of PL */
|
|
||||||
if (!pg_language_ownercheck(HeapTupleGetOid(tup), GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_LANGUAGE,
|
|
||||||
oldname);
|
|
||||||
|
|
||||||
/* rename */
|
|
||||||
namestrcpy(&(((Form_pg_language) GETSTRUCT(tup))->lanname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return lanId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* get_language_oid - given a language name, look up the OID
|
* get_language_oid - given a language name, look up the OID
|
||||||
*
|
*
|
||||||
|
@ -305,51 +305,6 @@ RemoveTSParserById(Oid prsId)
|
|||||||
heap_close(relation, RowExclusiveLock);
|
heap_close(relation, RowExclusiveLock);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* ALTER TEXT SEARCH PARSER RENAME
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameTSParser(List *oldname, const char *newname)
|
|
||||||
{
|
|
||||||
HeapTuple tup;
|
|
||||||
Relation rel;
|
|
||||||
Oid prsId;
|
|
||||||
Oid namespaceOid;
|
|
||||||
|
|
||||||
if (!superuser())
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
|
|
||||||
errmsg("must be superuser to rename text search parsers")));
|
|
||||||
|
|
||||||
rel = heap_open(TSParserRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
prsId = get_ts_parser_oid(oldname, false);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(TSPARSEROID, ObjectIdGetDatum(prsId));
|
|
||||||
|
|
||||||
if (!HeapTupleIsValid(tup)) /* should not happen */
|
|
||||||
elog(ERROR, "cache lookup failed for text search parser %u", prsId);
|
|
||||||
|
|
||||||
namespaceOid = ((Form_pg_ts_parser) GETSTRUCT(tup))->prsnamespace;
|
|
||||||
|
|
||||||
if (SearchSysCacheExists2(TSPARSERNAMENSP,
|
|
||||||
PointerGetDatum(newname),
|
|
||||||
ObjectIdGetDatum(namespaceOid)))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
|
||||||
errmsg("text search parser \"%s\" already exists",
|
|
||||||
newname)));
|
|
||||||
|
|
||||||
namestrcpy(&(((Form_pg_ts_parser) GETSTRUCT(tup))->prsname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return prsId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ---------------------- TS Dictionary commands -----------------------*/
|
/* ---------------------- TS Dictionary commands -----------------------*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -534,59 +489,6 @@ DefineTSDictionary(List *names, List *parameters)
|
|||||||
return dictOid;
|
return dictOid;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* ALTER TEXT SEARCH DICTIONARY RENAME
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameTSDictionary(List *oldname, const char *newname)
|
|
||||||
{
|
|
||||||
HeapTuple tup;
|
|
||||||
Relation rel;
|
|
||||||
Oid dictId;
|
|
||||||
Oid namespaceOid;
|
|
||||||
AclResult aclresult;
|
|
||||||
|
|
||||||
rel = heap_open(TSDictionaryRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
dictId = get_ts_dict_oid(oldname, false);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(TSDICTOID, ObjectIdGetDatum(dictId));
|
|
||||||
|
|
||||||
if (!HeapTupleIsValid(tup)) /* should not happen */
|
|
||||||
elog(ERROR, "cache lookup failed for text search dictionary %u",
|
|
||||||
dictId);
|
|
||||||
|
|
||||||
namespaceOid = ((Form_pg_ts_dict) GETSTRUCT(tup))->dictnamespace;
|
|
||||||
|
|
||||||
if (SearchSysCacheExists2(TSDICTNAMENSP,
|
|
||||||
PointerGetDatum(newname),
|
|
||||||
ObjectIdGetDatum(namespaceOid)))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
|
||||||
errmsg("text search dictionary \"%s\" already exists",
|
|
||||||
newname)));
|
|
||||||
|
|
||||||
/* must be owner */
|
|
||||||
if (!pg_ts_dict_ownercheck(dictId, GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_TSDICTIONARY,
|
|
||||||
NameListToString(oldname));
|
|
||||||
|
|
||||||
/* must have CREATE privilege on namespace */
|
|
||||||
aclresult = pg_namespace_aclcheck(namespaceOid, GetUserId(), ACL_CREATE);
|
|
||||||
if (aclresult != ACLCHECK_OK)
|
|
||||||
aclcheck_error(aclresult, ACL_KIND_NAMESPACE,
|
|
||||||
get_namespace_name(namespaceOid));
|
|
||||||
|
|
||||||
namestrcpy(&(((Form_pg_ts_dict) GETSTRUCT(tup))->dictname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return dictId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Guts of TS dictionary deletion.
|
* Guts of TS dictionary deletion.
|
||||||
*/
|
*/
|
||||||
@ -904,52 +806,6 @@ DefineTSTemplate(List *names, List *parameters)
|
|||||||
return tmplOid;
|
return tmplOid;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* ALTER TEXT SEARCH TEMPLATE RENAME
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameTSTemplate(List *oldname, const char *newname)
|
|
||||||
{
|
|
||||||
HeapTuple tup;
|
|
||||||
Relation rel;
|
|
||||||
Oid tmplId;
|
|
||||||
Oid namespaceOid;
|
|
||||||
|
|
||||||
if (!superuser())
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
|
|
||||||
errmsg("must be superuser to rename text search templates")));
|
|
||||||
|
|
||||||
rel = heap_open(TSTemplateRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
tmplId = get_ts_template_oid(oldname, false);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(TSTEMPLATEOID, ObjectIdGetDatum(tmplId));
|
|
||||||
|
|
||||||
if (!HeapTupleIsValid(tup)) /* should not happen */
|
|
||||||
elog(ERROR, "cache lookup failed for text search template %u",
|
|
||||||
tmplId);
|
|
||||||
|
|
||||||
namespaceOid = ((Form_pg_ts_template) GETSTRUCT(tup))->tmplnamespace;
|
|
||||||
|
|
||||||
if (SearchSysCacheExists2(TSTEMPLATENAMENSP,
|
|
||||||
PointerGetDatum(newname),
|
|
||||||
ObjectIdGetDatum(namespaceOid)))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
|
||||||
errmsg("text search template \"%s\" already exists",
|
|
||||||
newname)));
|
|
||||||
|
|
||||||
namestrcpy(&(((Form_pg_ts_template) GETSTRUCT(tup))->tmplname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return tmplId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Guts of TS template deletion.
|
* Guts of TS template deletion.
|
||||||
*/
|
*/
|
||||||
@ -1248,58 +1104,6 @@ DefineTSConfiguration(List *names, List *parameters)
|
|||||||
return cfgOid;
|
return cfgOid;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* ALTER TEXT SEARCH CONFIGURATION RENAME
|
|
||||||
*/
|
|
||||||
Oid
|
|
||||||
RenameTSConfiguration(List *oldname, const char *newname)
|
|
||||||
{
|
|
||||||
HeapTuple tup;
|
|
||||||
Relation rel;
|
|
||||||
Oid cfgId;
|
|
||||||
AclResult aclresult;
|
|
||||||
Oid namespaceOid;
|
|
||||||
|
|
||||||
rel = heap_open(TSConfigRelationId, RowExclusiveLock);
|
|
||||||
|
|
||||||
cfgId = get_ts_config_oid(oldname, false);
|
|
||||||
|
|
||||||
tup = SearchSysCacheCopy1(TSCONFIGOID, ObjectIdGetDatum(cfgId));
|
|
||||||
|
|
||||||
if (!HeapTupleIsValid(tup)) /* should not happen */
|
|
||||||
elog(ERROR, "cache lookup failed for text search configuration %u",
|
|
||||||
cfgId);
|
|
||||||
|
|
||||||
namespaceOid = ((Form_pg_ts_config) GETSTRUCT(tup))->cfgnamespace;
|
|
||||||
|
|
||||||
if (SearchSysCacheExists2(TSCONFIGNAMENSP,
|
|
||||||
PointerGetDatum(newname),
|
|
||||||
ObjectIdGetDatum(namespaceOid)))
|
|
||||||
ereport(ERROR,
|
|
||||||
(errcode(ERRCODE_DUPLICATE_OBJECT),
|
|
||||||
errmsg("text search configuration \"%s\" already exists",
|
|
||||||
newname)));
|
|
||||||
|
|
||||||
/* must be owner */
|
|
||||||
if (!pg_ts_config_ownercheck(cfgId, GetUserId()))
|
|
||||||
aclcheck_error(ACLCHECK_NOT_OWNER, ACL_KIND_TSCONFIGURATION,
|
|
||||||
NameListToString(oldname));
|
|
||||||
|
|
||||||
/* must have CREATE privilege on namespace */
|
|
||||||
aclresult = pg_namespace_aclcheck(namespaceOid, GetUserId(), ACL_CREATE);
|
|
||||||
aclcheck_error(aclresult, ACL_KIND_NAMESPACE,
|
|
||||||
get_namespace_name(namespaceOid));
|
|
||||||
|
|
||||||
namestrcpy(&(((Form_pg_ts_config) GETSTRUCT(tup))->cfgname), newname);
|
|
||||||
simple_heap_update(rel, &tup->t_self, tup);
|
|
||||||
CatalogUpdateIndexes(rel, tup);
|
|
||||||
|
|
||||||
heap_close(rel, NoLock);
|
|
||||||
heap_freetuple(tup);
|
|
||||||
|
|
||||||
return cfgId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Guts of TS configuration deletion.
|
* Guts of TS configuration deletion.
|
||||||
*/
|
*/
|
||||||
|
@ -6776,7 +6776,7 @@ RenameStmt: ALTER AGGREGATE func_name aggr_args RENAME TO name
|
|||||||
{
|
{
|
||||||
RenameStmt *n = makeNode(RenameStmt);
|
RenameStmt *n = makeNode(RenameStmt);
|
||||||
n->renameType = OBJECT_FDW;
|
n->renameType = OBJECT_FDW;
|
||||||
n->subname = $5;
|
n->object = list_make1(makeString($5));
|
||||||
n->newname = $8;
|
n->newname = $8;
|
||||||
n->missing_ok = false;
|
n->missing_ok = false;
|
||||||
$$ = (Node *)n;
|
$$ = (Node *)n;
|
||||||
@ -6804,7 +6804,7 @@ RenameStmt: ALTER AGGREGATE func_name aggr_args RENAME TO name
|
|||||||
{
|
{
|
||||||
RenameStmt *n = makeNode(RenameStmt);
|
RenameStmt *n = makeNode(RenameStmt);
|
||||||
n->renameType = OBJECT_LANGUAGE;
|
n->renameType = OBJECT_LANGUAGE;
|
||||||
n->subname = $4;
|
n->object = list_make1(makeString($4));
|
||||||
n->newname = $7;
|
n->newname = $7;
|
||||||
n->missing_ok = false;
|
n->missing_ok = false;
|
||||||
$$ = (Node *)n;
|
$$ = (Node *)n;
|
||||||
@ -6814,7 +6814,7 @@ RenameStmt: ALTER AGGREGATE func_name aggr_args RENAME TO name
|
|||||||
RenameStmt *n = makeNode(RenameStmt);
|
RenameStmt *n = makeNode(RenameStmt);
|
||||||
n->renameType = OBJECT_OPCLASS;
|
n->renameType = OBJECT_OPCLASS;
|
||||||
n->object = $4;
|
n->object = $4;
|
||||||
n->subname = $6;
|
n->objarg = list_make1(makeString($6));
|
||||||
n->newname = $9;
|
n->newname = $9;
|
||||||
n->missing_ok = false;
|
n->missing_ok = false;
|
||||||
$$ = (Node *)n;
|
$$ = (Node *)n;
|
||||||
@ -6824,7 +6824,7 @@ RenameStmt: ALTER AGGREGATE func_name aggr_args RENAME TO name
|
|||||||
RenameStmt *n = makeNode(RenameStmt);
|
RenameStmt *n = makeNode(RenameStmt);
|
||||||
n->renameType = OBJECT_OPFAMILY;
|
n->renameType = OBJECT_OPFAMILY;
|
||||||
n->object = $4;
|
n->object = $4;
|
||||||
n->subname = $6;
|
n->objarg = list_make1(makeString($6));
|
||||||
n->newname = $9;
|
n->newname = $9;
|
||||||
n->missing_ok = false;
|
n->missing_ok = false;
|
||||||
$$ = (Node *)n;
|
$$ = (Node *)n;
|
||||||
@ -6842,7 +6842,7 @@ RenameStmt: ALTER AGGREGATE func_name aggr_args RENAME TO name
|
|||||||
{
|
{
|
||||||
RenameStmt *n = makeNode(RenameStmt);
|
RenameStmt *n = makeNode(RenameStmt);
|
||||||
n->renameType = OBJECT_FOREIGN_SERVER;
|
n->renameType = OBJECT_FOREIGN_SERVER;
|
||||||
n->subname = $3;
|
n->object = list_make1(makeString($3));
|
||||||
n->newname = $6;
|
n->newname = $6;
|
||||||
n->missing_ok = false;
|
n->missing_ok = false;
|
||||||
$$ = (Node *)n;
|
$$ = (Node *)n;
|
||||||
@ -7015,7 +7015,7 @@ RenameStmt: ALTER AGGREGATE func_name aggr_args RENAME TO name
|
|||||||
{
|
{
|
||||||
RenameStmt *n = makeNode(RenameStmt);
|
RenameStmt *n = makeNode(RenameStmt);
|
||||||
n->renameType = OBJECT_EVENT_TRIGGER;
|
n->renameType = OBJECT_EVENT_TRIGGER;
|
||||||
n->subname = $4;
|
n->object = list_make1(makeString($4));
|
||||||
n->newname = $7;
|
n->newname = $7;
|
||||||
$$ = (Node *)n;
|
$$ = (Node *)n;
|
||||||
}
|
}
|
||||||
|
@ -18,7 +18,6 @@
|
|||||||
#include "nodes/parsenodes.h"
|
#include "nodes/parsenodes.h"
|
||||||
|
|
||||||
extern Oid DefineCollation(List *names, List *parameters);
|
extern Oid DefineCollation(List *names, List *parameters);
|
||||||
extern Oid RenameCollation(List *name, const char *newname);
|
extern void IsThereCollationInNamespace(const char *collname, Oid nspOid);
|
||||||
extern void IsThereCollationInNamespace(const char *collname, Oid newNspOid);
|
|
||||||
|
|
||||||
#endif /* COLLATIONCMDS_H */
|
#endif /* COLLATIONCMDS_H */
|
||||||
|
@ -18,6 +18,5 @@
|
|||||||
#include "nodes/parsenodes.h"
|
#include "nodes/parsenodes.h"
|
||||||
|
|
||||||
extern Oid CreateConversionCommand(CreateConversionStmt *parsetree);
|
extern Oid CreateConversionCommand(CreateConversionStmt *parsetree);
|
||||||
extern Oid RenameConversion(List *name, const char *newname);
|
|
||||||
|
|
||||||
#endif /* CONVERSIONCMDS_H */
|
#endif /* CONVERSIONCMDS_H */
|
||||||
|
@ -46,7 +46,6 @@ extern Oid CreateFunction(CreateFunctionStmt *stmt, const char *queryString);
|
|||||||
extern void RemoveFunctionById(Oid funcOid);
|
extern void RemoveFunctionById(Oid funcOid);
|
||||||
extern void SetFunctionReturnType(Oid funcOid, Oid newRetType);
|
extern void SetFunctionReturnType(Oid funcOid, Oid newRetType);
|
||||||
extern void SetFunctionArgType(Oid funcOid, int argIndex, Oid newArgType);
|
extern void SetFunctionArgType(Oid funcOid, int argIndex, Oid newArgType);
|
||||||
extern Oid RenameFunction(List *name, List *argtypes, const char *newname);
|
|
||||||
extern Oid AlterFunction(AlterFunctionStmt *stmt);
|
extern Oid AlterFunction(AlterFunctionStmt *stmt);
|
||||||
extern Oid CreateCast(CreateCastStmt *stmt);
|
extern Oid CreateCast(CreateCastStmt *stmt);
|
||||||
extern void DropCastById(Oid castOid);
|
extern void DropCastById(Oid castOid);
|
||||||
@ -62,7 +61,6 @@ extern void RemoveOperatorById(Oid operOid);
|
|||||||
/* commands/aggregatecmds.c */
|
/* commands/aggregatecmds.c */
|
||||||
extern Oid DefineAggregate(List *name, List *args, bool oldstyle,
|
extern Oid DefineAggregate(List *name, List *args, bool oldstyle,
|
||||||
List *parameters);
|
List *parameters);
|
||||||
extern Oid RenameAggregate(List *name, List *args, const char *newname);
|
|
||||||
|
|
||||||
/* commands/opclasscmds.c */
|
/* commands/opclasscmds.c */
|
||||||
extern Oid DefineOpClass(CreateOpClassStmt *stmt);
|
extern Oid DefineOpClass(CreateOpClassStmt *stmt);
|
||||||
@ -72,28 +70,26 @@ extern void RemoveOpClassById(Oid opclassOid);
|
|||||||
extern void RemoveOpFamilyById(Oid opfamilyOid);
|
extern void RemoveOpFamilyById(Oid opfamilyOid);
|
||||||
extern void RemoveAmOpEntryById(Oid entryOid);
|
extern void RemoveAmOpEntryById(Oid entryOid);
|
||||||
extern void RemoveAmProcEntryById(Oid entryOid);
|
extern void RemoveAmProcEntryById(Oid entryOid);
|
||||||
extern Oid RenameOpClass(List *name, const char *access_method, const char *newname);
|
extern void IsThereOpClassInNamespace(const char *opcname, Oid opcmethod,
|
||||||
extern Oid RenameOpFamily(List *name, const char *access_method, const char *newname);
|
Oid opcnamespace);
|
||||||
|
extern void IsThereOpFamilyInNamespace(const char *opfname, Oid opfmethod,
|
||||||
|
Oid opfnamespace);
|
||||||
extern Oid get_am_oid(const char *amname, bool missing_ok);
|
extern Oid get_am_oid(const char *amname, bool missing_ok);
|
||||||
extern Oid get_opclass_oid(Oid amID, List *opclassname, bool missing_ok);
|
extern Oid get_opclass_oid(Oid amID, List *opclassname, bool missing_ok);
|
||||||
extern Oid get_opfamily_oid(Oid amID, List *opfamilyname, bool missing_ok);
|
extern Oid get_opfamily_oid(Oid amID, List *opfamilyname, bool missing_ok);
|
||||||
|
|
||||||
/* commands/tsearchcmds.c */
|
/* commands/tsearchcmds.c */
|
||||||
extern Oid DefineTSParser(List *names, List *parameters);
|
extern Oid DefineTSParser(List *names, List *parameters);
|
||||||
extern Oid RenameTSParser(List *oldname, const char *newname);
|
|
||||||
extern void RemoveTSParserById(Oid prsId);
|
extern void RemoveTSParserById(Oid prsId);
|
||||||
|
|
||||||
extern Oid DefineTSDictionary(List *names, List *parameters);
|
extern Oid DefineTSDictionary(List *names, List *parameters);
|
||||||
extern Oid RenameTSDictionary(List *oldname, const char *newname);
|
|
||||||
extern void RemoveTSDictionaryById(Oid dictId);
|
extern void RemoveTSDictionaryById(Oid dictId);
|
||||||
extern Oid AlterTSDictionary(AlterTSDictionaryStmt *stmt);
|
extern Oid AlterTSDictionary(AlterTSDictionaryStmt *stmt);
|
||||||
|
|
||||||
extern Oid DefineTSTemplate(List *names, List *parameters);
|
extern Oid DefineTSTemplate(List *names, List *parameters);
|
||||||
extern Oid RenameTSTemplate(List *oldname, const char *newname);
|
|
||||||
extern void RemoveTSTemplateById(Oid tmplId);
|
extern void RemoveTSTemplateById(Oid tmplId);
|
||||||
|
|
||||||
extern Oid DefineTSConfiguration(List *names, List *parameters);
|
extern Oid DefineTSConfiguration(List *names, List *parameters);
|
||||||
extern Oid RenameTSConfiguration(List *oldname, const char *newname);
|
|
||||||
extern void RemoveTSConfigurationById(Oid cfgId);
|
extern void RemoveTSConfigurationById(Oid cfgId);
|
||||||
extern Oid AlterTSConfiguration(AlterTSConfigurationStmt *stmt);
|
extern Oid AlterTSConfiguration(AlterTSConfigurationStmt *stmt);
|
||||||
|
|
||||||
@ -101,8 +97,6 @@ extern text *serialize_deflist(List *deflist);
|
|||||||
extern List *deserialize_deflist(Datum txt);
|
extern List *deserialize_deflist(Datum txt);
|
||||||
|
|
||||||
/* commands/foreigncmds.c */
|
/* commands/foreigncmds.c */
|
||||||
extern Oid RenameForeignServer(const char *oldname, const char *newname);
|
|
||||||
extern Oid RenameForeignDataWrapper(const char *oldname, const char *newname);
|
|
||||||
extern Oid AlterForeignServerOwner(const char *name, Oid newOwnerId);
|
extern Oid AlterForeignServerOwner(const char *name, Oid newOwnerId);
|
||||||
extern void AlterForeignServerOwner_oid(Oid, Oid newOwnerId);
|
extern void AlterForeignServerOwner_oid(Oid, Oid newOwnerId);
|
||||||
extern Oid AlterForeignDataWrapperOwner(const char *name, Oid newOwnerId);
|
extern Oid AlterForeignDataWrapperOwner(const char *name, Oid newOwnerId);
|
||||||
|
@ -36,7 +36,6 @@ extern void RemoveEventTriggerById(Oid ctrigOid);
|
|||||||
extern Oid get_event_trigger_oid(const char *trigname, bool missing_ok);
|
extern Oid get_event_trigger_oid(const char *trigname, bool missing_ok);
|
||||||
|
|
||||||
extern Oid AlterEventTrigger(AlterEventTrigStmt *stmt);
|
extern Oid AlterEventTrigger(AlterEventTrigStmt *stmt);
|
||||||
extern Oid RenameEventTrigger(const char* trigname, const char *newname);
|
|
||||||
extern Oid AlterEventTriggerOwner(const char *name, Oid newOwnerId);
|
extern Oid AlterEventTriggerOwner(const char *name, Oid newOwnerId);
|
||||||
extern void AlterEventTriggerOwner_oid(Oid, Oid newOwnerId);
|
extern void AlterEventTriggerOwner_oid(Oid, Oid newOwnerId);
|
||||||
|
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
|
|
||||||
extern Oid CreateProceduralLanguage(CreatePLangStmt *stmt);
|
extern Oid CreateProceduralLanguage(CreatePLangStmt *stmt);
|
||||||
extern void DropProceduralLanguageById(Oid langOid);
|
extern void DropProceduralLanguageById(Oid langOid);
|
||||||
extern Oid RenameLanguage(const char *oldname, const char *newname);
|
|
||||||
extern bool PLTemplateExists(const char *languageName);
|
extern bool PLTemplateExists(const char *languageName);
|
||||||
extern Oid get_language_oid(const char *langname, bool missing_ok);
|
extern Oid get_language_oid(const char *langname, bool missing_ok);
|
||||||
|
|
||||||
|
@ -279,8 +279,8 @@ ALTER OPERATOR FAMILY alt_opf2 USING hash OWNER TO regtest_alter_user3; -- fail
|
|||||||
ERROR: must be member of role "regtest_alter_user3"
|
ERROR: must be member of role "regtest_alter_user3"
|
||||||
ALTER OPERATOR FAMILY alt_opf3 USING hash SET SCHEMA alt_nsp2; -- failed (not owner)
|
ALTER OPERATOR FAMILY alt_opf3 USING hash SET SCHEMA alt_nsp2; -- failed (not owner)
|
||||||
ERROR: must be owner of operator family alt_opf3
|
ERROR: must be owner of operator family alt_opf3
|
||||||
-- can't test this: the error message includes the raw oid of namespace
|
ALTER OPERATOR FAMILY alt_opf2 USING hash SET SCHEMA alt_nsp2; -- failed (name conflict)
|
||||||
-- ALTER OPERATOR FAMILY alt_opf2 USING hash SET SCHEMA alt_nsp2; -- failed (name conflict)
|
ERROR: operator family "alt_opf2" for access method "hash" already exists in schema "alt_nsp2"
|
||||||
ALTER OPERATOR CLASS alt_opc3 USING hash RENAME TO alt_opc4; -- failed (not owner)
|
ALTER OPERATOR CLASS alt_opc3 USING hash RENAME TO alt_opc4; -- failed (not owner)
|
||||||
ERROR: must be owner of operator class alt_opc3
|
ERROR: must be owner of operator class alt_opc3
|
||||||
ALTER OPERATOR CLASS alt_opc1 USING hash RENAME TO alt_opc4; -- OK
|
ALTER OPERATOR CLASS alt_opc1 USING hash RENAME TO alt_opc4; -- OK
|
||||||
@ -290,8 +290,8 @@ ALTER OPERATOR CLASS alt_opc2 USING hash OWNER TO regtest_alter_user3; -- faile
|
|||||||
ERROR: must be member of role "regtest_alter_user3"
|
ERROR: must be member of role "regtest_alter_user3"
|
||||||
ALTER OPERATOR CLASS alt_opc3 USING hash SET SCHEMA alt_nsp2; -- failed (not owner)
|
ALTER OPERATOR CLASS alt_opc3 USING hash SET SCHEMA alt_nsp2; -- failed (not owner)
|
||||||
ERROR: must be owner of operator class alt_opc3
|
ERROR: must be owner of operator class alt_opc3
|
||||||
-- can't test this: the error message includes the raw oid of namespace
|
ALTER OPERATOR CLASS alt_opc2 USING hash SET SCHEMA alt_nsp2; -- failed (name conflict)
|
||||||
-- ALTER OPERATOR CLASS alt_opc2 USING hash SET SCHEMA alt_nsp2; -- failed (name conflict)
|
ERROR: operator class "alt_opc2" for access method "hash" already exists in schema "alt_nsp2"
|
||||||
RESET SESSION AUTHORIZATION;
|
RESET SESSION AUTHORIZATION;
|
||||||
SELECT nspname, opfname, amname, rolname
|
SELECT nspname, opfname, amname, rolname
|
||||||
FROM pg_opfamily o, pg_am m, pg_namespace n, pg_authid a
|
FROM pg_opfamily o, pg_am m, pg_namespace n, pg_authid a
|
||||||
@ -327,7 +327,7 @@ SET SESSION AUTHORIZATION regtest_alter_user1;
|
|||||||
CREATE TEXT SEARCH DICTIONARY alt_ts_dict1 (template=simple);
|
CREATE TEXT SEARCH DICTIONARY alt_ts_dict1 (template=simple);
|
||||||
CREATE TEXT SEARCH DICTIONARY alt_ts_dict2 (template=simple);
|
CREATE TEXT SEARCH DICTIONARY alt_ts_dict2 (template=simple);
|
||||||
ALTER TEXT SEARCH DICTIONARY alt_ts_dict1 RENAME TO alt_ts_dict2; -- failed (name conflict)
|
ALTER TEXT SEARCH DICTIONARY alt_ts_dict1 RENAME TO alt_ts_dict2; -- failed (name conflict)
|
||||||
ERROR: text search dictionary "alt_ts_dict2" already exists
|
ERROR: text search dictionary "alt_ts_dict2" already exists in schema "alt_nsp1"
|
||||||
ALTER TEXT SEARCH DICTIONARY alt_ts_dict1 RENAME TO alt_ts_dict3; -- OK
|
ALTER TEXT SEARCH DICTIONARY alt_ts_dict1 RENAME TO alt_ts_dict3; -- OK
|
||||||
ALTER TEXT SEARCH DICTIONARY alt_ts_dict2 OWNER TO regtest_alter_user2; -- failed (no role membership)
|
ALTER TEXT SEARCH DICTIONARY alt_ts_dict2 OWNER TO regtest_alter_user2; -- failed (no role membership)
|
||||||
ERROR: must be member of role "regtest_alter_user2"
|
ERROR: must be member of role "regtest_alter_user2"
|
||||||
@ -368,7 +368,7 @@ SET SESSION AUTHORIZATION regtest_alter_user1;
|
|||||||
CREATE TEXT SEARCH CONFIGURATION alt_ts_conf1 (copy=english);
|
CREATE TEXT SEARCH CONFIGURATION alt_ts_conf1 (copy=english);
|
||||||
CREATE TEXT SEARCH CONFIGURATION alt_ts_conf2 (copy=english);
|
CREATE TEXT SEARCH CONFIGURATION alt_ts_conf2 (copy=english);
|
||||||
ALTER TEXT SEARCH CONFIGURATION alt_ts_conf1 RENAME TO alt_ts_conf2; -- failed (name conflict)
|
ALTER TEXT SEARCH CONFIGURATION alt_ts_conf1 RENAME TO alt_ts_conf2; -- failed (name conflict)
|
||||||
ERROR: text search configuration "alt_ts_conf2" already exists
|
ERROR: text search configuration "alt_ts_conf2" already exists in schema "alt_nsp1"
|
||||||
ALTER TEXT SEARCH CONFIGURATION alt_ts_conf1 RENAME TO alt_ts_conf3; -- OK
|
ALTER TEXT SEARCH CONFIGURATION alt_ts_conf1 RENAME TO alt_ts_conf3; -- OK
|
||||||
ALTER TEXT SEARCH CONFIGURATION alt_ts_conf2 OWNER TO regtest_alter_user2; -- failed (no role membership)
|
ALTER TEXT SEARCH CONFIGURATION alt_ts_conf2 OWNER TO regtest_alter_user2; -- failed (no role membership)
|
||||||
ERROR: must be member of role "regtest_alter_user2"
|
ERROR: must be member of role "regtest_alter_user2"
|
||||||
@ -408,7 +408,7 @@ SELECT nspname, cfgname, rolname
|
|||||||
CREATE TEXT SEARCH TEMPLATE alt_ts_temp1 (lexize=dsimple_lexize);
|
CREATE TEXT SEARCH TEMPLATE alt_ts_temp1 (lexize=dsimple_lexize);
|
||||||
CREATE TEXT SEARCH TEMPLATE alt_ts_temp2 (lexize=dsimple_lexize);
|
CREATE TEXT SEARCH TEMPLATE alt_ts_temp2 (lexize=dsimple_lexize);
|
||||||
ALTER TEXT SEARCH TEMPLATE alt_ts_temp1 RENAME TO alt_ts_temp2; -- failed (name conflict)
|
ALTER TEXT SEARCH TEMPLATE alt_ts_temp1 RENAME TO alt_ts_temp2; -- failed (name conflict)
|
||||||
ERROR: text search template "alt_ts_temp2" already exists
|
ERROR: text search template "alt_ts_temp2" already exists in schema "alt_nsp1"
|
||||||
ALTER TEXT SEARCH TEMPLATE alt_ts_temp1 RENAME TO alt_ts_temp3; -- OK
|
ALTER TEXT SEARCH TEMPLATE alt_ts_temp1 RENAME TO alt_ts_temp3; -- OK
|
||||||
ALTER TEXT SEARCH TEMPLATE alt_ts_temp2 SET SCHEMA alt_nsp2; -- OK
|
ALTER TEXT SEARCH TEMPLATE alt_ts_temp2 SET SCHEMA alt_nsp2; -- OK
|
||||||
CREATE TEXT SEARCH TEMPLATE alt_ts_temp2 (lexize=dsimple_lexize);
|
CREATE TEXT SEARCH TEMPLATE alt_ts_temp2 (lexize=dsimple_lexize);
|
||||||
@ -433,7 +433,7 @@ CREATE TEXT SEARCH PARSER alt_ts_prs1
|
|||||||
CREATE TEXT SEARCH PARSER alt_ts_prs2
|
CREATE TEXT SEARCH PARSER alt_ts_prs2
|
||||||
(start = prsd_start, gettoken = prsd_nexttoken, end = prsd_end, lextypes = prsd_lextype);
|
(start = prsd_start, gettoken = prsd_nexttoken, end = prsd_end, lextypes = prsd_lextype);
|
||||||
ALTER TEXT SEARCH PARSER alt_ts_prs1 RENAME TO alt_ts_prs2; -- failed (name conflict)
|
ALTER TEXT SEARCH PARSER alt_ts_prs1 RENAME TO alt_ts_prs2; -- failed (name conflict)
|
||||||
ERROR: text search parser "alt_ts_prs2" already exists
|
ERROR: text search parser "alt_ts_prs2" already exists in schema "alt_nsp1"
|
||||||
ALTER TEXT SEARCH PARSER alt_ts_prs1 RENAME TO alt_ts_prs3; -- OK
|
ALTER TEXT SEARCH PARSER alt_ts_prs1 RENAME TO alt_ts_prs3; -- OK
|
||||||
ALTER TEXT SEARCH PARSER alt_ts_prs2 SET SCHEMA alt_nsp2; -- OK
|
ALTER TEXT SEARCH PARSER alt_ts_prs2 SET SCHEMA alt_nsp2; -- OK
|
||||||
CREATE TEXT SEARCH PARSER alt_ts_prs2
|
CREATE TEXT SEARCH PARSER alt_ts_prs2
|
||||||
|
@ -242,16 +242,14 @@ ALTER OPERATOR FAMILY alt_opf1 USING hash RENAME TO alt_opf4; -- OK
|
|||||||
ALTER OPERATOR FAMILY alt_opf3 USING hash OWNER TO regtest_alter_user2; -- failed (not owner)
|
ALTER OPERATOR FAMILY alt_opf3 USING hash OWNER TO regtest_alter_user2; -- failed (not owner)
|
||||||
ALTER OPERATOR FAMILY alt_opf2 USING hash OWNER TO regtest_alter_user3; -- failed (no role membership)
|
ALTER OPERATOR FAMILY alt_opf2 USING hash OWNER TO regtest_alter_user3; -- failed (no role membership)
|
||||||
ALTER OPERATOR FAMILY alt_opf3 USING hash SET SCHEMA alt_nsp2; -- failed (not owner)
|
ALTER OPERATOR FAMILY alt_opf3 USING hash SET SCHEMA alt_nsp2; -- failed (not owner)
|
||||||
-- can't test this: the error message includes the raw oid of namespace
|
ALTER OPERATOR FAMILY alt_opf2 USING hash SET SCHEMA alt_nsp2; -- failed (name conflict)
|
||||||
-- ALTER OPERATOR FAMILY alt_opf2 USING hash SET SCHEMA alt_nsp2; -- failed (name conflict)
|
|
||||||
|
|
||||||
ALTER OPERATOR CLASS alt_opc3 USING hash RENAME TO alt_opc4; -- failed (not owner)
|
ALTER OPERATOR CLASS alt_opc3 USING hash RENAME TO alt_opc4; -- failed (not owner)
|
||||||
ALTER OPERATOR CLASS alt_opc1 USING hash RENAME TO alt_opc4; -- OK
|
ALTER OPERATOR CLASS alt_opc1 USING hash RENAME TO alt_opc4; -- OK
|
||||||
ALTER OPERATOR CLASS alt_opc3 USING hash OWNER TO regtest_alter_user2; -- failed (not owner)
|
ALTER OPERATOR CLASS alt_opc3 USING hash OWNER TO regtest_alter_user2; -- failed (not owner)
|
||||||
ALTER OPERATOR CLASS alt_opc2 USING hash OWNER TO regtest_alter_user3; -- failed (no role membership)
|
ALTER OPERATOR CLASS alt_opc2 USING hash OWNER TO regtest_alter_user3; -- failed (no role membership)
|
||||||
ALTER OPERATOR CLASS alt_opc3 USING hash SET SCHEMA alt_nsp2; -- failed (not owner)
|
ALTER OPERATOR CLASS alt_opc3 USING hash SET SCHEMA alt_nsp2; -- failed (not owner)
|
||||||
-- can't test this: the error message includes the raw oid of namespace
|
ALTER OPERATOR CLASS alt_opc2 USING hash SET SCHEMA alt_nsp2; -- failed (name conflict)
|
||||||
-- ALTER OPERATOR CLASS alt_opc2 USING hash SET SCHEMA alt_nsp2; -- failed (name conflict)
|
|
||||||
|
|
||||||
RESET SESSION AUTHORIZATION;
|
RESET SESSION AUTHORIZATION;
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user