mirror of
https://git.postgresql.org/git/postgresql.git
synced 2025-01-18 18:44:06 +08:00
Make view/rule permission checking behave properly with
subqueries in the rule.
This commit is contained in:
parent
7c578904d7
commit
b59c03f842
@ -7,7 +7,7 @@
|
||||
*
|
||||
*
|
||||
* IDENTIFICATION
|
||||
* $Header: /cvsroot/pgsql/src/backend/rewrite/Attic/locks.c,v 1.29 2000/05/30 00:49:51 momjian Exp $
|
||||
* $Header: /cvsroot/pgsql/src/backend/rewrite/Attic/locks.c,v 1.30 2000/07/09 04:56:32 tgl Exp $
|
||||
*
|
||||
*-------------------------------------------------------------------------
|
||||
*/
|
||||
@ -17,9 +17,9 @@
|
||||
#include "catalog/pg_shadow.h"
|
||||
#include "optimizer/clauses.h"
|
||||
#include "rewrite/locks.h"
|
||||
#include "parser/parsetree.h"
|
||||
#include "utils/acl.h"
|
||||
#include "utils/syscache.h"
|
||||
#include "utils/syscache.h"
|
||||
|
||||
|
||||
/*
|
||||
@ -152,61 +152,76 @@ matchLocks(CmdType event,
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
checkLockPerms(List *locks, Query *parsetree, int rt_index)
|
||||
/*
|
||||
* Check the access permissions of tables that are referred to by a rule.
|
||||
* We want to check the access permissions using the userid of the rule's
|
||||
* owner, *not* of the current user (the one accessing the rule). So, we
|
||||
* do the permission check here and set skipAcl = TRUE in each of the rule's
|
||||
* RTEs, to prevent the executor from running another check with the current
|
||||
* user's ID.
|
||||
*
|
||||
* XXX This routine is called before the rule's query tree has been copied
|
||||
* out of the relcache entry where it is kept. Therefore, when we set
|
||||
* skipAcl = TRUE, we are destructively modifying the relcache entry for
|
||||
* the event relation! This seems fairly harmless because the relcache
|
||||
* querytree is only used as a source for the rewriter, but it's a tad
|
||||
* unclean anyway.
|
||||
*
|
||||
* Note that we must check permissions every time, even if skipAcl was
|
||||
* already set TRUE by a prior call. This ensures that we enforce the
|
||||
* current permission settings for each referenced table, even if they
|
||||
* have changed since the relcache entry was loaded.
|
||||
*/
|
||||
|
||||
typedef struct
|
||||
{
|
||||
Relation ev_rel;
|
||||
HeapTuple usertup;
|
||||
char *evowner;
|
||||
RangeTblEntry *rte;
|
||||
} checkLockPerms_context;
|
||||
|
||||
static bool
|
||||
checkLockPerms_walker(Node *node,
|
||||
checkLockPerms_context *context)
|
||||
{
|
||||
if (node == NULL)
|
||||
return false;
|
||||
if (IsA(node, SubLink))
|
||||
{
|
||||
/*
|
||||
* Standard expression_tree_walker will not recurse into
|
||||
* subselect, but here we must do so.
|
||||
*/
|
||||
SubLink *sub = (SubLink *) node;
|
||||
|
||||
if (checkLockPerms_walker((Node *) (sub->lefthand), context))
|
||||
return true;
|
||||
if (checkLockPerms_walker((Node *) (sub->subselect), context))
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
if (IsA(node, Query))
|
||||
{
|
||||
/* Reach here after recursing down into subselect above... */
|
||||
Query *qry = (Query *) node;
|
||||
int rtablength = length(qry->rtable);
|
||||
int i;
|
||||
|
||||
/* Check all the RTEs in this query node, except OLD and NEW */
|
||||
for (i = 1; i <= rtablength; i++)
|
||||
{
|
||||
RangeTblEntry *rte = rt_fetch(i, qry->rtable);
|
||||
int32 reqperm;
|
||||
int32 aclcheck_res;
|
||||
int i;
|
||||
List *l;
|
||||
|
||||
if (locks == NIL)
|
||||
return;
|
||||
|
||||
/*
|
||||
* Get the usename of the rules event relation owner
|
||||
*/
|
||||
rte = (RangeTblEntry *) nth(rt_index - 1, parsetree->rtable);
|
||||
ev_rel = heap_openr(rte->relname, AccessShareLock);
|
||||
usertup = SearchSysCacheTuple(SHADOWSYSID,
|
||||
ObjectIdGetDatum(ev_rel->rd_rel->relowner),
|
||||
0, 0, 0);
|
||||
if (!HeapTupleIsValid(usertup))
|
||||
if (rte->ref != NULL)
|
||||
{
|
||||
elog(ERROR, "cache lookup for userid %d failed",
|
||||
ev_rel->rd_rel->relowner);
|
||||
if (strcmp(rte->ref->relname, "*NEW*") == 0)
|
||||
continue;
|
||||
if (strcmp(rte->ref->relname, "*OLD*") == 0)
|
||||
continue;
|
||||
}
|
||||
heap_close(ev_rel, AccessShareLock);
|
||||
evowner = pstrdup(NameStr(((Form_pg_shadow) GETSTRUCT(usertup))->usename));
|
||||
|
||||
/*
|
||||
* Check all the locks, that should get fired on this query
|
||||
*/
|
||||
foreach(l, locks)
|
||||
{
|
||||
RewriteRule *onelock = (RewriteRule *) lfirst(l);
|
||||
List *action;
|
||||
|
||||
/*
|
||||
* In each lock check every action
|
||||
*/
|
||||
foreach(action, onelock->actions)
|
||||
{
|
||||
Query *query = (Query *) lfirst(action);
|
||||
|
||||
/*
|
||||
* In each action check every rangetable entry for read/write
|
||||
* permission of the event relations owner depending on if
|
||||
* it's the result relation (write) or not (read)
|
||||
*/
|
||||
for (i = 2; i < length(query->rtable); i++)
|
||||
{
|
||||
if (i + 1 == query->resultRelation)
|
||||
switch (query->resultRelation)
|
||||
if (i == qry->resultRelation)
|
||||
switch (qry->commandType)
|
||||
{
|
||||
case CMD_INSERT:
|
||||
reqperm = ACL_AP;
|
||||
@ -218,27 +233,82 @@ checkLockPerms(List *locks, Query *parsetree, int rt_index)
|
||||
else
|
||||
reqperm = ACL_RD;
|
||||
|
||||
rte = (RangeTblEntry *) nth(i, query->rtable);
|
||||
aclcheck_res = pg_aclcheck(rte->relname,
|
||||
evowner, reqperm);
|
||||
context->evowner,
|
||||
reqperm);
|
||||
if (aclcheck_res != ACLCHECK_OK)
|
||||
{
|
||||
elog(ERROR, "%s: %s",
|
||||
rte->relname,
|
||||
aclcheck_error_strings[aclcheck_res]);
|
||||
}
|
||||
|
||||
/*
|
||||
* So this is allowed due to the permissions of the rules
|
||||
* event relation owner. But let's see if the next one too
|
||||
* Mark RTE to prevent executor from checking again with the
|
||||
* current user's ID...
|
||||
*/
|
||||
rte->skipAcl = TRUE;
|
||||
}
|
||||
}
|
||||
rte->skipAcl = true;
|
||||
}
|
||||
|
||||
/*
|
||||
* Phew, that was close
|
||||
*/
|
||||
return;
|
||||
/* If there are sublinks, search for them and check their RTEs */
|
||||
if (qry->hasSubLinks)
|
||||
{
|
||||
if (checkLockPerms_walker((Node *) (qry->targetList), context))
|
||||
return true;
|
||||
if (checkLockPerms_walker((Node *) (qry->qual), context))
|
||||
return true;
|
||||
if (checkLockPerms_walker((Node *) (qry->havingQual), context))
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
return expression_tree_walker(node, checkLockPerms_walker,
|
||||
(void *) context);
|
||||
}
|
||||
|
||||
void
|
||||
checkLockPerms(List *locks, Query *parsetree, int rt_index)
|
||||
{
|
||||
RangeTblEntry *rte;
|
||||
Relation ev_rel;
|
||||
HeapTuple usertup;
|
||||
Form_pg_shadow userform;
|
||||
checkLockPerms_context context;
|
||||
List *l;
|
||||
|
||||
if (locks == NIL)
|
||||
return; /* nothing to check */
|
||||
|
||||
/*
|
||||
* Get the usename of the rule's event relation owner
|
||||
*/
|
||||
rte = rt_fetch(rt_index, parsetree->rtable);
|
||||
ev_rel = heap_openr(rte->relname, AccessShareLock);
|
||||
usertup = SearchSysCacheTuple(SHADOWSYSID,
|
||||
ObjectIdGetDatum(ev_rel->rd_rel->relowner),
|
||||
0, 0, 0);
|
||||
if (!HeapTupleIsValid(usertup))
|
||||
elog(ERROR, "cache lookup for userid %d failed",
|
||||
ev_rel->rd_rel->relowner);
|
||||
userform = (Form_pg_shadow) GETSTRUCT(usertup);
|
||||
context.evowner = pstrdup(NameStr(userform->usename));
|
||||
heap_close(ev_rel, AccessShareLock);
|
||||
|
||||
/*
|
||||
* Check all the locks that should get fired on this query
|
||||
*/
|
||||
foreach(l, locks)
|
||||
{
|
||||
RewriteRule *onelock = (RewriteRule *) lfirst(l);
|
||||
List *action;
|
||||
|
||||
/*
|
||||
* In each lock check every action. We must scan the action
|
||||
* recursively in case there are any sub-queries within it.
|
||||
*/
|
||||
foreach(action, onelock->actions)
|
||||
{
|
||||
Query *query = (Query *) lfirst(action);
|
||||
|
||||
checkLockPerms_walker((Node *) query, &context);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user