mirror of
https://github.com/postgres/postgres.git
synced 2025-05-05 09:19:17 +03:00
2035 lines
53 KiB
C
2035 lines
53 KiB
C
/*-------------------------------------------------------------------------
|
|
*
|
|
* parse_relation.c
|
|
* parser support routines dealing with relations
|
|
*
|
|
* Portions Copyright (c) 1996-2004, PostgreSQL Global Development Group
|
|
* Portions Copyright (c) 1994, Regents of the University of California
|
|
*
|
|
*
|
|
* IDENTIFICATION
|
|
* $PostgreSQL: pgsql/src/backend/parser/parse_relation.c,v 1.99 2004/08/29 04:12:41 momjian Exp $
|
|
*
|
|
*-------------------------------------------------------------------------
|
|
*/
|
|
#include "postgres.h"
|
|
|
|
#include <ctype.h>
|
|
|
|
#include "access/heapam.h"
|
|
#include "access/htup.h"
|
|
#include "catalog/heap.h"
|
|
#include "catalog/namespace.h"
|
|
#include "catalog/pg_type.h"
|
|
#include "nodes/makefuncs.h"
|
|
#include "parser/parsetree.h"
|
|
#include "parser/parse_coerce.h"
|
|
#include "parser/parse_expr.h"
|
|
#include "parser/parse_relation.h"
|
|
#include "parser/parse_type.h"
|
|
#include "rewrite/rewriteManip.h"
|
|
#include "utils/builtins.h"
|
|
#include "utils/lsyscache.h"
|
|
#include "utils/syscache.h"
|
|
|
|
/* GUC parameter */
|
|
bool add_missing_from;
|
|
|
|
static Node *scanNameSpaceForRefname(ParseState *pstate, Node *nsnode,
|
|
const char *refname);
|
|
static Node *scanNameSpaceForRelid(ParseState *pstate, Node *nsnode,
|
|
Oid relid);
|
|
static void scanNameSpaceForConflict(ParseState *pstate, Node *nsnode,
|
|
RangeTblEntry *rte1, const char *aliasname1);
|
|
static bool isForUpdate(ParseState *pstate, char *refname);
|
|
static void expandRelation(Oid relid, Alias *eref,
|
|
int rtindex, int sublevels_up,
|
|
bool include_dropped,
|
|
List **colnames, List **colvars);
|
|
static int specialAttNum(const char *attname);
|
|
static void warnAutoRange(ParseState *pstate, RangeVar *relation);
|
|
|
|
|
|
/*
|
|
* refnameRangeTblEntry
|
|
* Given a possibly-qualified refname, look to see if it matches any RTE.
|
|
* If so, return a pointer to the RangeTblEntry; else return NULL.
|
|
*
|
|
* Optionally get RTE's nesting depth (0 = current) into *sublevels_up.
|
|
* If sublevels_up is NULL, only consider items at the current nesting
|
|
* level.
|
|
*
|
|
* An unqualified refname (schemaname == NULL) can match any RTE with matching
|
|
* alias, or matching unqualified relname in the case of alias-less relation
|
|
* RTEs. It is possible that such a refname matches multiple RTEs in the
|
|
* nearest nesting level that has a match; if so, we report an error via
|
|
* ereport().
|
|
*
|
|
* A qualified refname (schemaname != NULL) can only match a relation RTE
|
|
* that (a) has no alias and (b) is for the same relation identified by
|
|
* schemaname.refname. In this case we convert schemaname.refname to a
|
|
* relation OID and search by relid, rather than by alias name. This is
|
|
* peculiar, but it's what SQL92 says to do.
|
|
*/
|
|
RangeTblEntry *
|
|
refnameRangeTblEntry(ParseState *pstate,
|
|
const char *schemaname,
|
|
const char *refname,
|
|
int *sublevels_up)
|
|
{
|
|
Oid relId = InvalidOid;
|
|
|
|
if (sublevels_up)
|
|
*sublevels_up = 0;
|
|
|
|
if (schemaname != NULL)
|
|
{
|
|
Oid namespaceId;
|
|
|
|
namespaceId = LookupExplicitNamespace(schemaname);
|
|
relId = get_relname_relid(refname, namespaceId);
|
|
if (!OidIsValid(relId))
|
|
return NULL;
|
|
}
|
|
|
|
while (pstate != NULL)
|
|
{
|
|
Node *nsnode;
|
|
|
|
if (OidIsValid(relId))
|
|
nsnode = scanNameSpaceForRelid(pstate,
|
|
(Node *) pstate->p_namespace,
|
|
relId);
|
|
else
|
|
nsnode = scanNameSpaceForRefname(pstate,
|
|
(Node *) pstate->p_namespace,
|
|
refname);
|
|
|
|
if (nsnode)
|
|
{
|
|
/* should get an RTE or JoinExpr */
|
|
if (IsA(nsnode, RangeTblEntry))
|
|
return (RangeTblEntry *) nsnode;
|
|
Assert(IsA(nsnode, JoinExpr));
|
|
return rt_fetch(((JoinExpr *) nsnode)->rtindex, pstate->p_rtable);
|
|
}
|
|
|
|
pstate = pstate->parentParseState;
|
|
if (sublevels_up)
|
|
(*sublevels_up)++;
|
|
else
|
|
break;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Recursively search a namespace for an RTE or joinexpr matching the
|
|
* given unqualified refname. Return the node if a unique match, or NULL
|
|
* if no match. Raise error if multiple matches.
|
|
*
|
|
* The top level of p_namespace is a list, and we recurse into any joins
|
|
* that are not subqueries.
|
|
*/
|
|
static Node *
|
|
scanNameSpaceForRefname(ParseState *pstate, Node *nsnode,
|
|
const char *refname)
|
|
{
|
|
Node *result = NULL;
|
|
Node *newresult;
|
|
|
|
if (nsnode == NULL)
|
|
return NULL;
|
|
if (IsA(nsnode, RangeTblRef))
|
|
{
|
|
int varno = ((RangeTblRef *) nsnode)->rtindex;
|
|
RangeTblEntry *rte = rt_fetch(varno, pstate->p_rtable);
|
|
|
|
if (strcmp(rte->eref->aliasname, refname) == 0)
|
|
result = (Node *) rte;
|
|
}
|
|
else if (IsA(nsnode, JoinExpr))
|
|
{
|
|
JoinExpr *j = (JoinExpr *) nsnode;
|
|
|
|
if (j->alias)
|
|
{
|
|
if (strcmp(j->alias->aliasname, refname) == 0)
|
|
return (Node *) j; /* matched a join alias */
|
|
|
|
/*
|
|
* Tables within an aliased join are invisible from outside
|
|
* the join, according to the scope rules of SQL92 (the join
|
|
* is considered a subquery). So, stop here.
|
|
*/
|
|
return NULL;
|
|
}
|
|
result = scanNameSpaceForRefname(pstate, j->larg, refname);
|
|
newresult = scanNameSpaceForRefname(pstate, j->rarg, refname);
|
|
if (!result)
|
|
result = newresult;
|
|
else if (newresult)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_AMBIGUOUS_ALIAS),
|
|
errmsg("table reference \"%s\" is ambiguous",
|
|
refname)));
|
|
}
|
|
else if (IsA(nsnode, List))
|
|
{
|
|
ListCell *l;
|
|
|
|
foreach(l, (List *) nsnode)
|
|
{
|
|
newresult = scanNameSpaceForRefname(pstate, lfirst(l), refname);
|
|
if (!result)
|
|
result = newresult;
|
|
else if (newresult)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_AMBIGUOUS_ALIAS),
|
|
errmsg("table reference \"%s\" is ambiguous",
|
|
refname)));
|
|
}
|
|
}
|
|
else
|
|
elog(ERROR, "unrecognized node type: %d", (int) nodeTag(nsnode));
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Recursively search a namespace for a relation RTE matching the
|
|
* given relation OID. Return the node if a unique match, or NULL
|
|
* if no match. Raise error if multiple matches (which shouldn't
|
|
* happen if the namespace was checked correctly when it was created).
|
|
*
|
|
* The top level of p_namespace is a list, and we recurse into any joins
|
|
* that are not subqueries.
|
|
*
|
|
* See the comments for refnameRangeTblEntry to understand why this
|
|
* acts the way it does.
|
|
*/
|
|
static Node *
|
|
scanNameSpaceForRelid(ParseState *pstate, Node *nsnode, Oid relid)
|
|
{
|
|
Node *result = NULL;
|
|
Node *newresult;
|
|
|
|
if (nsnode == NULL)
|
|
return NULL;
|
|
if (IsA(nsnode, RangeTblRef))
|
|
{
|
|
int varno = ((RangeTblRef *) nsnode)->rtindex;
|
|
RangeTblEntry *rte = rt_fetch(varno, pstate->p_rtable);
|
|
|
|
/* yes, the test for alias==NULL should be there... */
|
|
if (rte->rtekind == RTE_RELATION &&
|
|
rte->relid == relid &&
|
|
rte->alias == NULL)
|
|
result = (Node *) rte;
|
|
}
|
|
else if (IsA(nsnode, JoinExpr))
|
|
{
|
|
JoinExpr *j = (JoinExpr *) nsnode;
|
|
|
|
if (j->alias)
|
|
{
|
|
/*
|
|
* Tables within an aliased join are invisible from outside
|
|
* the join, according to the scope rules of SQL92 (the join
|
|
* is considered a subquery). So, stop here.
|
|
*/
|
|
return NULL;
|
|
}
|
|
result = scanNameSpaceForRelid(pstate, j->larg, relid);
|
|
newresult = scanNameSpaceForRelid(pstate, j->rarg, relid);
|
|
if (!result)
|
|
result = newresult;
|
|
else if (newresult)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_AMBIGUOUS_ALIAS),
|
|
errmsg("table reference %u is ambiguous",
|
|
relid)));
|
|
}
|
|
else if (IsA(nsnode, List))
|
|
{
|
|
ListCell *l;
|
|
|
|
foreach(l, (List *) nsnode)
|
|
{
|
|
newresult = scanNameSpaceForRelid(pstate, lfirst(l), relid);
|
|
if (!result)
|
|
result = newresult;
|
|
else if (newresult)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_AMBIGUOUS_ALIAS),
|
|
errmsg("table reference %u is ambiguous",
|
|
relid)));
|
|
}
|
|
}
|
|
else
|
|
elog(ERROR, "unrecognized node type: %d", (int) nodeTag(nsnode));
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Recursively check for name conflicts between two namespaces or
|
|
* namespace subtrees. Raise an error if any is found.
|
|
*
|
|
* Works by recursively scanning namespace1 for RTEs and join nodes,
|
|
* and for each one recursively scanning namespace2 for a match.
|
|
*
|
|
* Note: we assume that each given argument does not contain conflicts
|
|
* itself; we just want to know if the two can be merged together.
|
|
*
|
|
* Per SQL92, two alias-less plain relation RTEs do not conflict even if
|
|
* they have the same eref->aliasname (ie, same relation name), if they
|
|
* are for different relation OIDs (implying they are in different schemas).
|
|
*/
|
|
void
|
|
checkNameSpaceConflicts(ParseState *pstate, Node *namespace1,
|
|
Node *namespace2)
|
|
{
|
|
if (namespace1 == NULL)
|
|
return;
|
|
if (IsA(namespace1, RangeTblRef))
|
|
{
|
|
int varno = ((RangeTblRef *) namespace1)->rtindex;
|
|
RangeTblEntry *rte = rt_fetch(varno, pstate->p_rtable);
|
|
|
|
if (rte->rtekind == RTE_RELATION && rte->alias == NULL)
|
|
scanNameSpaceForConflict(pstate, namespace2,
|
|
rte, rte->eref->aliasname);
|
|
else
|
|
scanNameSpaceForConflict(pstate, namespace2,
|
|
NULL, rte->eref->aliasname);
|
|
}
|
|
else if (IsA(namespace1, JoinExpr))
|
|
{
|
|
JoinExpr *j = (JoinExpr *) namespace1;
|
|
|
|
if (j->alias)
|
|
{
|
|
scanNameSpaceForConflict(pstate, namespace2,
|
|
NULL, j->alias->aliasname);
|
|
|
|
/*
|
|
* Tables within an aliased join are invisible from outside
|
|
* the join, according to the scope rules of SQL92 (the join
|
|
* is considered a subquery). So, stop here.
|
|
*/
|
|
return;
|
|
}
|
|
checkNameSpaceConflicts(pstate, j->larg, namespace2);
|
|
checkNameSpaceConflicts(pstate, j->rarg, namespace2);
|
|
}
|
|
else if (IsA(namespace1, List))
|
|
{
|
|
ListCell *l;
|
|
|
|
foreach(l, (List *) namespace1)
|
|
checkNameSpaceConflicts(pstate, lfirst(l), namespace2);
|
|
}
|
|
else
|
|
elog(ERROR, "unrecognized node type: %d", (int) nodeTag(namespace1));
|
|
}
|
|
|
|
/*
|
|
* Subroutine for checkNameSpaceConflicts: scan namespace2
|
|
*/
|
|
static void
|
|
scanNameSpaceForConflict(ParseState *pstate, Node *nsnode,
|
|
RangeTblEntry *rte1, const char *aliasname1)
|
|
{
|
|
if (nsnode == NULL)
|
|
return;
|
|
if (IsA(nsnode, RangeTblRef))
|
|
{
|
|
int varno = ((RangeTblRef *) nsnode)->rtindex;
|
|
RangeTblEntry *rte = rt_fetch(varno, pstate->p_rtable);
|
|
|
|
if (strcmp(rte->eref->aliasname, aliasname1) != 0)
|
|
return; /* definitely no conflict */
|
|
if (rte->rtekind == RTE_RELATION && rte->alias == NULL &&
|
|
rte1 != NULL && rte->relid != rte1->relid)
|
|
return; /* no conflict per SQL92 rule */
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_DUPLICATE_ALIAS),
|
|
errmsg("table name \"%s\" specified more than once",
|
|
aliasname1)));
|
|
}
|
|
else if (IsA(nsnode, JoinExpr))
|
|
{
|
|
JoinExpr *j = (JoinExpr *) nsnode;
|
|
|
|
if (j->alias)
|
|
{
|
|
if (strcmp(j->alias->aliasname, aliasname1) == 0)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_DUPLICATE_ALIAS),
|
|
errmsg("table name \"%s\" specified more than once",
|
|
aliasname1)));
|
|
|
|
/*
|
|
* Tables within an aliased join are invisible from outside
|
|
* the join, according to the scope rules of SQL92 (the join
|
|
* is considered a subquery). So, stop here.
|
|
*/
|
|
return;
|
|
}
|
|
scanNameSpaceForConflict(pstate, j->larg, rte1, aliasname1);
|
|
scanNameSpaceForConflict(pstate, j->rarg, rte1, aliasname1);
|
|
}
|
|
else if (IsA(nsnode, List))
|
|
{
|
|
ListCell *l;
|
|
|
|
foreach(l, (List *) nsnode)
|
|
scanNameSpaceForConflict(pstate, lfirst(l), rte1, aliasname1);
|
|
}
|
|
else
|
|
elog(ERROR, "unrecognized node type: %d", (int) nodeTag(nsnode));
|
|
}
|
|
|
|
/*
|
|
* given an RTE, return RT index (starting with 1) of the entry,
|
|
* and optionally get its nesting depth (0 = current). If sublevels_up
|
|
* is NULL, only consider rels at the current nesting level.
|
|
* Raises error if RTE not found.
|
|
*/
|
|
int
|
|
RTERangeTablePosn(ParseState *pstate, RangeTblEntry *rte, int *sublevels_up)
|
|
{
|
|
int index;
|
|
ListCell *l;
|
|
|
|
if (sublevels_up)
|
|
*sublevels_up = 0;
|
|
|
|
while (pstate != NULL)
|
|
{
|
|
index = 1;
|
|
foreach(l, pstate->p_rtable)
|
|
{
|
|
if (rte == (RangeTblEntry *) lfirst(l))
|
|
return index;
|
|
index++;
|
|
}
|
|
pstate = pstate->parentParseState;
|
|
if (sublevels_up)
|
|
(*sublevels_up)++;
|
|
else
|
|
break;
|
|
}
|
|
|
|
elog(ERROR, "RTE not found (internal error)");
|
|
return 0; /* keep compiler quiet */
|
|
}
|
|
|
|
/*
|
|
* Given an RT index and nesting depth, find the corresponding RTE.
|
|
* This is the inverse of RTERangeTablePosn.
|
|
*/
|
|
RangeTblEntry *
|
|
GetRTEByRangeTablePosn(ParseState *pstate,
|
|
int varno,
|
|
int sublevels_up)
|
|
{
|
|
while (sublevels_up-- > 0)
|
|
{
|
|
pstate = pstate->parentParseState;
|
|
Assert(pstate != NULL);
|
|
}
|
|
Assert(varno > 0 && varno <= list_length(pstate->p_rtable));
|
|
return rt_fetch(varno, pstate->p_rtable);
|
|
}
|
|
|
|
/*
|
|
* GetLevelNRangeTable
|
|
* Get the rangetable list for the N'th query level up from current.
|
|
*/
|
|
List *
|
|
GetLevelNRangeTable(ParseState *pstate, int sublevels_up)
|
|
{
|
|
int index = 0;
|
|
|
|
while (pstate != NULL)
|
|
{
|
|
if (index == sublevels_up)
|
|
return pstate->p_rtable;
|
|
index++;
|
|
pstate = pstate->parentParseState;
|
|
}
|
|
|
|
elog(ERROR, "rangetable not found (internal error)");
|
|
return NIL; /* keep compiler quiet */
|
|
}
|
|
|
|
/*
|
|
* scanRTEForColumn
|
|
* Search the column names of a single RTE for the given name.
|
|
* If found, return an appropriate Var node, else return NULL.
|
|
* If the name proves ambiguous within this RTE, raise error.
|
|
*
|
|
* Side effect: if we find a match, mark the RTE as requiring read access.
|
|
* See comments in setTargetTable().
|
|
*
|
|
* NOTE: if the RTE is for a join, marking it as requiring read access does
|
|
* nothing. It might seem that we need to propagate the mark to all the
|
|
* contained RTEs, but that is not necessary. This is so because a join
|
|
* expression can only appear in a FROM clause, and any table named in
|
|
* FROM will be marked as requiring read access from the beginning.
|
|
*/
|
|
Node *
|
|
scanRTEForColumn(ParseState *pstate, RangeTblEntry *rte, char *colname)
|
|
{
|
|
Node *result = NULL;
|
|
int attnum = 0;
|
|
ListCell *c;
|
|
|
|
/*
|
|
* Scan the user column names (or aliases) for a match. Complain if
|
|
* multiple matches.
|
|
*
|
|
* Note: eref->colnames may include entries for dropped columns,
|
|
* but those will be empty strings that cannot match any legal SQL
|
|
* identifier, so we don't bother to test for that case here.
|
|
*
|
|
* Should this somehow go wrong and we try to access a dropped column,
|
|
* we'll still catch it by virtue of the checks in
|
|
* get_rte_attribute_type(), which is called by make_var(). That routine
|
|
* has to do a cache lookup anyway, so the check there is cheap.
|
|
*/
|
|
foreach(c, rte->eref->colnames)
|
|
{
|
|
attnum++;
|
|
if (strcmp(strVal(lfirst(c)), colname) == 0)
|
|
{
|
|
if (result)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_AMBIGUOUS_COLUMN),
|
|
errmsg("column reference \"%s\" is ambiguous",
|
|
colname)));
|
|
result = (Node *) make_var(pstate, rte, attnum);
|
|
/* Require read access */
|
|
rte->requiredPerms |= ACL_SELECT;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* If we have a unique match, return it. Note that this allows a user
|
|
* alias to override a system column name (such as OID) without error.
|
|
*/
|
|
if (result)
|
|
return result;
|
|
|
|
/*
|
|
* If the RTE represents a real table, consider system column names.
|
|
*/
|
|
if (rte->rtekind == RTE_RELATION)
|
|
{
|
|
/* quick check to see if name could be a system column */
|
|
attnum = specialAttNum(colname);
|
|
if (attnum != InvalidAttrNumber)
|
|
{
|
|
/* now check to see if column actually is defined */
|
|
if (SearchSysCacheExists(ATTNUM,
|
|
ObjectIdGetDatum(rte->relid),
|
|
Int16GetDatum(attnum),
|
|
0, 0))
|
|
{
|
|
result = (Node *) make_var(pstate, rte, attnum);
|
|
/* Require read access */
|
|
rte->requiredPerms |= ACL_SELECT;
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* colNameToVar
|
|
* Search for an unqualified column name.
|
|
* If found, return the appropriate Var node (or expression).
|
|
* If not found, return NULL. If the name proves ambiguous, raise error.
|
|
* If localonly is true, only names in the innermost query are considered.
|
|
*/
|
|
Node *
|
|
colNameToVar(ParseState *pstate, char *colname, bool localonly)
|
|
{
|
|
Node *result = NULL;
|
|
ParseState *orig_pstate = pstate;
|
|
int levels_up = 0;
|
|
|
|
while (pstate != NULL)
|
|
{
|
|
ListCell *ns;
|
|
|
|
/*
|
|
* We need to look only at top-level namespace items, and even for
|
|
* those, ignore RTEs that are marked as not inFromCl and not the
|
|
* query's target relation.
|
|
*/
|
|
foreach(ns, pstate->p_namespace)
|
|
{
|
|
Node *nsnode = (Node *) lfirst(ns);
|
|
Node *newresult = NULL;
|
|
|
|
if (IsA(nsnode, RangeTblRef))
|
|
{
|
|
int varno = ((RangeTblRef *) nsnode)->rtindex;
|
|
RangeTblEntry *rte = rt_fetch(varno, pstate->p_rtable);
|
|
|
|
if (!rte->inFromCl &&
|
|
rte != pstate->p_target_rangetblentry)
|
|
continue;
|
|
|
|
/* use orig_pstate here to get the right sublevels_up */
|
|
newresult = scanRTEForColumn(orig_pstate, rte, colname);
|
|
}
|
|
else if (IsA(nsnode, JoinExpr))
|
|
{
|
|
int varno = ((JoinExpr *) nsnode)->rtindex;
|
|
RangeTblEntry *rte = rt_fetch(varno, pstate->p_rtable);
|
|
|
|
/* joins are always inFromCl, so no need to check */
|
|
|
|
/* use orig_pstate here to get the right sublevels_up */
|
|
newresult = scanRTEForColumn(orig_pstate, rte, colname);
|
|
}
|
|
else
|
|
elog(ERROR, "unrecognized node type: %d",
|
|
(int) nodeTag(nsnode));
|
|
|
|
if (newresult)
|
|
{
|
|
if (result)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_AMBIGUOUS_COLUMN),
|
|
errmsg("column reference \"%s\" is ambiguous",
|
|
colname)));
|
|
result = newresult;
|
|
}
|
|
}
|
|
|
|
if (result != NULL || localonly)
|
|
break; /* found, or don't want to look at parent */
|
|
|
|
pstate = pstate->parentParseState;
|
|
levels_up++;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* qualifiedNameToVar
|
|
* Search for a qualified column name: either refname.colname or
|
|
* schemaname.relname.colname.
|
|
*
|
|
* If found, return the appropriate Var node.
|
|
* If not found, return NULL. If the name proves ambiguous, raise error.
|
|
*/
|
|
Node *
|
|
qualifiedNameToVar(ParseState *pstate,
|
|
char *schemaname,
|
|
char *refname,
|
|
char *colname,
|
|
bool implicitRTEOK)
|
|
{
|
|
RangeTblEntry *rte;
|
|
int sublevels_up;
|
|
|
|
rte = refnameRangeTblEntry(pstate, schemaname, refname, &sublevels_up);
|
|
|
|
if (rte == NULL)
|
|
{
|
|
if (!implicitRTEOK)
|
|
return NULL;
|
|
rte = addImplicitRTE(pstate, makeRangeVar(schemaname, refname));
|
|
}
|
|
|
|
return scanRTEForColumn(pstate, rte, colname);
|
|
}
|
|
|
|
/*
|
|
* buildRelationAliases
|
|
* Construct the eref column name list for a relation RTE.
|
|
* This code is also used for the case of a function RTE returning
|
|
* a named composite type.
|
|
*
|
|
* tupdesc: the physical column information
|
|
* alias: the user-supplied alias, or NULL if none
|
|
* eref: the eref Alias to store column names in
|
|
*
|
|
* eref->colnames is filled in. Also, alias->colnames is rebuilt to insert
|
|
* empty strings for any dropped columns, so that it will be one-to-one with
|
|
* physical column numbers.
|
|
*/
|
|
static void
|
|
buildRelationAliases(TupleDesc tupdesc, Alias *alias, Alias *eref)
|
|
{
|
|
int maxattrs = tupdesc->natts;
|
|
ListCell *aliaslc;
|
|
int numaliases;
|
|
int varattno;
|
|
int numdropped = 0;
|
|
|
|
Assert(eref->colnames == NIL);
|
|
|
|
if (alias)
|
|
{
|
|
aliaslc = list_head(alias->colnames);
|
|
numaliases = list_length(alias->colnames);
|
|
/* We'll rebuild the alias colname list */
|
|
alias->colnames = NIL;
|
|
}
|
|
else
|
|
{
|
|
aliaslc = NULL;
|
|
numaliases = 0;
|
|
}
|
|
|
|
for (varattno = 0; varattno < maxattrs; varattno++)
|
|
{
|
|
Form_pg_attribute attr = tupdesc->attrs[varattno];
|
|
Value *attrname;
|
|
|
|
if (attr->attisdropped)
|
|
{
|
|
/* Always insert an empty string for a dropped column */
|
|
attrname = makeString(pstrdup(""));
|
|
if (aliaslc)
|
|
alias->colnames = lappend(alias->colnames, attrname);
|
|
numdropped++;
|
|
}
|
|
else if (aliaslc)
|
|
{
|
|
/* Use the next user-supplied alias */
|
|
attrname = (Value *) lfirst(aliaslc);
|
|
aliaslc = lnext(aliaslc);
|
|
alias->colnames = lappend(alias->colnames, attrname);
|
|
}
|
|
else
|
|
{
|
|
attrname = makeString(pstrdup(NameStr(attr->attname)));
|
|
/* we're done with the alias if any */
|
|
}
|
|
|
|
eref->colnames = lappend(eref->colnames, attrname);
|
|
}
|
|
|
|
/* Too many user-supplied aliases? */
|
|
if (aliaslc)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_INVALID_COLUMN_REFERENCE),
|
|
errmsg("table \"%s\" has %d columns available but %d columns specified",
|
|
eref->aliasname, maxattrs - numdropped, numaliases)));
|
|
}
|
|
|
|
/*
|
|
* Add an entry for a relation to the pstate's range table (p_rtable).
|
|
*
|
|
* If pstate is NULL, we just build an RTE and return it without adding it
|
|
* to an rtable list.
|
|
*
|
|
* Note: formerly this checked for refname conflicts, but that's wrong.
|
|
* Caller is responsible for checking for conflicts in the appropriate scope.
|
|
*/
|
|
RangeTblEntry *
|
|
addRangeTableEntry(ParseState *pstate,
|
|
RangeVar *relation,
|
|
Alias *alias,
|
|
bool inh,
|
|
bool inFromCl)
|
|
{
|
|
RangeTblEntry *rte = makeNode(RangeTblEntry);
|
|
char *refname = alias ? alias->aliasname : relation->relname;
|
|
LOCKMODE lockmode;
|
|
Relation rel;
|
|
|
|
rte->rtekind = RTE_RELATION;
|
|
rte->alias = alias;
|
|
|
|
/*
|
|
* Get the rel's OID. This access also ensures that we have an
|
|
* up-to-date relcache entry for the rel. Since this is typically the
|
|
* first access to a rel in a statement, be careful to get the right
|
|
* access level depending on whether we're doing SELECT FOR UPDATE.
|
|
*/
|
|
lockmode = isForUpdate(pstate, refname) ? RowShareLock : AccessShareLock;
|
|
rel = heap_openrv(relation, lockmode);
|
|
rte->relid = RelationGetRelid(rel);
|
|
|
|
/*
|
|
* Build the list of effective column names using user-supplied aliases
|
|
* and/or actual column names.
|
|
*/
|
|
rte->eref = makeAlias(refname, NIL);
|
|
buildRelationAliases(rel->rd_att, alias, rte->eref);
|
|
|
|
/*
|
|
* Drop the rel refcount, but keep the access lock till end of
|
|
* transaction so that the table can't be deleted or have its schema
|
|
* modified underneath us.
|
|
*/
|
|
heap_close(rel, NoLock);
|
|
|
|
/*----------
|
|
* Flags:
|
|
* - this RTE should be expanded to include descendant tables,
|
|
* - this RTE is in the FROM clause,
|
|
* - this RTE should be checked for appropriate access rights.
|
|
*
|
|
* The initial default on access checks is always check-for-READ-access,
|
|
* which is the right thing for all except target tables.
|
|
*----------
|
|
*/
|
|
rte->inh = inh;
|
|
rte->inFromCl = inFromCl;
|
|
|
|
rte->requiredPerms = ACL_SELECT;
|
|
rte->checkAsUser = 0; /* not set-uid by default, either */
|
|
|
|
/*
|
|
* Add completed RTE to pstate's range table list, but not to join
|
|
* list nor namespace --- caller must do that if appropriate.
|
|
*/
|
|
if (pstate != NULL)
|
|
pstate->p_rtable = lappend(pstate->p_rtable, rte);
|
|
|
|
return rte;
|
|
}
|
|
|
|
/*
|
|
* Add an entry for a relation to the pstate's range table (p_rtable).
|
|
*
|
|
* This is just like addRangeTableEntry() except that it makes an RTE
|
|
* given a relation OID instead of a RangeVar reference.
|
|
*
|
|
* Note that an alias clause *must* be supplied.
|
|
*/
|
|
RangeTblEntry *
|
|
addRangeTableEntryForRelation(ParseState *pstate,
|
|
Oid relid,
|
|
Alias *alias,
|
|
bool inh,
|
|
bool inFromCl)
|
|
{
|
|
RangeTblEntry *rte = makeNode(RangeTblEntry);
|
|
char *refname = alias->aliasname;
|
|
LOCKMODE lockmode;
|
|
Relation rel;
|
|
|
|
rte->rtekind = RTE_RELATION;
|
|
rte->alias = alias;
|
|
|
|
/*
|
|
* Get the rel's relcache entry. This access ensures that we have an
|
|
* up-to-date relcache entry for the rel. Since this is typically the
|
|
* first access to a rel in a statement, be careful to get the right
|
|
* access level depending on whether we're doing SELECT FOR UPDATE.
|
|
*/
|
|
lockmode = isForUpdate(pstate, refname) ? RowShareLock : AccessShareLock;
|
|
rel = heap_open(relid, lockmode);
|
|
rte->relid = relid;
|
|
|
|
/*
|
|
* Build the list of effective column names using user-supplied aliases
|
|
* and/or actual column names.
|
|
*/
|
|
rte->eref = makeAlias(refname, NIL);
|
|
buildRelationAliases(rel->rd_att, alias, rte->eref);
|
|
|
|
/*
|
|
* Drop the rel refcount, but keep the access lock till end of
|
|
* transaction so that the table can't be deleted or have its schema
|
|
* modified underneath us.
|
|
*/
|
|
heap_close(rel, NoLock);
|
|
|
|
/*----------
|
|
* Flags:
|
|
* - this RTE should be expanded to include descendant tables,
|
|
* - this RTE is in the FROM clause,
|
|
* - this RTE should be checked for appropriate access rights.
|
|
*
|
|
* The initial default on access checks is always check-for-READ-access,
|
|
* which is the right thing for all except target tables.
|
|
*----------
|
|
*/
|
|
rte->inh = inh;
|
|
rte->inFromCl = inFromCl;
|
|
|
|
rte->requiredPerms = ACL_SELECT;
|
|
rte->checkAsUser = 0; /* not set-uid by default, either */
|
|
|
|
/*
|
|
* Add completed RTE to pstate's range table list, but not to join
|
|
* list nor namespace --- caller must do that if appropriate.
|
|
*/
|
|
if (pstate != NULL)
|
|
pstate->p_rtable = lappend(pstate->p_rtable, rte);
|
|
|
|
return rte;
|
|
}
|
|
|
|
/*
|
|
* Add an entry for a subquery to the pstate's range table (p_rtable).
|
|
*
|
|
* This is just like addRangeTableEntry() except that it makes a subquery RTE.
|
|
* Note that an alias clause *must* be supplied.
|
|
*/
|
|
RangeTblEntry *
|
|
addRangeTableEntryForSubquery(ParseState *pstate,
|
|
Query *subquery,
|
|
Alias *alias,
|
|
bool inFromCl)
|
|
{
|
|
RangeTblEntry *rte = makeNode(RangeTblEntry);
|
|
char *refname = alias->aliasname;
|
|
Alias *eref;
|
|
int numaliases;
|
|
int varattno;
|
|
ListCell *tlistitem;
|
|
|
|
rte->rtekind = RTE_SUBQUERY;
|
|
rte->relid = InvalidOid;
|
|
rte->subquery = subquery;
|
|
rte->alias = alias;
|
|
|
|
eref = copyObject(alias);
|
|
numaliases = list_length(eref->colnames);
|
|
|
|
/* fill in any unspecified alias columns */
|
|
varattno = 0;
|
|
foreach(tlistitem, subquery->targetList)
|
|
{
|
|
TargetEntry *te = (TargetEntry *) lfirst(tlistitem);
|
|
|
|
if (te->resdom->resjunk)
|
|
continue;
|
|
varattno++;
|
|
Assert(varattno == te->resdom->resno);
|
|
if (varattno > numaliases)
|
|
{
|
|
char *attrname;
|
|
|
|
attrname = pstrdup(te->resdom->resname);
|
|
eref->colnames = lappend(eref->colnames, makeString(attrname));
|
|
}
|
|
}
|
|
if (varattno < numaliases)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_INVALID_COLUMN_REFERENCE),
|
|
errmsg("table \"%s\" has %d columns available but %d columns specified",
|
|
refname, varattno, numaliases)));
|
|
|
|
rte->eref = eref;
|
|
|
|
/*----------
|
|
* Flags:
|
|
* - this RTE should be expanded to include descendant tables,
|
|
* - this RTE is in the FROM clause,
|
|
* - this RTE should be checked for appropriate access rights.
|
|
*
|
|
* Subqueries are never checked for access rights.
|
|
*----------
|
|
*/
|
|
rte->inh = false; /* never true for subqueries */
|
|
rte->inFromCl = inFromCl;
|
|
|
|
rte->requiredPerms = 0;
|
|
rte->checkAsUser = 0;
|
|
|
|
/*
|
|
* Add completed RTE to pstate's range table list, but not to join
|
|
* list nor namespace --- caller must do that if appropriate.
|
|
*/
|
|
if (pstate != NULL)
|
|
pstate->p_rtable = lappend(pstate->p_rtable, rte);
|
|
|
|
return rte;
|
|
}
|
|
|
|
/*
|
|
* Add an entry for a function to the pstate's range table (p_rtable).
|
|
*
|
|
* This is just like addRangeTableEntry() except that it makes a function RTE.
|
|
*/
|
|
RangeTblEntry *
|
|
addRangeTableEntryForFunction(ParseState *pstate,
|
|
char *funcname,
|
|
Node *funcexpr,
|
|
RangeFunction *rangefunc,
|
|
bool inFromCl)
|
|
{
|
|
RangeTblEntry *rte = makeNode(RangeTblEntry);
|
|
Oid funcrettype = exprType(funcexpr);
|
|
char functyptype;
|
|
Alias *alias = rangefunc->alias;
|
|
List *coldeflist = rangefunc->coldeflist;
|
|
Alias *eref;
|
|
|
|
rte->rtekind = RTE_FUNCTION;
|
|
rte->relid = InvalidOid;
|
|
rte->subquery = NULL;
|
|
rte->funcexpr = funcexpr;
|
|
rte->coldeflist = coldeflist;
|
|
rte->alias = alias;
|
|
|
|
eref = makeAlias(alias ? alias->aliasname : funcname, NIL);
|
|
rte->eref = eref;
|
|
|
|
/*
|
|
* Now determine if the function returns a simple or composite type,
|
|
* and check/add column aliases.
|
|
*/
|
|
if (coldeflist != NIL)
|
|
{
|
|
/*
|
|
* we *only* allow a coldeflist for functions returning a RECORD
|
|
* pseudo-type
|
|
*/
|
|
if (funcrettype != RECORDOID)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_SYNTAX_ERROR),
|
|
errmsg("a column definition list is only allowed for functions returning \"record\"")));
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
* ... and a coldeflist is *required* for functions returning a
|
|
* RECORD pseudo-type
|
|
*/
|
|
if (funcrettype == RECORDOID)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_SYNTAX_ERROR),
|
|
errmsg("a column definition list is required for functions returning \"record\"")));
|
|
}
|
|
|
|
functyptype = get_typtype(funcrettype);
|
|
|
|
if (functyptype == 'c')
|
|
{
|
|
/*
|
|
* Named composite data type, i.e. a table's row type
|
|
*/
|
|
Oid funcrelid = typeidTypeRelid(funcrettype);
|
|
Relation rel;
|
|
|
|
if (!OidIsValid(funcrelid)) /* shouldn't happen if typtype is
|
|
* 'c' */
|
|
elog(ERROR, "invalid typrelid for complex type %u", funcrettype);
|
|
|
|
/*
|
|
* Get the rel's relcache entry. This access ensures that we have
|
|
* an up-to-date relcache entry for the rel.
|
|
*/
|
|
rel = relation_open(funcrelid, AccessShareLock);
|
|
|
|
/* Build the column alias list */
|
|
buildRelationAliases(rel->rd_att, alias, eref);
|
|
|
|
/*
|
|
* Drop the rel refcount, but keep the access lock till end of
|
|
* transaction so that the table can't be deleted or have its
|
|
* schema modified underneath us.
|
|
*/
|
|
relation_close(rel, NoLock);
|
|
}
|
|
else if (functyptype == 'b' || functyptype == 'd')
|
|
{
|
|
/*
|
|
* Must be a base data type, i.e. scalar. Just add one alias
|
|
* column named for the function.
|
|
*/
|
|
if (alias && alias->colnames != NIL)
|
|
{
|
|
if (list_length(alias->colnames) != 1)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_INVALID_COLUMN_REFERENCE),
|
|
errmsg("too many column aliases specified for function %s",
|
|
funcname)));
|
|
eref->colnames = copyObject(alias->colnames);
|
|
}
|
|
else
|
|
eref->colnames = list_make1(makeString(eref->aliasname));
|
|
}
|
|
else if (functyptype == 'p' && funcrettype == RECORDOID)
|
|
{
|
|
ListCell *col;
|
|
|
|
/* Use the column definition list to form the alias list */
|
|
foreach(col, coldeflist)
|
|
{
|
|
ColumnDef *n = lfirst(col);
|
|
char *attrname;
|
|
|
|
attrname = pstrdup(n->colname);
|
|
eref->colnames = lappend(eref->colnames, makeString(attrname));
|
|
}
|
|
}
|
|
else
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_DATATYPE_MISMATCH),
|
|
errmsg("function \"%s\" in FROM has unsupported return type",
|
|
funcname)));
|
|
|
|
/*----------
|
|
* Flags:
|
|
* - this RTE should be expanded to include descendant tables,
|
|
* - this RTE is in the FROM clause,
|
|
* - this RTE should be checked for appropriate access rights.
|
|
*
|
|
* Functions are never checked for access rights (at least, not by
|
|
* the RTE permissions mechanism).
|
|
*----------
|
|
*/
|
|
rte->inh = false; /* never true for functions */
|
|
rte->inFromCl = inFromCl;
|
|
|
|
rte->requiredPerms = 0;
|
|
rte->checkAsUser = 0;
|
|
|
|
/*
|
|
* Add completed RTE to pstate's range table list, but not to join
|
|
* list nor namespace --- caller must do that if appropriate.
|
|
*/
|
|
if (pstate != NULL)
|
|
pstate->p_rtable = lappend(pstate->p_rtable, rte);
|
|
|
|
return rte;
|
|
}
|
|
|
|
/*
|
|
* Add an entry for a join to the pstate's range table (p_rtable).
|
|
*
|
|
* This is much like addRangeTableEntry() except that it makes a join RTE.
|
|
*/
|
|
RangeTblEntry *
|
|
addRangeTableEntryForJoin(ParseState *pstate,
|
|
List *colnames,
|
|
JoinType jointype,
|
|
List *aliasvars,
|
|
Alias *alias,
|
|
bool inFromCl)
|
|
{
|
|
RangeTblEntry *rte = makeNode(RangeTblEntry);
|
|
Alias *eref;
|
|
int numaliases;
|
|
|
|
rte->rtekind = RTE_JOIN;
|
|
rte->relid = InvalidOid;
|
|
rte->subquery = NULL;
|
|
rte->jointype = jointype;
|
|
rte->joinaliasvars = aliasvars;
|
|
rte->alias = alias;
|
|
|
|
eref = alias ? (Alias *) copyObject(alias) : makeAlias("unnamed_join", NIL);
|
|
numaliases = list_length(eref->colnames);
|
|
|
|
/* fill in any unspecified alias columns */
|
|
if (numaliases < list_length(colnames))
|
|
eref->colnames = list_concat(eref->colnames,
|
|
list_copy_tail(colnames, numaliases));
|
|
|
|
rte->eref = eref;
|
|
|
|
/*----------
|
|
* Flags:
|
|
* - this RTE should be expanded to include descendant tables,
|
|
* - this RTE is in the FROM clause,
|
|
* - this RTE should be checked for appropriate access rights.
|
|
*
|
|
* Joins are never checked for access rights.
|
|
*----------
|
|
*/
|
|
rte->inh = false; /* never true for joins */
|
|
rte->inFromCl = inFromCl;
|
|
|
|
rte->requiredPerms = 0;
|
|
rte->checkAsUser = 0;
|
|
|
|
/*
|
|
* Add completed RTE to pstate's range table list, but not to join
|
|
* list nor namespace --- caller must do that if appropriate.
|
|
*/
|
|
if (pstate != NULL)
|
|
pstate->p_rtable = lappend(pstate->p_rtable, rte);
|
|
|
|
return rte;
|
|
}
|
|
|
|
/*
|
|
* Has the specified refname been selected FOR UPDATE?
|
|
*/
|
|
static bool
|
|
isForUpdate(ParseState *pstate, char *refname)
|
|
{
|
|
/* Outer loop to check parent query levels as well as this one */
|
|
while (pstate != NULL)
|
|
{
|
|
if (pstate->p_forUpdate != NIL)
|
|
{
|
|
if (linitial(pstate->p_forUpdate) == NULL)
|
|
{
|
|
/* all tables used in query */
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
/* just the named tables */
|
|
ListCell *l;
|
|
|
|
foreach(l, pstate->p_forUpdate)
|
|
{
|
|
char *rname = strVal(lfirst(l));
|
|
|
|
if (strcmp(refname, rname) == 0)
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
pstate = pstate->parentParseState;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
* Add the given RTE as a top-level entry in the pstate's join list
|
|
* and/or name space list. (We assume caller has checked for any
|
|
* namespace conflict.)
|
|
*/
|
|
void
|
|
addRTEtoQuery(ParseState *pstate, RangeTblEntry *rte,
|
|
bool addToJoinList, bool addToNameSpace)
|
|
{
|
|
int rtindex = RTERangeTablePosn(pstate, rte, NULL);
|
|
RangeTblRef *rtr = makeNode(RangeTblRef);
|
|
|
|
rtr->rtindex = rtindex;
|
|
|
|
if (addToJoinList)
|
|
pstate->p_joinlist = lappend(pstate->p_joinlist, rtr);
|
|
if (addToNameSpace)
|
|
pstate->p_namespace = lappend(pstate->p_namespace, rtr);
|
|
}
|
|
|
|
/*
|
|
* Add a POSTQUEL-style implicit RTE.
|
|
*
|
|
* We assume caller has already checked that there is no RTE or join with
|
|
* a conflicting name.
|
|
*/
|
|
RangeTblEntry *
|
|
addImplicitRTE(ParseState *pstate, RangeVar *relation)
|
|
{
|
|
RangeTblEntry *rte;
|
|
|
|
rte = addRangeTableEntry(pstate, relation, NULL, false, false);
|
|
addRTEtoQuery(pstate, rte, true, true);
|
|
warnAutoRange(pstate, relation);
|
|
|
|
return rte;
|
|
}
|
|
|
|
/*
|
|
* expandRTE -- expand the columns of a rangetable entry
|
|
*
|
|
* This creates lists of an RTE's column names (aliases if provided, else
|
|
* real names) and Vars for each column. Only user columns are considered.
|
|
* If include_dropped is FALSE then dropped columns are omitted from the
|
|
* results. If include_dropped is TRUE then empty strings and NULL constants
|
|
* (not Vars!) are returned for dropped columns.
|
|
*
|
|
* The target RTE is the rtindex'th entry of rtable. (The whole rangetable
|
|
* must be passed since we need it to determine dropped-ness for JOIN columns.)
|
|
* sublevels_up is the varlevelsup value to use in the created Vars.
|
|
*
|
|
* The output lists go into *colnames and *colvars.
|
|
* If only one of the two kinds of output list is needed, pass NULL for the
|
|
* output pointer for the unwanted one.
|
|
*/
|
|
void
|
|
expandRTE(List *rtable, int rtindex, int sublevels_up,
|
|
bool include_dropped,
|
|
List **colnames, List **colvars)
|
|
{
|
|
RangeTblEntry *rte = rt_fetch(rtindex, rtable);
|
|
int varattno;
|
|
|
|
if (colnames)
|
|
*colnames = NIL;
|
|
if (colvars)
|
|
*colvars = NIL;
|
|
|
|
switch (rte->rtekind)
|
|
{
|
|
case RTE_RELATION:
|
|
/* Ordinary relation RTE */
|
|
expandRelation(rte->relid, rte->eref, rtindex, sublevels_up,
|
|
include_dropped, colnames, colvars);
|
|
break;
|
|
case RTE_SUBQUERY:
|
|
{
|
|
/* Subquery RTE */
|
|
ListCell *aliasp_item = list_head(rte->eref->colnames);
|
|
ListCell *tlistitem;
|
|
|
|
varattno = 0;
|
|
foreach(tlistitem, rte->subquery->targetList)
|
|
{
|
|
TargetEntry *te = (TargetEntry *) lfirst(tlistitem);
|
|
|
|
if (te->resdom->resjunk)
|
|
continue;
|
|
varattno++;
|
|
Assert(varattno == te->resdom->resno);
|
|
|
|
if (colnames)
|
|
{
|
|
/* Assume there is one alias per target item */
|
|
char *label = strVal(lfirst(aliasp_item));
|
|
|
|
*colnames = lappend(*colnames, makeString(pstrdup(label)));
|
|
aliasp_item = lnext(aliasp_item);
|
|
}
|
|
|
|
if (colvars)
|
|
{
|
|
Var *varnode;
|
|
|
|
varnode = makeVar(rtindex, varattno,
|
|
te->resdom->restype,
|
|
te->resdom->restypmod,
|
|
sublevels_up);
|
|
|
|
*colvars = lappend(*colvars, varnode);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case RTE_FUNCTION:
|
|
{
|
|
/* Function RTE */
|
|
Oid funcrettype = exprType(rte->funcexpr);
|
|
char functyptype = get_typtype(funcrettype);
|
|
|
|
if (functyptype == 'c')
|
|
{
|
|
/*
|
|
* Composite data type, i.e. a table's row type
|
|
*
|
|
* Same as ordinary relation RTE
|
|
*/
|
|
Oid funcrelid = typeidTypeRelid(funcrettype);
|
|
|
|
if (!OidIsValid(funcrelid)) /* shouldn't happen */
|
|
elog(ERROR, "invalid typrelid for complex type %u",
|
|
funcrettype);
|
|
|
|
expandRelation(funcrelid, rte->eref, rtindex, sublevels_up,
|
|
include_dropped, colnames, colvars);
|
|
}
|
|
else if (functyptype == 'b' || functyptype == 'd')
|
|
{
|
|
/*
|
|
* Must be a base data type, i.e. scalar
|
|
*/
|
|
if (colnames)
|
|
*colnames = lappend(*colnames,
|
|
linitial(rte->eref->colnames));
|
|
|
|
if (colvars)
|
|
{
|
|
Var *varnode;
|
|
|
|
varnode = makeVar(rtindex, 1,
|
|
funcrettype, -1,
|
|
sublevels_up);
|
|
|
|
*colvars = lappend(*colvars, varnode);
|
|
}
|
|
}
|
|
else if (functyptype == 'p' && funcrettype == RECORDOID)
|
|
{
|
|
List *coldeflist = rte->coldeflist;
|
|
ListCell *col;
|
|
int attnum = 0;
|
|
|
|
foreach(col, coldeflist)
|
|
{
|
|
ColumnDef *colDef = lfirst(col);
|
|
|
|
attnum++;
|
|
if (colnames)
|
|
{
|
|
char *attrname;
|
|
|
|
attrname = pstrdup(colDef->colname);
|
|
*colnames = lappend(*colnames, makeString(attrname));
|
|
}
|
|
|
|
if (colvars)
|
|
{
|
|
Var *varnode;
|
|
Oid atttypid;
|
|
|
|
atttypid = typenameTypeId(colDef->typename);
|
|
|
|
varnode = makeVar(rtindex,
|
|
attnum,
|
|
atttypid,
|
|
-1,
|
|
sublevels_up);
|
|
|
|
*colvars = lappend(*colvars, varnode);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_DATATYPE_MISMATCH),
|
|
errmsg("function in FROM has unsupported return type")));
|
|
}
|
|
break;
|
|
case RTE_JOIN:
|
|
{
|
|
/* Join RTE */
|
|
ListCell *colname;
|
|
ListCell *aliasvar;
|
|
|
|
Assert(list_length(rte->eref->colnames) == list_length(rte->joinaliasvars));
|
|
|
|
varattno = 0;
|
|
forboth (colname, rte->eref->colnames, aliasvar, rte->joinaliasvars)
|
|
{
|
|
varattno++;
|
|
|
|
/*
|
|
* During ordinary parsing, there will never be any
|
|
* deleted columns in the join; but we have to check
|
|
* since this routine is also used by the rewriter,
|
|
* and joins found in stored rules might have join
|
|
* columns for since-deleted columns.
|
|
*/
|
|
if (get_rte_attribute_is_dropped(rtable, rtindex,
|
|
varattno))
|
|
{
|
|
if (include_dropped)
|
|
{
|
|
if (colnames)
|
|
*colnames = lappend(*colnames,
|
|
makeString(pstrdup("")));
|
|
if (colvars)
|
|
{
|
|
/*
|
|
* can't use atttypid here, but it doesn't
|
|
* really matter what type the Const claims to
|
|
* be.
|
|
*/
|
|
*colvars = lappend(*colvars,
|
|
makeNullConst(INT4OID));
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (colnames)
|
|
{
|
|
char *label = strVal(lfirst(colname));
|
|
|
|
*colnames = lappend(*colnames,
|
|
makeString(pstrdup(label)));
|
|
}
|
|
|
|
if (colvars)
|
|
{
|
|
Node *avar = (Node *) lfirst(aliasvar);
|
|
Var *varnode;
|
|
|
|
varnode = makeVar(rtindex, varattno,
|
|
exprType(avar),
|
|
exprTypmod(avar),
|
|
sublevels_up);
|
|
|
|
*colvars = lappend(*colvars, varnode);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
elog(ERROR, "unrecognized RTE kind: %d", (int) rte->rtekind);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* expandRelation -- expandRTE subroutine
|
|
*/
|
|
static void
|
|
expandRelation(Oid relid, Alias *eref, int rtindex, int sublevels_up,
|
|
bool include_dropped,
|
|
List **colnames, List **colvars)
|
|
{
|
|
Relation rel;
|
|
int varattno;
|
|
int maxattrs;
|
|
int numaliases;
|
|
|
|
rel = relation_open(relid, AccessShareLock);
|
|
maxattrs = RelationGetNumberOfAttributes(rel);
|
|
numaliases = list_length(eref->colnames);
|
|
|
|
for (varattno = 0; varattno < maxattrs; varattno++)
|
|
{
|
|
Form_pg_attribute attr = rel->rd_att->attrs[varattno];
|
|
|
|
if (attr->attisdropped)
|
|
{
|
|
if (include_dropped)
|
|
{
|
|
if (colnames)
|
|
*colnames = lappend(*colnames, makeString(pstrdup("")));
|
|
if (colvars)
|
|
{
|
|
/*
|
|
* can't use atttypid here, but it doesn't really matter
|
|
* what type the Const claims to be.
|
|
*/
|
|
*colvars = lappend(*colvars, makeNullConst(INT4OID));
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (colnames)
|
|
{
|
|
char *label;
|
|
|
|
if (varattno < numaliases)
|
|
label = strVal(list_nth(eref->colnames, varattno));
|
|
else
|
|
label = NameStr(attr->attname);
|
|
*colnames = lappend(*colnames, makeString(pstrdup(label)));
|
|
}
|
|
|
|
if (colvars)
|
|
{
|
|
Var *varnode;
|
|
|
|
varnode = makeVar(rtindex, attr->attnum,
|
|
attr->atttypid, attr->atttypmod,
|
|
sublevels_up);
|
|
|
|
*colvars = lappend(*colvars, varnode);
|
|
}
|
|
}
|
|
|
|
relation_close(rel, AccessShareLock);
|
|
}
|
|
|
|
/*
|
|
* expandRelAttrs -
|
|
* Workhorse for "*" expansion: produce a list of targetentries
|
|
* for the attributes of the rte
|
|
*/
|
|
List *
|
|
expandRelAttrs(ParseState *pstate, List *rtable, int rtindex, int sublevels_up)
|
|
{
|
|
List *names,
|
|
*vars;
|
|
ListCell *name,
|
|
*var;
|
|
List *te_list = NIL;
|
|
|
|
expandRTE(rtable, rtindex, sublevels_up, false, &names, &vars);
|
|
|
|
forboth(name, names, var, vars)
|
|
{
|
|
char *label = strVal(lfirst(name));
|
|
Node *varnode = (Node *) lfirst(var);
|
|
TargetEntry *te = makeNode(TargetEntry);
|
|
|
|
te->resdom = makeResdom((AttrNumber) pstate->p_next_resno++,
|
|
exprType(varnode),
|
|
exprTypmod(varnode),
|
|
label,
|
|
false);
|
|
te->expr = (Expr *) varnode;
|
|
te_list = lappend(te_list, te);
|
|
}
|
|
|
|
Assert(name == NULL && var == NULL); /* lists not the same length? */
|
|
|
|
return te_list;
|
|
}
|
|
|
|
/*
|
|
* get_rte_attribute_name
|
|
* Get an attribute name from a RangeTblEntry
|
|
*
|
|
* This is unlike get_attname() because we use aliases if available.
|
|
* In particular, it will work on an RTE for a subselect or join, whereas
|
|
* get_attname() only works on real relations.
|
|
*
|
|
* "*" is returned if the given attnum is InvalidAttrNumber --- this case
|
|
* occurs when a Var represents a whole tuple of a relation.
|
|
*/
|
|
char *
|
|
get_rte_attribute_name(RangeTblEntry *rte, AttrNumber attnum)
|
|
{
|
|
if (attnum == InvalidAttrNumber)
|
|
return "*";
|
|
|
|
/*
|
|
* If there is a user-written column alias, use it.
|
|
*/
|
|
if (rte->alias &&
|
|
attnum > 0 && attnum <= list_length(rte->alias->colnames))
|
|
return strVal(list_nth(rte->alias->colnames, attnum - 1));
|
|
|
|
/*
|
|
* If the RTE is a relation, go to the system catalogs not the
|
|
* eref->colnames list. This is a little slower but it will give the
|
|
* right answer if the column has been renamed since the eref list was
|
|
* built (which can easily happen for rules).
|
|
*/
|
|
if (rte->rtekind == RTE_RELATION)
|
|
return get_relid_attribute_name(rte->relid, attnum);
|
|
|
|
/*
|
|
* Otherwise use the column name from eref. There should always be
|
|
* one.
|
|
*/
|
|
if (attnum > 0 && attnum <= list_length(rte->eref->colnames))
|
|
return strVal(list_nth(rte->eref->colnames, attnum - 1));
|
|
|
|
/* else caller gave us a bogus attnum */
|
|
elog(ERROR, "invalid attnum %d for rangetable entry %s",
|
|
attnum, rte->eref->aliasname);
|
|
return NULL; /* keep compiler quiet */
|
|
}
|
|
|
|
/*
|
|
* get_rte_attribute_type
|
|
* Get attribute type information from a RangeTblEntry
|
|
*/
|
|
void
|
|
get_rte_attribute_type(RangeTblEntry *rte, AttrNumber attnum,
|
|
Oid *vartype, int32 *vartypmod)
|
|
{
|
|
switch (rte->rtekind)
|
|
{
|
|
case RTE_RELATION:
|
|
{
|
|
/* Plain relation RTE --- get the attribute's type info */
|
|
HeapTuple tp;
|
|
Form_pg_attribute att_tup;
|
|
|
|
tp = SearchSysCache(ATTNUM,
|
|
ObjectIdGetDatum(rte->relid),
|
|
Int16GetDatum(attnum),
|
|
0, 0);
|
|
if (!HeapTupleIsValid(tp)) /* shouldn't happen */
|
|
elog(ERROR, "cache lookup failed for attribute %d of relation %u",
|
|
attnum, rte->relid);
|
|
att_tup = (Form_pg_attribute) GETSTRUCT(tp);
|
|
|
|
/*
|
|
* If dropped column, pretend it ain't there. See notes
|
|
* in scanRTEForColumn.
|
|
*/
|
|
if (att_tup->attisdropped)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_UNDEFINED_COLUMN),
|
|
errmsg("column \"%s\" of relation \"%s\" does not exist",
|
|
NameStr(att_tup->attname),
|
|
get_rel_name(rte->relid))));
|
|
*vartype = att_tup->atttypid;
|
|
*vartypmod = att_tup->atttypmod;
|
|
ReleaseSysCache(tp);
|
|
}
|
|
break;
|
|
case RTE_SUBQUERY:
|
|
{
|
|
/* Subselect RTE --- get type info from subselect's tlist */
|
|
TargetEntry *te = get_tle_by_resno(rte->subquery->targetList,
|
|
attnum);
|
|
|
|
if (te == NULL || te->resdom->resjunk)
|
|
elog(ERROR, "subquery %s does not have attribute %d",
|
|
rte->eref->aliasname, attnum);
|
|
*vartype = te->resdom->restype;
|
|
*vartypmod = te->resdom->restypmod;
|
|
}
|
|
break;
|
|
case RTE_FUNCTION:
|
|
{
|
|
/* Function RTE */
|
|
Oid funcrettype = exprType(rte->funcexpr);
|
|
char functyptype = get_typtype(funcrettype);
|
|
List *coldeflist = rte->coldeflist;
|
|
|
|
if (functyptype == 'c')
|
|
{
|
|
/*
|
|
* Composite data type, i.e. a table's row type Same
|
|
* as ordinary relation RTE
|
|
*/
|
|
Oid funcrelid = typeidTypeRelid(funcrettype);
|
|
HeapTuple tp;
|
|
Form_pg_attribute att_tup;
|
|
|
|
if (!OidIsValid(funcrelid)) /* shouldn't happen */
|
|
elog(ERROR, "invalid typrelid for complex type %u",
|
|
funcrettype);
|
|
|
|
tp = SearchSysCache(ATTNUM,
|
|
ObjectIdGetDatum(funcrelid),
|
|
Int16GetDatum(attnum),
|
|
0, 0);
|
|
if (!HeapTupleIsValid(tp)) /* shouldn't happen */
|
|
elog(ERROR, "cache lookup failed for attribute %d of relation %u",
|
|
attnum, funcrelid);
|
|
att_tup = (Form_pg_attribute) GETSTRUCT(tp);
|
|
|
|
/*
|
|
* If dropped column, pretend it ain't there. See
|
|
* notes in scanRTEForColumn.
|
|
*/
|
|
if (att_tup->attisdropped)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_UNDEFINED_COLUMN),
|
|
errmsg("column \"%s\" of relation \"%s\" does not exist",
|
|
NameStr(att_tup->attname),
|
|
get_rel_name(funcrelid))));
|
|
*vartype = att_tup->atttypid;
|
|
*vartypmod = att_tup->atttypmod;
|
|
ReleaseSysCache(tp);
|
|
}
|
|
else if (functyptype == 'b' || functyptype == 'd')
|
|
{
|
|
/*
|
|
* Must be a base data type, i.e. scalar
|
|
*/
|
|
*vartype = funcrettype;
|
|
*vartypmod = -1;
|
|
}
|
|
else if (functyptype == 'p' && funcrettype == RECORDOID)
|
|
{
|
|
ColumnDef *colDef = list_nth(coldeflist, attnum - 1);
|
|
|
|
*vartype = typenameTypeId(colDef->typename);
|
|
*vartypmod = -1;
|
|
}
|
|
else
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_DATATYPE_MISMATCH),
|
|
errmsg("function in FROM has unsupported return type")));
|
|
}
|
|
break;
|
|
case RTE_JOIN:
|
|
{
|
|
/*
|
|
* Join RTE --- get type info from join RTE's alias
|
|
* variable
|
|
*/
|
|
Node *aliasvar;
|
|
|
|
Assert(attnum > 0 && attnum <= list_length(rte->joinaliasvars));
|
|
aliasvar = (Node *) list_nth(rte->joinaliasvars, attnum - 1);
|
|
*vartype = exprType(aliasvar);
|
|
*vartypmod = exprTypmod(aliasvar);
|
|
}
|
|
break;
|
|
default:
|
|
elog(ERROR, "unrecognized RTE kind: %d", (int) rte->rtekind);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* get_rte_attribute_is_dropped
|
|
* Check whether attempted attribute ref is to a dropped column
|
|
*/
|
|
bool
|
|
get_rte_attribute_is_dropped(List *rtable, int rtindex, AttrNumber attnum)
|
|
{
|
|
RangeTblEntry *rte = rt_fetch(rtindex, rtable);
|
|
bool result;
|
|
|
|
switch (rte->rtekind)
|
|
{
|
|
case RTE_RELATION:
|
|
{
|
|
/* Plain relation RTE --- get the attribute's catalog entry */
|
|
HeapTuple tp;
|
|
Form_pg_attribute att_tup;
|
|
|
|
tp = SearchSysCache(ATTNUM,
|
|
ObjectIdGetDatum(rte->relid),
|
|
Int16GetDatum(attnum),
|
|
0, 0);
|
|
if (!HeapTupleIsValid(tp)) /* shouldn't happen */
|
|
elog(ERROR, "cache lookup failed for attribute %d of relation %u",
|
|
attnum, rte->relid);
|
|
att_tup = (Form_pg_attribute) GETSTRUCT(tp);
|
|
result = att_tup->attisdropped;
|
|
ReleaseSysCache(tp);
|
|
}
|
|
break;
|
|
case RTE_SUBQUERY:
|
|
/* Subselect RTEs never have dropped columns */
|
|
result = false;
|
|
break;
|
|
case RTE_JOIN:
|
|
{
|
|
/*
|
|
* A join RTE would not have dropped columns when constructed,
|
|
* but one in a stored rule might contain columns that were
|
|
* dropped from the underlying tables, if said columns are
|
|
* nowhere explicitly referenced in the rule. So we have to
|
|
* recursively look at the referenced column.
|
|
*/
|
|
Var *aliasvar;
|
|
|
|
if (attnum <= 0 ||
|
|
attnum > list_length(rte->joinaliasvars))
|
|
elog(ERROR, "invalid varattno %d", attnum);
|
|
aliasvar = (Var *) list_nth(rte->joinaliasvars, attnum - 1);
|
|
/*
|
|
* If the list item isn't a simple Var, then it must
|
|
* represent a merged column, ie a USING column, and so it
|
|
* couldn't possibly be dropped (since it's referenced in
|
|
* the join clause).
|
|
*/
|
|
if (!IsA(aliasvar, Var))
|
|
result = false;
|
|
else
|
|
result = get_rte_attribute_is_dropped(rtable,
|
|
aliasvar->varno,
|
|
aliasvar->varattno);
|
|
}
|
|
break;
|
|
case RTE_FUNCTION:
|
|
{
|
|
/* Function RTE */
|
|
Oid funcrettype = exprType(rte->funcexpr);
|
|
Oid funcrelid = typeidTypeRelid(funcrettype);
|
|
|
|
if (OidIsValid(funcrelid))
|
|
{
|
|
/*
|
|
* Composite data type, i.e. a table's row type
|
|
*
|
|
* Same as ordinary relation RTE
|
|
*/
|
|
HeapTuple tp;
|
|
Form_pg_attribute att_tup;
|
|
|
|
tp = SearchSysCache(ATTNUM,
|
|
ObjectIdGetDatum(funcrelid),
|
|
Int16GetDatum(attnum),
|
|
0, 0);
|
|
if (!HeapTupleIsValid(tp)) /* shouldn't happen */
|
|
elog(ERROR, "cache lookup failed for attribute %d of relation %u",
|
|
attnum, funcrelid);
|
|
att_tup = (Form_pg_attribute) GETSTRUCT(tp);
|
|
result = att_tup->attisdropped;
|
|
ReleaseSysCache(tp);
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
* Must be a base data type, i.e. scalar
|
|
*/
|
|
result = false;
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
elog(ERROR, "unrecognized RTE kind: %d", (int) rte->rtekind);
|
|
result = false; /* keep compiler quiet */
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Given a targetlist and a resno, return the matching TargetEntry
|
|
*
|
|
* Returns NULL if resno is not present in list.
|
|
*
|
|
* Note: we need to search, rather than just indexing with list_nth(),
|
|
* because not all tlists are sorted by resno.
|
|
*/
|
|
TargetEntry *
|
|
get_tle_by_resno(List *tlist, AttrNumber resno)
|
|
{
|
|
ListCell *l;
|
|
|
|
foreach(l, tlist)
|
|
{
|
|
TargetEntry *tle = (TargetEntry *) lfirst(l);
|
|
|
|
if (tle->resdom->resno == resno)
|
|
return tle;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* given relation and att name, return id of variable
|
|
*
|
|
* This should only be used if the relation is already
|
|
* heap_open()'ed. Use the cache version get_attnum()
|
|
* for access to non-opened relations.
|
|
*/
|
|
int
|
|
attnameAttNum(Relation rd, const char *attname, bool sysColOK)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < rd->rd_rel->relnatts; i++)
|
|
{
|
|
Form_pg_attribute att = rd->rd_att->attrs[i];
|
|
|
|
if (namestrcmp(&(att->attname), attname) == 0 && !att->attisdropped)
|
|
return i + 1;
|
|
}
|
|
|
|
if (sysColOK)
|
|
{
|
|
if ((i = specialAttNum(attname)) != InvalidAttrNumber)
|
|
{
|
|
if (i != ObjectIdAttributeNumber || rd->rd_rel->relhasoids)
|
|
return i;
|
|
}
|
|
}
|
|
|
|
/* on failure */
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_UNDEFINED_COLUMN),
|
|
errmsg("column \"%s\" of relation \"%s\" does not exist",
|
|
attname, RelationGetRelationName(rd))));
|
|
return InvalidAttrNumber; /* keep compiler quiet */
|
|
}
|
|
|
|
/* specialAttNum()
|
|
*
|
|
* Check attribute name to see if it is "special", e.g. "oid".
|
|
* - thomas 2000-02-07
|
|
*
|
|
* Note: this only discovers whether the name could be a system attribute.
|
|
* Caller needs to verify that it really is an attribute of the rel,
|
|
* at least in the case of "oid", which is now optional.
|
|
*/
|
|
static int
|
|
specialAttNum(const char *attname)
|
|
{
|
|
Form_pg_attribute sysatt;
|
|
|
|
sysatt = SystemAttributeByName(attname,
|
|
true /* "oid" will be accepted */ );
|
|
if (sysatt != NULL)
|
|
return sysatt->attnum;
|
|
return InvalidAttrNumber;
|
|
}
|
|
|
|
|
|
/*
|
|
* given attribute id, return name of that attribute
|
|
*
|
|
* This should only be used if the relation is already
|
|
* heap_open()'ed. Use the cache version get_atttype()
|
|
* for access to non-opened relations.
|
|
*/
|
|
Name
|
|
attnumAttName(Relation rd, int attid)
|
|
{
|
|
if (attid <= 0)
|
|
{
|
|
Form_pg_attribute sysatt;
|
|
|
|
sysatt = SystemAttributeDefinition(attid, rd->rd_rel->relhasoids);
|
|
return &sysatt->attname;
|
|
}
|
|
if (attid > rd->rd_att->natts)
|
|
elog(ERROR, "invalid attribute number %d", attid);
|
|
return &rd->rd_att->attrs[attid - 1]->attname;
|
|
}
|
|
|
|
/*
|
|
* given attribute id, return type of that attribute
|
|
*
|
|
* This should only be used if the relation is already
|
|
* heap_open()'ed. Use the cache version get_atttype()
|
|
* for access to non-opened relations.
|
|
*/
|
|
Oid
|
|
attnumTypeId(Relation rd, int attid)
|
|
{
|
|
if (attid <= 0)
|
|
{
|
|
Form_pg_attribute sysatt;
|
|
|
|
sysatt = SystemAttributeDefinition(attid, rd->rd_rel->relhasoids);
|
|
return sysatt->atttypid;
|
|
}
|
|
if (attid > rd->rd_att->natts)
|
|
elog(ERROR, "invalid attribute number %d", attid);
|
|
return rd->rd_att->attrs[attid - 1]->atttypid;
|
|
}
|
|
|
|
/*
|
|
* Generate a warning or error about an implicit RTE, if appropriate.
|
|
*
|
|
* If ADD_MISSING_FROM is not enabled, raise an error.
|
|
*
|
|
* Our current theory on warnings is that we should allow "SELECT foo.*"
|
|
* but warn about a mixture of explicit and implicit RTEs.
|
|
*/
|
|
static void
|
|
warnAutoRange(ParseState *pstate, RangeVar *relation)
|
|
{
|
|
bool foundInFromCl = false;
|
|
ListCell *temp;
|
|
|
|
if (!add_missing_from)
|
|
{
|
|
if (pstate->parentParseState != NULL)
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_UNDEFINED_TABLE),
|
|
errmsg("missing FROM-clause entry in subquery for table \"%s\"",
|
|
relation->relname)));
|
|
else
|
|
ereport(ERROR,
|
|
(errcode(ERRCODE_UNDEFINED_TABLE),
|
|
errmsg("missing FROM-clause entry for table \"%s\"",
|
|
relation->relname)));
|
|
}
|
|
|
|
foreach(temp, pstate->p_rtable)
|
|
{
|
|
RangeTblEntry *rte = lfirst(temp);
|
|
|
|
if (rte->inFromCl)
|
|
{
|
|
foundInFromCl = true;
|
|
break;
|
|
}
|
|
}
|
|
if (foundInFromCl)
|
|
{
|
|
if (pstate->parentParseState != NULL)
|
|
ereport(NOTICE,
|
|
(errcode(ERRCODE_UNDEFINED_TABLE),
|
|
errmsg("adding missing FROM-clause entry in subquery for table \"%s\"",
|
|
relation->relname)));
|
|
else
|
|
ereport(NOTICE,
|
|
(errcode(ERRCODE_UNDEFINED_TABLE),
|
|
errmsg("adding missing FROM-clause entry for table \"%s\"",
|
|
relation->relname)));
|
|
}
|
|
}
|