
an 'opclass owner' column in pg_opclass. Nothing is done with it at present, but since there are plans to invent a CREATE OPERATOR CLASS command soon, we'll probably want DROP OPERATOR CLASS too, which suggests that a notion of ownership would be a good idea.
1801 lines
41 KiB
C
1801 lines
41 KiB
C
/*
|
|
* outfuncs.c
|
|
* routines to convert a node to ascii representation
|
|
*
|
|
* Portions Copyright (c) 1996-2001, PostgreSQL Global Development Group
|
|
* Portions Copyright (c) 1994, Regents of the University of California
|
|
*
|
|
* $Header: /cvsroot/pgsql/src/backend/nodes/outfuncs.c,v 1.156 2002/04/17 20:57:56 tgl Exp $
|
|
*
|
|
* NOTES
|
|
* Every (plan) node in POSTGRES has an associated "out" routine which
|
|
* knows how to create its ascii representation. These functions are
|
|
* useful for debugging as well as for storing plans in the system
|
|
* catalogs (eg. views).
|
|
*/
|
|
#include "postgres.h"
|
|
|
|
#include <ctype.h>
|
|
|
|
#include "lib/stringinfo.h"
|
|
#include "nodes/nodes.h"
|
|
#include "nodes/parsenodes.h"
|
|
#include "nodes/plannodes.h"
|
|
#include "nodes/primnodes.h"
|
|
#include "nodes/relation.h"
|
|
#include "parser/parse.h"
|
|
#include "utils/datum.h"
|
|
|
|
|
|
#define booltostr(x) ((x) ? "true" : "false")
|
|
|
|
static void _outDatum(StringInfo str, Datum value, int typlen, bool typbyval);
|
|
static void _outNode(StringInfo str, void *obj);
|
|
|
|
/*
|
|
* _outToken
|
|
* Convert an ordinary string (eg, an identifier) into a form that
|
|
* will be decoded back to a plain token by read.c's functions.
|
|
*
|
|
* If a null or empty string is given, it is encoded as "<>".
|
|
*/
|
|
static void
|
|
_outToken(StringInfo str, char *s)
|
|
{
|
|
if (s == NULL || *s == '\0')
|
|
{
|
|
appendStringInfo(str, "<>");
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Look for characters or patterns that are treated specially by
|
|
* read.c (either in pg_strtok() or in nodeRead()), and therefore need
|
|
* a protective backslash.
|
|
*/
|
|
/* These characters only need to be quoted at the start of the string */
|
|
if (*s == '<' ||
|
|
*s == '\"' ||
|
|
*s == '@' ||
|
|
isdigit((unsigned char) *s) ||
|
|
((*s == '+' || *s == '-') &&
|
|
(isdigit((unsigned char) s[1]) || s[1] == '.')))
|
|
appendStringInfoChar(str, '\\');
|
|
while (*s)
|
|
{
|
|
/* These chars must be backslashed anywhere in the string */
|
|
if (*s == ' ' || *s == '\n' || *s == '\t' ||
|
|
*s == '(' || *s == ')' || *s == '{' || *s == '}' ||
|
|
*s == '\\')
|
|
appendStringInfoChar(str, '\\');
|
|
appendStringInfoChar(str, *s++);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* _outIntList -
|
|
* converts a List of integers
|
|
*/
|
|
static void
|
|
_outIntList(StringInfo str, List *list)
|
|
{
|
|
List *l;
|
|
|
|
appendStringInfoChar(str, '(');
|
|
foreach(l, list)
|
|
appendStringInfo(str, " %d", lfirsti(l));
|
|
appendStringInfoChar(str, ')');
|
|
}
|
|
|
|
/*
|
|
* _outOidList -
|
|
* converts a List of OIDs
|
|
*/
|
|
static void
|
|
_outOidList(StringInfo str, List *list)
|
|
{
|
|
List *l;
|
|
|
|
appendStringInfoChar(str, '(');
|
|
foreach(l, list)
|
|
appendStringInfo(str, " %u", (Oid) lfirsti(l));
|
|
appendStringInfoChar(str, ')');
|
|
}
|
|
|
|
static void
|
|
_outCreateStmt(StringInfo str, CreateStmt *node)
|
|
{
|
|
appendStringInfo(str, " CREATE :relation ");
|
|
_outNode(str, node->relation);
|
|
|
|
appendStringInfo(str, " :tableElts ");
|
|
_outNode(str, node->tableElts);
|
|
|
|
appendStringInfo(str, " :inhRelations ");
|
|
_outNode(str, node->inhRelations);
|
|
|
|
appendStringInfo(str, " :constraints ");
|
|
_outNode(str, node->constraints);
|
|
|
|
appendStringInfo(str, " :hasoids %s ",
|
|
booltostr(node->hasoids));
|
|
}
|
|
|
|
static void
|
|
_outIndexStmt(StringInfo str, IndexStmt *node)
|
|
{
|
|
appendStringInfo(str, " INDEX :idxname ");
|
|
_outToken(str, node->idxname);
|
|
appendStringInfo(str, " :relation ");
|
|
_outNode(str, node->relation);
|
|
appendStringInfo(str, " :accessMethod ");
|
|
_outToken(str, node->accessMethod);
|
|
appendStringInfo(str, " :indexParams ");
|
|
_outNode(str, node->indexParams);
|
|
appendStringInfo(str, " :whereClause ");
|
|
_outNode(str, node->whereClause);
|
|
appendStringInfo(str, " :rangetable ");
|
|
_outNode(str, node->rangetable);
|
|
appendStringInfo(str, " :unique %s :primary %s ",
|
|
booltostr(node->unique),
|
|
booltostr(node->primary));
|
|
}
|
|
|
|
static void
|
|
_outNotifyStmt(StringInfo str, NotifyStmt *node)
|
|
{
|
|
appendStringInfo(str, "NOTIFY :relation ");
|
|
_outNode(str, node->relation);
|
|
}
|
|
|
|
static void
|
|
_outSelectStmt(StringInfo str, SelectStmt *node)
|
|
{
|
|
/* XXX this is pretty durn incomplete */
|
|
appendStringInfo(str, "SELECT :where ");
|
|
_outNode(str, node->whereClause);
|
|
}
|
|
|
|
static void
|
|
_outFuncCall(StringInfo str, FuncCall *node)
|
|
{
|
|
appendStringInfo(str, "FUNCTION ");
|
|
_outNode(str, node->funcname);
|
|
appendStringInfo(str, " :args ");
|
|
_outNode(str, node->args);
|
|
appendStringInfo(str, " :agg_star %s :agg_distinct %s ",
|
|
booltostr(node->agg_star),
|
|
booltostr(node->agg_distinct));
|
|
}
|
|
|
|
static void
|
|
_outColumnDef(StringInfo str, ColumnDef *node)
|
|
{
|
|
appendStringInfo(str, " COLUMNDEF :colname ");
|
|
_outToken(str, node->colname);
|
|
appendStringInfo(str, " :typename ");
|
|
_outNode(str, node->typename);
|
|
appendStringInfo(str, " :is_not_null %s :raw_default ",
|
|
booltostr(node->is_not_null));
|
|
_outNode(str, node->raw_default);
|
|
appendStringInfo(str, " :cooked_default ");
|
|
_outToken(str, node->cooked_default);
|
|
appendStringInfo(str, " :constraints ");
|
|
_outNode(str, node->constraints);
|
|
}
|
|
|
|
static void
|
|
_outTypeName(StringInfo str, TypeName *node)
|
|
{
|
|
appendStringInfo(str, " TYPENAME :names ");
|
|
_outNode(str, node->names);
|
|
appendStringInfo(str, " :typeid %u :timezone %s :setof %s"
|
|
" :pct_type %s typmod %d :arrayBounds ",
|
|
node->typeid,
|
|
booltostr(node->timezone),
|
|
booltostr(node->setof),
|
|
booltostr(node->pct_type),
|
|
node->typmod);
|
|
_outNode(str, node->arrayBounds);
|
|
}
|
|
|
|
static void
|
|
_outTypeCast(StringInfo str, TypeCast *node)
|
|
{
|
|
appendStringInfo(str, " TYPECAST :arg ");
|
|
_outNode(str, node->arg);
|
|
appendStringInfo(str, " :typename ");
|
|
_outNode(str, node->typename);
|
|
}
|
|
|
|
static void
|
|
_outIndexElem(StringInfo str, IndexElem *node)
|
|
{
|
|
appendStringInfo(str, " INDEXELEM :name ");
|
|
_outToken(str, node->name);
|
|
appendStringInfo(str, " :funcname ");
|
|
_outNode(str, node->funcname);
|
|
appendStringInfo(str, " :args ");
|
|
_outNode(str, node->args);
|
|
appendStringInfo(str, " :opclass ");
|
|
_outNode(str, node->opclass);
|
|
}
|
|
|
|
static void
|
|
_outQuery(StringInfo str, Query *node)
|
|
{
|
|
appendStringInfo(str, " QUERY :command %d :utility ", node->commandType);
|
|
|
|
/*
|
|
* Hack to work around missing outfuncs routines for a lot of the
|
|
* utility-statement node types. (The only one we actually *need*
|
|
* for rules support is NotifyStmt.) Someday we ought to support
|
|
* 'em all, but for the meantime do this to avoid getting lots of
|
|
* warnings when running with debug_print_parse on.
|
|
*/
|
|
if (node->utilityStmt)
|
|
{
|
|
switch (nodeTag(node->utilityStmt))
|
|
{
|
|
case T_CreateStmt:
|
|
case T_IndexStmt:
|
|
case T_NotifyStmt:
|
|
_outNode(str, node->utilityStmt);
|
|
break;
|
|
default:
|
|
appendStringInfo(str, "?");
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
appendStringInfo(str, "<>");
|
|
|
|
appendStringInfo(str, " :resultRelation %d :into ",
|
|
node->resultRelation);
|
|
_outNode(str, node->into);
|
|
|
|
appendStringInfo(str, " :isPortal %s :isBinary %s"
|
|
" :hasAggs %s :hasSubLinks %s :rtable ",
|
|
booltostr(node->isPortal),
|
|
booltostr(node->isBinary),
|
|
booltostr(node->hasAggs),
|
|
booltostr(node->hasSubLinks));
|
|
_outNode(str, node->rtable);
|
|
|
|
appendStringInfo(str, " :jointree ");
|
|
_outNode(str, node->jointree);
|
|
|
|
appendStringInfo(str, " :rowMarks ");
|
|
_outIntList(str, node->rowMarks);
|
|
|
|
appendStringInfo(str, " :targetList ");
|
|
_outNode(str, node->targetList);
|
|
|
|
appendStringInfo(str, " :groupClause ");
|
|
_outNode(str, node->groupClause);
|
|
|
|
appendStringInfo(str, " :havingQual ");
|
|
_outNode(str, node->havingQual);
|
|
|
|
appendStringInfo(str, " :distinctClause ");
|
|
_outNode(str, node->distinctClause);
|
|
|
|
appendStringInfo(str, " :sortClause ");
|
|
_outNode(str, node->sortClause);
|
|
|
|
appendStringInfo(str, " :limitOffset ");
|
|
_outNode(str, node->limitOffset);
|
|
|
|
appendStringInfo(str, " :limitCount ");
|
|
_outNode(str, node->limitCount);
|
|
|
|
appendStringInfo(str, " :setOperations ");
|
|
_outNode(str, node->setOperations);
|
|
|
|
appendStringInfo(str, " :resultRelations ");
|
|
_outIntList(str, node->resultRelations);
|
|
}
|
|
|
|
static void
|
|
_outSortClause(StringInfo str, SortClause *node)
|
|
{
|
|
appendStringInfo(str, " SORTCLAUSE :tleSortGroupRef %u :sortop %u ",
|
|
node->tleSortGroupRef, node->sortop);
|
|
}
|
|
|
|
static void
|
|
_outGroupClause(StringInfo str, GroupClause *node)
|
|
{
|
|
appendStringInfo(str, " GROUPCLAUSE :tleSortGroupRef %u :sortop %u ",
|
|
node->tleSortGroupRef, node->sortop);
|
|
}
|
|
|
|
static void
|
|
_outSetOperationStmt(StringInfo str, SetOperationStmt *node)
|
|
{
|
|
appendStringInfo(str, " SETOPERATIONSTMT :op %d :all %s :larg ",
|
|
(int) node->op,
|
|
booltostr(node->all));
|
|
_outNode(str, node->larg);
|
|
appendStringInfo(str, " :rarg ");
|
|
_outNode(str, node->rarg);
|
|
appendStringInfo(str, " :colTypes ");
|
|
_outOidList(str, node->colTypes);
|
|
}
|
|
|
|
/*
|
|
* print the basic stuff of all nodes that inherit from Plan
|
|
*
|
|
* NOTE: we deliberately omit the execution state (EState)
|
|
*/
|
|
static void
|
|
_outPlanInfo(StringInfo str, Plan *node)
|
|
{
|
|
appendStringInfo(str,
|
|
":startup_cost %.2f :total_cost %.2f :rows %.0f :width %d :qptargetlist ",
|
|
node->startup_cost,
|
|
node->total_cost,
|
|
node->plan_rows,
|
|
node->plan_width);
|
|
_outNode(str, node->targetlist);
|
|
|
|
appendStringInfo(str, " :qpqual ");
|
|
_outNode(str, node->qual);
|
|
|
|
appendStringInfo(str, " :lefttree ");
|
|
_outNode(str, node->lefttree);
|
|
|
|
appendStringInfo(str, " :righttree ");
|
|
_outNode(str, node->righttree);
|
|
|
|
appendStringInfo(str, " :extprm ");
|
|
_outIntList(str, node->extParam);
|
|
|
|
appendStringInfo(str, " :locprm ");
|
|
_outIntList(str, node->locParam);
|
|
|
|
appendStringInfo(str, " :initplan ");
|
|
_outNode(str, node->initPlan);
|
|
|
|
appendStringInfo(str, " :nprm %d ", node->nParamExec);
|
|
}
|
|
|
|
/*
|
|
* Stuff from plannodes.h
|
|
*/
|
|
static void
|
|
_outPlan(StringInfo str, Plan *node)
|
|
{
|
|
appendStringInfo(str, " PLAN ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
}
|
|
|
|
static void
|
|
_outResult(StringInfo str, Result *node)
|
|
{
|
|
appendStringInfo(str, " RESULT ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :resconstantqual ");
|
|
_outNode(str, node->resconstantqual);
|
|
|
|
}
|
|
|
|
/*
|
|
* Append is a subclass of Plan.
|
|
*/
|
|
static void
|
|
_outAppend(StringInfo str, Append *node)
|
|
{
|
|
appendStringInfo(str, " APPEND ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :appendplans ");
|
|
_outNode(str, node->appendplans);
|
|
|
|
appendStringInfo(str, " :isTarget %s ",
|
|
booltostr(node->isTarget));
|
|
}
|
|
|
|
/*
|
|
* Join is a subclass of Plan
|
|
*/
|
|
static void
|
|
_outJoin(StringInfo str, Join *node)
|
|
{
|
|
appendStringInfo(str, " JOIN ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
appendStringInfo(str, " :jointype %d :joinqual ",
|
|
(int) node->jointype);
|
|
_outNode(str, node->joinqual);
|
|
appendStringInfo(str, " :joinrti %d ",
|
|
node->joinrti);
|
|
}
|
|
|
|
/*
|
|
* NestLoop is a subclass of Join
|
|
*/
|
|
static void
|
|
_outNestLoop(StringInfo str, NestLoop *node)
|
|
{
|
|
appendStringInfo(str, " NESTLOOP ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
appendStringInfo(str, " :jointype %d :joinqual ",
|
|
(int) node->join.jointype);
|
|
_outNode(str, node->join.joinqual);
|
|
appendStringInfo(str, " :joinrti %d ",
|
|
node->join.joinrti);
|
|
}
|
|
|
|
/*
|
|
* MergeJoin is a subclass of Join
|
|
*/
|
|
static void
|
|
_outMergeJoin(StringInfo str, MergeJoin *node)
|
|
{
|
|
appendStringInfo(str, " MERGEJOIN ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
appendStringInfo(str, " :jointype %d :joinqual ",
|
|
(int) node->join.jointype);
|
|
_outNode(str, node->join.joinqual);
|
|
appendStringInfo(str, " :joinrti %d ",
|
|
node->join.joinrti);
|
|
|
|
appendStringInfo(str, " :mergeclauses ");
|
|
_outNode(str, node->mergeclauses);
|
|
}
|
|
|
|
/*
|
|
* HashJoin is a subclass of Join.
|
|
*/
|
|
static void
|
|
_outHashJoin(StringInfo str, HashJoin *node)
|
|
{
|
|
appendStringInfo(str, " HASHJOIN ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
appendStringInfo(str, " :jointype %d :joinqual ",
|
|
(int) node->join.jointype);
|
|
_outNode(str, node->join.joinqual);
|
|
appendStringInfo(str, " :joinrti %d ",
|
|
node->join.joinrti);
|
|
|
|
appendStringInfo(str, " :hashclauses ");
|
|
_outNode(str, node->hashclauses);
|
|
appendStringInfo(str, " :hashjoinop %u ",
|
|
node->hashjoinop);
|
|
}
|
|
|
|
static void
|
|
_outSubPlan(StringInfo str, SubPlan *node)
|
|
{
|
|
appendStringInfo(str, " SUBPLAN :plan ");
|
|
_outNode(str, node->plan);
|
|
|
|
appendStringInfo(str, " :planid %d :rtable ", node->plan_id);
|
|
_outNode(str, node->rtable);
|
|
|
|
appendStringInfo(str, " :setprm ");
|
|
_outIntList(str, node->setParam);
|
|
|
|
appendStringInfo(str, " :parprm ");
|
|
_outIntList(str, node->parParam);
|
|
|
|
appendStringInfo(str, " :slink ");
|
|
_outNode(str, node->sublink);
|
|
}
|
|
|
|
/*
|
|
* Scan is a subclass of Node
|
|
*/
|
|
static void
|
|
_outScan(StringInfo str, Scan *node)
|
|
{
|
|
appendStringInfo(str, " SCAN ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :scanrelid %u ", node->scanrelid);
|
|
}
|
|
|
|
/*
|
|
* SeqScan is a subclass of Scan
|
|
*/
|
|
static void
|
|
_outSeqScan(StringInfo str, SeqScan *node)
|
|
{
|
|
appendStringInfo(str, " SEQSCAN ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :scanrelid %u ", node->scanrelid);
|
|
}
|
|
|
|
/*
|
|
* IndexScan is a subclass of Scan
|
|
*/
|
|
static void
|
|
_outIndexScan(StringInfo str, IndexScan *node)
|
|
{
|
|
appendStringInfo(str, " INDEXSCAN ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :scanrelid %u :indxid ", node->scan.scanrelid);
|
|
_outOidList(str, node->indxid);
|
|
|
|
appendStringInfo(str, " :indxqual ");
|
|
_outNode(str, node->indxqual);
|
|
|
|
appendStringInfo(str, " :indxqualorig ");
|
|
_outNode(str, node->indxqualorig);
|
|
|
|
appendStringInfo(str, " :indxorderdir %d ", node->indxorderdir);
|
|
}
|
|
|
|
/*
|
|
* TidScan is a subclass of Scan
|
|
*/
|
|
static void
|
|
_outTidScan(StringInfo str, TidScan *node)
|
|
{
|
|
appendStringInfo(str, " TIDSCAN ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :scanrelid %u ", node->scan.scanrelid);
|
|
appendStringInfo(str, " :needrescan %d ", node->needRescan);
|
|
|
|
appendStringInfo(str, " :tideval ");
|
|
_outNode(str, node->tideval);
|
|
|
|
}
|
|
|
|
/*
|
|
* SubqueryScan is a subclass of Scan
|
|
*/
|
|
static void
|
|
_outSubqueryScan(StringInfo str, SubqueryScan *node)
|
|
{
|
|
appendStringInfo(str, " SUBQUERYSCAN ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :scanrelid %u :subplan ", node->scan.scanrelid);
|
|
_outNode(str, node->subplan);
|
|
}
|
|
|
|
/*
|
|
* Material is a subclass of Plan
|
|
*/
|
|
static void
|
|
_outMaterial(StringInfo str, Material *node)
|
|
{
|
|
appendStringInfo(str, " MATERIAL ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
}
|
|
|
|
/*
|
|
* Sort is a subclass of Plan
|
|
*/
|
|
static void
|
|
_outSort(StringInfo str, Sort *node)
|
|
{
|
|
appendStringInfo(str, " SORT ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
appendStringInfo(str, " :keycount %d ", node->keycount);
|
|
}
|
|
|
|
static void
|
|
_outAgg(StringInfo str, Agg *node)
|
|
{
|
|
appendStringInfo(str, " AGG ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
}
|
|
|
|
static void
|
|
_outGroup(StringInfo str, Group *node)
|
|
{
|
|
appendStringInfo(str, " GRP ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
/* the actual Group fields */
|
|
appendStringInfo(str, " :numCols %d :tuplePerGroup %s ",
|
|
node->numCols,
|
|
booltostr(node->tuplePerGroup));
|
|
}
|
|
|
|
static void
|
|
_outUnique(StringInfo str, Unique *node)
|
|
{
|
|
int i;
|
|
|
|
appendStringInfo(str, " UNIQUE ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :numCols %d :uniqColIdx ",
|
|
node->numCols);
|
|
for (i = 0; i < node->numCols; i++)
|
|
appendStringInfo(str, "%d ", (int) node->uniqColIdx[i]);
|
|
}
|
|
|
|
static void
|
|
_outSetOp(StringInfo str, SetOp *node)
|
|
{
|
|
int i;
|
|
|
|
appendStringInfo(str, " SETOP ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :cmd %d :numCols %d :dupColIdx ",
|
|
(int) node->cmd, node->numCols);
|
|
for (i = 0; i < node->numCols; i++)
|
|
appendStringInfo(str, "%d ", (int) node->dupColIdx[i]);
|
|
appendStringInfo(str, " :flagColIdx %d ",
|
|
(int) node->flagColIdx);
|
|
}
|
|
|
|
static void
|
|
_outLimit(StringInfo str, Limit *node)
|
|
{
|
|
appendStringInfo(str, " LIMIT ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :limitOffset ");
|
|
_outNode(str, node->limitOffset);
|
|
appendStringInfo(str, " :limitCount ");
|
|
_outNode(str, node->limitCount);
|
|
}
|
|
|
|
/*
|
|
* Hash is a subclass of Plan
|
|
*/
|
|
static void
|
|
_outHash(StringInfo str, Hash *node)
|
|
{
|
|
appendStringInfo(str, " HASH ");
|
|
_outPlanInfo(str, (Plan *) node);
|
|
|
|
appendStringInfo(str, " :hashkey ");
|
|
_outNode(str, node->hashkey);
|
|
}
|
|
|
|
/*****************************************************************************
|
|
*
|
|
* Stuff from primnodes.h.
|
|
*
|
|
*****************************************************************************/
|
|
|
|
/*
|
|
* Resdom is a subclass of Node
|
|
*/
|
|
static void
|
|
_outResdom(StringInfo str, Resdom *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" RESDOM :resno %d :restype %u :restypmod %d :resname ",
|
|
node->resno,
|
|
node->restype,
|
|
node->restypmod);
|
|
_outToken(str, node->resname);
|
|
appendStringInfo(str, " :reskey %u :reskeyop %u :ressortgroupref %u :resjunk %s ",
|
|
node->reskey,
|
|
node->reskeyop,
|
|
node->ressortgroupref,
|
|
booltostr(node->resjunk));
|
|
}
|
|
|
|
static void
|
|
_outFjoin(StringInfo str, Fjoin *node)
|
|
{
|
|
int i;
|
|
|
|
appendStringInfo(str, " FJOIN :initialized %s :nNodes %d ",
|
|
booltostr(node->fj_initialized),
|
|
node->fj_nNodes);
|
|
|
|
appendStringInfo(str, " :innerNode ");
|
|
_outNode(str, node->fj_innerNode);
|
|
|
|
appendStringInfo(str, " :results @ 0x%p :alwaysdone",
|
|
node->fj_results);
|
|
|
|
for (i = 0; i < node->fj_nNodes; i++)
|
|
appendStringInfo(str,
|
|
booltostr(node->fj_alwaysDone[i]));
|
|
}
|
|
|
|
/*
|
|
* Expr is a subclass of Node
|
|
*/
|
|
static void
|
|
_outExpr(StringInfo str, Expr *node)
|
|
{
|
|
char *opstr = NULL;
|
|
|
|
appendStringInfo(str, " EXPR :typeOid %u ",
|
|
node->typeOid);
|
|
|
|
switch (node->opType)
|
|
{
|
|
case OP_EXPR:
|
|
opstr = "op";
|
|
break;
|
|
case FUNC_EXPR:
|
|
opstr = "func";
|
|
break;
|
|
case OR_EXPR:
|
|
opstr = "or";
|
|
break;
|
|
case AND_EXPR:
|
|
opstr = "and";
|
|
break;
|
|
case NOT_EXPR:
|
|
opstr = "not";
|
|
break;
|
|
case SUBPLAN_EXPR:
|
|
opstr = "subp";
|
|
break;
|
|
}
|
|
appendStringInfo(str, " :opType ");
|
|
_outToken(str, opstr);
|
|
appendStringInfo(str, " :oper ");
|
|
_outNode(str, node->oper);
|
|
|
|
appendStringInfo(str, " :args ");
|
|
_outNode(str, node->args);
|
|
}
|
|
|
|
/*
|
|
* Var is a subclass of Expr
|
|
*/
|
|
static void
|
|
_outVar(StringInfo str, Var *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" VAR :varno %u :varattno %d :vartype %u :vartypmod %d ",
|
|
node->varno,
|
|
node->varattno,
|
|
node->vartype,
|
|
node->vartypmod);
|
|
|
|
appendStringInfo(str, " :varlevelsup %u :varnoold %u :varoattno %d",
|
|
node->varlevelsup,
|
|
node->varnoold,
|
|
node->varoattno);
|
|
}
|
|
|
|
/*
|
|
* Const is a subclass of Expr
|
|
*/
|
|
static void
|
|
_outConst(StringInfo str, Const *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" CONST :consttype %u :constlen %d :constbyval %s"
|
|
" :constisnull %s :constvalue ",
|
|
node->consttype,
|
|
node->constlen,
|
|
booltostr(node->constbyval),
|
|
booltostr(node->constisnull));
|
|
|
|
if (node->constisnull)
|
|
appendStringInfo(str, "<>");
|
|
else
|
|
_outDatum(str, node->constvalue, node->constlen, node->constbyval);
|
|
}
|
|
|
|
/*
|
|
* Aggref
|
|
*/
|
|
static void
|
|
_outAggref(StringInfo str, Aggref *node)
|
|
{
|
|
appendStringInfo(str, " AGGREG :aggfnoid %u :aggtype %u :target ",
|
|
node->aggfnoid, node->aggtype);
|
|
_outNode(str, node->target);
|
|
|
|
appendStringInfo(str, " :aggstar %s :aggdistinct %s ",
|
|
booltostr(node->aggstar),
|
|
booltostr(node->aggdistinct));
|
|
/* aggno is not dumped */
|
|
}
|
|
|
|
/*
|
|
* SubLink
|
|
*/
|
|
static void
|
|
_outSubLink(StringInfo str, SubLink *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" SUBLINK :subLinkType %d :useor %s :lefthand ",
|
|
node->subLinkType,
|
|
booltostr(node->useor));
|
|
_outNode(str, node->lefthand);
|
|
|
|
appendStringInfo(str, " :oper ");
|
|
_outNode(str, node->oper);
|
|
|
|
appendStringInfo(str, " :subselect ");
|
|
_outNode(str, node->subselect);
|
|
}
|
|
|
|
/*
|
|
* ArrayRef is a subclass of Expr
|
|
*/
|
|
static void
|
|
_outArrayRef(StringInfo str, ArrayRef *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" ARRAYREF :refelemtype %u :refattrlength %d :refelemlength %d ",
|
|
node->refelemtype,
|
|
node->refattrlength,
|
|
node->refelemlength);
|
|
|
|
appendStringInfo(str, " :refelembyval %c :refupperindex ",
|
|
node->refelembyval ? 't' : 'f');
|
|
_outNode(str, node->refupperindexpr);
|
|
|
|
appendStringInfo(str, " :reflowerindex ");
|
|
_outNode(str, node->reflowerindexpr);
|
|
|
|
appendStringInfo(str, " :refexpr ");
|
|
_outNode(str, node->refexpr);
|
|
|
|
appendStringInfo(str, " :refassgnexpr ");
|
|
_outNode(str, node->refassgnexpr);
|
|
}
|
|
|
|
/*
|
|
* Func is a subclass of Expr
|
|
*/
|
|
static void
|
|
_outFunc(StringInfo str, Func *node)
|
|
{
|
|
appendStringInfo(str, " FUNC :funcid %u :functype %u ",
|
|
node->funcid,
|
|
node->functype);
|
|
}
|
|
|
|
/*
|
|
* Oper is a subclass of Expr
|
|
*/
|
|
static void
|
|
_outOper(StringInfo str, Oper *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" OPER :opno %u :opid %u :opresulttype %u ",
|
|
node->opno,
|
|
node->opid,
|
|
node->opresulttype);
|
|
}
|
|
|
|
/*
|
|
* Param is a subclass of Expr
|
|
*/
|
|
static void
|
|
_outParam(StringInfo str, Param *node)
|
|
{
|
|
appendStringInfo(str, " PARAM :paramkind %d :paramid %d :paramname ",
|
|
node->paramkind,
|
|
node->paramid);
|
|
_outToken(str, node->paramname);
|
|
appendStringInfo(str, " :paramtype %u ", node->paramtype);
|
|
}
|
|
|
|
/*
|
|
* FieldSelect
|
|
*/
|
|
static void
|
|
_outFieldSelect(StringInfo str, FieldSelect *node)
|
|
{
|
|
appendStringInfo(str, " FIELDSELECT :arg ");
|
|
_outNode(str, node->arg);
|
|
|
|
appendStringInfo(str, " :fieldnum %d :resulttype %u :resulttypmod %d ",
|
|
node->fieldnum, node->resulttype, node->resulttypmod);
|
|
}
|
|
|
|
/*
|
|
* RelabelType
|
|
*/
|
|
static void
|
|
_outRelabelType(StringInfo str, RelabelType *node)
|
|
{
|
|
appendStringInfo(str, " RELABELTYPE :arg ");
|
|
_outNode(str, node->arg);
|
|
|
|
appendStringInfo(str, " :resulttype %u :resulttypmod %d ",
|
|
node->resulttype, node->resulttypmod);
|
|
}
|
|
|
|
/*
|
|
* RangeTblRef
|
|
*/
|
|
static void
|
|
_outRangeTblRef(StringInfo str, RangeTblRef *node)
|
|
{
|
|
appendStringInfo(str, " RANGETBLREF %d ",
|
|
node->rtindex);
|
|
}
|
|
|
|
/*
|
|
* FromExpr
|
|
*/
|
|
static void
|
|
_outFromExpr(StringInfo str, FromExpr *node)
|
|
{
|
|
appendStringInfo(str, " FROMEXPR :fromlist ");
|
|
_outNode(str, node->fromlist);
|
|
appendStringInfo(str, " :quals ");
|
|
_outNode(str, node->quals);
|
|
}
|
|
|
|
/*
|
|
* JoinExpr
|
|
*/
|
|
static void
|
|
_outJoinExpr(StringInfo str, JoinExpr *node)
|
|
{
|
|
appendStringInfo(str, " JOINEXPR :jointype %d :isNatural %s :larg ",
|
|
(int) node->jointype,
|
|
booltostr(node->isNatural));
|
|
_outNode(str, node->larg);
|
|
appendStringInfo(str, " :rarg ");
|
|
_outNode(str, node->rarg);
|
|
appendStringInfo(str, " :using ");
|
|
_outNode(str, node->using);
|
|
appendStringInfo(str, " :quals ");
|
|
_outNode(str, node->quals);
|
|
appendStringInfo(str, " :alias ");
|
|
_outNode(str, node->alias);
|
|
appendStringInfo(str, " :rtindex %d ", node->rtindex);
|
|
}
|
|
|
|
/*
|
|
* TargetEntry is a subclass of Node.
|
|
*/
|
|
static void
|
|
_outTargetEntry(StringInfo str, TargetEntry *node)
|
|
{
|
|
appendStringInfo(str, " TARGETENTRY :resdom ");
|
|
_outNode(str, node->resdom);
|
|
|
|
appendStringInfo(str, " :expr ");
|
|
_outNode(str, node->expr);
|
|
}
|
|
|
|
static void
|
|
_outAlias(StringInfo str, Alias *node)
|
|
{
|
|
appendStringInfo(str, " ALIAS :aliasname ");
|
|
_outToken(str, node->aliasname);
|
|
appendStringInfo(str, " :colnames ");
|
|
_outNode(str, node->colnames);
|
|
}
|
|
|
|
static void
|
|
_outRangeTblEntry(StringInfo str, RangeTblEntry *node)
|
|
{
|
|
/* put alias + eref first to make dump more legible */
|
|
appendStringInfo(str, " RTE :alias ");
|
|
_outNode(str, node->alias);
|
|
appendStringInfo(str, " :eref ");
|
|
_outNode(str, node->eref);
|
|
appendStringInfo(str, " :rtekind %d ",
|
|
(int) node->rtekind);
|
|
switch (node->rtekind)
|
|
{
|
|
case RTE_RELATION:
|
|
case RTE_SPECIAL:
|
|
appendStringInfo(str, ":relid %u ", node->relid);
|
|
break;
|
|
case RTE_SUBQUERY:
|
|
appendStringInfo(str, ":subquery ");
|
|
_outNode(str, node->subquery);
|
|
break;
|
|
case RTE_JOIN:
|
|
appendStringInfo(str, ":jointype %d :joincoltypes ",
|
|
(int) node->jointype);
|
|
_outOidList(str, node->joincoltypes);
|
|
appendStringInfo(str, " :joincoltypmods ");
|
|
_outIntList(str, node->joincoltypmods);
|
|
appendStringInfo(str, " :joinleftcols ");
|
|
_outIntList(str, node->joinleftcols);
|
|
appendStringInfo(str, " :joinrightcols ");
|
|
_outIntList(str, node->joinrightcols);
|
|
break;
|
|
default:
|
|
elog(ERROR, "bogus rte kind %d", (int) node->rtekind);
|
|
break;
|
|
}
|
|
appendStringInfo(str, " :inh %s :inFromCl %s :checkForRead %s"
|
|
" :checkForWrite %s :checkAsUser %u",
|
|
booltostr(node->inh),
|
|
booltostr(node->inFromCl),
|
|
booltostr(node->checkForRead),
|
|
booltostr(node->checkForWrite),
|
|
node->checkAsUser);
|
|
}
|
|
|
|
/*
|
|
* Path is a subclass of Node.
|
|
*/
|
|
static void
|
|
_outPath(StringInfo str, Path *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" PATH :pathtype %d :startup_cost %.2f :total_cost %.2f :pathkeys ",
|
|
node->pathtype,
|
|
node->startup_cost,
|
|
node->total_cost);
|
|
_outNode(str, node->pathkeys);
|
|
}
|
|
|
|
/*
|
|
* IndexPath is a subclass of Path.
|
|
*/
|
|
static void
|
|
_outIndexPath(StringInfo str, IndexPath *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" INDEXPATH :pathtype %d :startup_cost %.2f :total_cost %.2f :pathkeys ",
|
|
node->path.pathtype,
|
|
node->path.startup_cost,
|
|
node->path.total_cost);
|
|
_outNode(str, node->path.pathkeys);
|
|
|
|
appendStringInfo(str, " :indexinfo ");
|
|
_outNode(str, node->indexinfo);
|
|
|
|
appendStringInfo(str, " :indexqual ");
|
|
_outNode(str, node->indexqual);
|
|
|
|
appendStringInfo(str, " :indexscandir %d :joinrelids ",
|
|
(int) node->indexscandir);
|
|
_outIntList(str, node->joinrelids);
|
|
|
|
appendStringInfo(str, " :alljoinquals %s :rows %.2f ",
|
|
booltostr(node->alljoinquals),
|
|
node->rows);
|
|
}
|
|
|
|
/*
|
|
* TidPath is a subclass of Path.
|
|
*/
|
|
static void
|
|
_outTidPath(StringInfo str, TidPath *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" TIDPATH :pathtype %d :startup_cost %.2f :total_cost %.2f :pathkeys ",
|
|
node->path.pathtype,
|
|
node->path.startup_cost,
|
|
node->path.total_cost);
|
|
_outNode(str, node->path.pathkeys);
|
|
|
|
appendStringInfo(str, " :tideval ");
|
|
_outNode(str, node->tideval);
|
|
|
|
appendStringInfo(str, " :unjoined_relids ");
|
|
_outIntList(str, node->unjoined_relids);
|
|
}
|
|
|
|
/*
|
|
* AppendPath is a subclass of Path.
|
|
*/
|
|
static void
|
|
_outAppendPath(StringInfo str, AppendPath *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" APPENDPATH :pathtype %d :startup_cost %.2f :total_cost %.2f :pathkeys ",
|
|
node->path.pathtype,
|
|
node->path.startup_cost,
|
|
node->path.total_cost);
|
|
_outNode(str, node->path.pathkeys);
|
|
|
|
appendStringInfo(str, " :subpaths ");
|
|
_outNode(str, node->subpaths);
|
|
}
|
|
|
|
/*
|
|
* NestPath is a subclass of Path
|
|
*/
|
|
static void
|
|
_outNestPath(StringInfo str, NestPath *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" NESTPATH :pathtype %d :startup_cost %.2f :total_cost %.2f :pathkeys ",
|
|
node->path.pathtype,
|
|
node->path.startup_cost,
|
|
node->path.total_cost);
|
|
_outNode(str, node->path.pathkeys);
|
|
appendStringInfo(str, " :jointype %d :outerjoinpath ",
|
|
(int) node->jointype);
|
|
_outNode(str, node->outerjoinpath);
|
|
appendStringInfo(str, " :innerjoinpath ");
|
|
_outNode(str, node->innerjoinpath);
|
|
appendStringInfo(str, " :joinrestrictinfo ");
|
|
_outNode(str, node->joinrestrictinfo);
|
|
}
|
|
|
|
/*
|
|
* MergePath is a subclass of NestPath.
|
|
*/
|
|
static void
|
|
_outMergePath(StringInfo str, MergePath *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" MERGEPATH :pathtype %d :startup_cost %.2f :total_cost %.2f :pathkeys ",
|
|
node->jpath.path.pathtype,
|
|
node->jpath.path.startup_cost,
|
|
node->jpath.path.total_cost);
|
|
_outNode(str, node->jpath.path.pathkeys);
|
|
appendStringInfo(str, " :jointype %d :outerjoinpath ",
|
|
(int) node->jpath.jointype);
|
|
_outNode(str, node->jpath.outerjoinpath);
|
|
appendStringInfo(str, " :innerjoinpath ");
|
|
_outNode(str, node->jpath.innerjoinpath);
|
|
appendStringInfo(str, " :joinrestrictinfo ");
|
|
_outNode(str, node->jpath.joinrestrictinfo);
|
|
|
|
appendStringInfo(str, " :path_mergeclauses ");
|
|
_outNode(str, node->path_mergeclauses);
|
|
|
|
appendStringInfo(str, " :outersortkeys ");
|
|
_outNode(str, node->outersortkeys);
|
|
|
|
appendStringInfo(str, " :innersortkeys ");
|
|
_outNode(str, node->innersortkeys);
|
|
}
|
|
|
|
/*
|
|
* HashPath is a subclass of NestPath.
|
|
*/
|
|
static void
|
|
_outHashPath(StringInfo str, HashPath *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" HASHPATH :pathtype %d :startup_cost %.2f :total_cost %.2f :pathkeys ",
|
|
node->jpath.path.pathtype,
|
|
node->jpath.path.startup_cost,
|
|
node->jpath.path.total_cost);
|
|
_outNode(str, node->jpath.path.pathkeys);
|
|
appendStringInfo(str, " :jointype %d :outerjoinpath ",
|
|
(int) node->jpath.jointype);
|
|
_outNode(str, node->jpath.outerjoinpath);
|
|
appendStringInfo(str, " :innerjoinpath ");
|
|
_outNode(str, node->jpath.innerjoinpath);
|
|
appendStringInfo(str, " :joinrestrictinfo ");
|
|
_outNode(str, node->jpath.joinrestrictinfo);
|
|
|
|
appendStringInfo(str, " :path_hashclauses ");
|
|
_outNode(str, node->path_hashclauses);
|
|
}
|
|
|
|
/*
|
|
* PathKeyItem is a subclass of Node.
|
|
*/
|
|
static void
|
|
_outPathKeyItem(StringInfo str, PathKeyItem *node)
|
|
{
|
|
appendStringInfo(str, " PATHKEYITEM :sortop %u :key ",
|
|
node->sortop);
|
|
_outNode(str, node->key);
|
|
}
|
|
|
|
/*
|
|
* RestrictInfo is a subclass of Node.
|
|
*/
|
|
static void
|
|
_outRestrictInfo(StringInfo str, RestrictInfo *node)
|
|
{
|
|
appendStringInfo(str, " RESTRICTINFO :clause ");
|
|
_outNode(str, node->clause);
|
|
|
|
appendStringInfo(str, " :ispusheddown %s :subclauseindices ",
|
|
booltostr(node->ispusheddown));
|
|
_outNode(str, node->subclauseindices);
|
|
|
|
appendStringInfo(str, " :mergejoinoperator %u ", node->mergejoinoperator);
|
|
appendStringInfo(str, " :left_sortop %u ", node->left_sortop);
|
|
appendStringInfo(str, " :right_sortop %u ", node->right_sortop);
|
|
appendStringInfo(str, " :hashjoinoperator %u ", node->hashjoinoperator);
|
|
}
|
|
|
|
/*
|
|
* JoinInfo is a subclass of Node.
|
|
*/
|
|
static void
|
|
_outJoinInfo(StringInfo str, JoinInfo *node)
|
|
{
|
|
appendStringInfo(str, " JINFO :unjoined_relids ");
|
|
_outIntList(str, node->unjoined_relids);
|
|
|
|
appendStringInfo(str, " :jinfo_restrictinfo ");
|
|
_outNode(str, node->jinfo_restrictinfo);
|
|
}
|
|
|
|
/*
|
|
* Print the value of a Datum given its type.
|
|
*/
|
|
static void
|
|
_outDatum(StringInfo str, Datum value, int typlen, bool typbyval)
|
|
{
|
|
Size length,
|
|
i;
|
|
char *s;
|
|
|
|
length = datumGetSize(value, typbyval, typlen);
|
|
|
|
if (typbyval)
|
|
{
|
|
s = (char *) (&value);
|
|
appendStringInfo(str, " %u [ ", (unsigned int) length);
|
|
for (i = 0; i < (Size) sizeof(Datum); i++)
|
|
appendStringInfo(str, "%d ", (int) (s[i]));
|
|
appendStringInfo(str, "] ");
|
|
}
|
|
else
|
|
{
|
|
s = (char *) DatumGetPointer(value);
|
|
if (!PointerIsValid(s))
|
|
appendStringInfo(str, " 0 [ ] ");
|
|
else
|
|
{
|
|
appendStringInfo(str, " %u [ ", (unsigned int) length);
|
|
for (i = 0; i < length; i++)
|
|
appendStringInfo(str, "%d ", (int) (s[i]));
|
|
appendStringInfo(str, "] ");
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
_outIter(StringInfo str, Iter *node)
|
|
{
|
|
appendStringInfo(str, " ITER :iterexpr ");
|
|
_outNode(str, node->iterexpr);
|
|
}
|
|
|
|
static void
|
|
_outStream(StringInfo str, Stream *node)
|
|
{
|
|
appendStringInfo(str,
|
|
" STREAM :pathptr @ %p :cinfo @ %p :clausetype %p :upstream @ %p ",
|
|
node->pathptr,
|
|
node->cinfo,
|
|
node->clausetype,
|
|
node->upstream);
|
|
|
|
appendStringInfo(str,
|
|
" :downstream @ %p :groupup %d :groupcost %f :groupsel %f ",
|
|
node->downstream,
|
|
node->groupup,
|
|
node->groupcost,
|
|
node->groupsel);
|
|
}
|
|
|
|
static void
|
|
_outAExpr(StringInfo str, A_Expr *node)
|
|
{
|
|
appendStringInfo(str, " AEXPR ");
|
|
switch (node->oper)
|
|
{
|
|
case AND:
|
|
appendStringInfo(str, "AND ");
|
|
break;
|
|
case OR:
|
|
appendStringInfo(str, "OR ");
|
|
break;
|
|
case NOT:
|
|
appendStringInfo(str, "NOT ");
|
|
break;
|
|
case OP:
|
|
_outNode(str, node->name);
|
|
appendStringInfo(str, " ");
|
|
break;
|
|
default:
|
|
appendStringInfo(str, "?? ");
|
|
break;
|
|
}
|
|
_outNode(str, node->lexpr);
|
|
appendStringInfo(str, " ");
|
|
_outNode(str, node->rexpr);
|
|
}
|
|
|
|
static void
|
|
_outValue(StringInfo str, Value *value)
|
|
{
|
|
switch (value->type)
|
|
{
|
|
case T_Integer:
|
|
appendStringInfo(str, " %ld ", value->val.ival);
|
|
break;
|
|
case T_Float:
|
|
|
|
/*
|
|
* We assume the value is a valid numeric literal and so does
|
|
* not need quoting.
|
|
*/
|
|
appendStringInfo(str, " %s ", value->val.str);
|
|
break;
|
|
case T_String:
|
|
appendStringInfo(str, " \"");
|
|
_outToken(str, value->val.str);
|
|
appendStringInfo(str, "\" ");
|
|
break;
|
|
case T_BitString:
|
|
/* internal representation already has leading 'b' */
|
|
appendStringInfo(str, " %s ", value->val.str);
|
|
break;
|
|
default:
|
|
elog(WARNING, "_outValue: don't know how to print type %d ",
|
|
value->type);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
_outRangeVar(StringInfo str, RangeVar *node)
|
|
{
|
|
appendStringInfo(str, " RANGEVAR :relation ");
|
|
/*
|
|
* we deliberately ignore catalogname here, since it is presently not
|
|
* semantically meaningful
|
|
*/
|
|
_outToken(str, node->schemaname);
|
|
appendStringInfo(str, " . ");
|
|
_outToken(str, node->relname);
|
|
appendStringInfo(str, " :inhopt %d :istemp %s",
|
|
(int) node->inhOpt,
|
|
booltostr(node->istemp));
|
|
appendStringInfo(str, " :alias ");
|
|
_outNode(str, node->alias);
|
|
}
|
|
|
|
static void
|
|
_outColumnRef(StringInfo str, ColumnRef *node)
|
|
{
|
|
appendStringInfo(str, " COLUMNREF :fields ");
|
|
_outNode(str, node->fields);
|
|
appendStringInfo(str, " :indirection ");
|
|
_outNode(str, node->indirection);
|
|
}
|
|
|
|
static void
|
|
_outParamRef(StringInfo str, ParamRef *node)
|
|
{
|
|
appendStringInfo(str, " PARAMREF :number %d :fields ", node->number);
|
|
_outNode(str, node->fields);
|
|
appendStringInfo(str, " :indirection ");
|
|
_outNode(str, node->indirection);
|
|
}
|
|
|
|
static void
|
|
_outIdent(StringInfo str, Ident *node)
|
|
{
|
|
appendStringInfo(str, " IDENT ");
|
|
_outToken(str, node->name);
|
|
}
|
|
|
|
static void
|
|
_outAConst(StringInfo str, A_Const *node)
|
|
{
|
|
appendStringInfo(str, "CONST ");
|
|
_outValue(str, &(node->val));
|
|
appendStringInfo(str, " :typename ");
|
|
_outNode(str, node->typename);
|
|
}
|
|
|
|
static void
|
|
_outExprFieldSelect(StringInfo str, ExprFieldSelect *node)
|
|
{
|
|
appendStringInfo(str, " EXPRFIELDSELECT :arg ");
|
|
_outNode(str, node->arg);
|
|
appendStringInfo(str, " :fields ");
|
|
_outNode(str, node->fields);
|
|
appendStringInfo(str, " :indirection ");
|
|
_outNode(str, node->indirection);
|
|
}
|
|
|
|
static void
|
|
_outConstraint(StringInfo str, Constraint *node)
|
|
{
|
|
appendStringInfo(str, " CONSTRAINT :name ");
|
|
_outToken(str, node->name);
|
|
appendStringInfo(str, " :type ");
|
|
|
|
switch (node->contype)
|
|
{
|
|
case CONSTR_PRIMARY:
|
|
appendStringInfo(str, "PRIMARY_KEY :keys ");
|
|
_outNode(str, node->keys);
|
|
break;
|
|
|
|
case CONSTR_CHECK:
|
|
appendStringInfo(str, "CHECK :raw ");
|
|
_outNode(str, node->raw_expr);
|
|
appendStringInfo(str, " :cooked ");
|
|
_outToken(str, node->cooked_expr);
|
|
break;
|
|
|
|
case CONSTR_DEFAULT:
|
|
appendStringInfo(str, "DEFAULT :raw ");
|
|
_outNode(str, node->raw_expr);
|
|
appendStringInfo(str, " :cooked ");
|
|
_outToken(str, node->cooked_expr);
|
|
break;
|
|
|
|
case CONSTR_NOTNULL:
|
|
appendStringInfo(str, "NOT_NULL");
|
|
break;
|
|
|
|
case CONSTR_UNIQUE:
|
|
appendStringInfo(str, "UNIQUE :keys ");
|
|
_outNode(str, node->keys);
|
|
break;
|
|
|
|
default:
|
|
appendStringInfo(str, "<unrecognized_constraint>");
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
_outFkConstraint(StringInfo str, FkConstraint *node)
|
|
{
|
|
appendStringInfo(str, " FKCONSTRAINT :constr_name ");
|
|
_outToken(str, node->constr_name);
|
|
appendStringInfo(str, " :pktable ");
|
|
_outNode(str, node->pktable);
|
|
appendStringInfo(str, " :fk_attrs ");
|
|
_outNode(str, node->fk_attrs);
|
|
appendStringInfo(str, " :pk_attrs ");
|
|
_outNode(str, node->pk_attrs);
|
|
appendStringInfo(str, " :match_type ");
|
|
_outToken(str, node->match_type);
|
|
appendStringInfo(str, " :actions %d :deferrable %s :initdeferred %s",
|
|
node->actions,
|
|
booltostr(node->deferrable),
|
|
booltostr(node->initdeferred));
|
|
}
|
|
|
|
static void
|
|
_outCaseExpr(StringInfo str, CaseExpr *node)
|
|
{
|
|
appendStringInfo(str, " CASE :casetype %u :arg ",
|
|
node->casetype);
|
|
_outNode(str, node->arg);
|
|
|
|
appendStringInfo(str, " :args ");
|
|
_outNode(str, node->args);
|
|
|
|
appendStringInfo(str, " :defresult ");
|
|
_outNode(str, node->defresult);
|
|
}
|
|
|
|
static void
|
|
_outCaseWhen(StringInfo str, CaseWhen *node)
|
|
{
|
|
appendStringInfo(str, " WHEN ");
|
|
_outNode(str, node->expr);
|
|
|
|
appendStringInfo(str, " :then ");
|
|
_outNode(str, node->result);
|
|
}
|
|
|
|
/*
|
|
* NullTest
|
|
*/
|
|
static void
|
|
_outNullTest(StringInfo str, NullTest *node)
|
|
{
|
|
appendStringInfo(str, " NULLTEST :arg ");
|
|
_outNode(str, node->arg);
|
|
|
|
appendStringInfo(str, " :nulltesttype %d ",
|
|
(int) node->nulltesttype);
|
|
}
|
|
|
|
/*
|
|
* BooleanTest
|
|
*/
|
|
static void
|
|
_outBooleanTest(StringInfo str, BooleanTest *node)
|
|
{
|
|
appendStringInfo(str, " BOOLEANTEST :arg ");
|
|
_outNode(str, node->arg);
|
|
|
|
appendStringInfo(str, " :booltesttype %d ",
|
|
(int) node->booltesttype);
|
|
}
|
|
|
|
/*
|
|
* _outNode -
|
|
* converts a Node into ascii string and append it to 'str'
|
|
*/
|
|
static void
|
|
_outNode(StringInfo str, void *obj)
|
|
{
|
|
if (obj == NULL)
|
|
{
|
|
appendStringInfo(str, "<>");
|
|
return;
|
|
}
|
|
|
|
if (IsA(obj, List))
|
|
{
|
|
List *l;
|
|
|
|
appendStringInfoChar(str, '(');
|
|
foreach(l, (List *) obj)
|
|
{
|
|
_outNode(str, lfirst(l));
|
|
if (lnext(l))
|
|
appendStringInfoChar(str, ' ');
|
|
}
|
|
appendStringInfoChar(str, ')');
|
|
}
|
|
else if (IsA(obj, Integer) || IsA(obj, Float) || IsA(obj, String) || IsA(obj, BitString))
|
|
{
|
|
/* nodeRead does not want to see { } around these! */
|
|
_outValue(str, obj);
|
|
}
|
|
else
|
|
{
|
|
appendStringInfoChar(str, '{');
|
|
switch (nodeTag(obj))
|
|
{
|
|
case T_CreateStmt:
|
|
_outCreateStmt(str, obj);
|
|
break;
|
|
case T_IndexStmt:
|
|
_outIndexStmt(str, obj);
|
|
break;
|
|
case T_NotifyStmt:
|
|
_outNotifyStmt(str, obj);
|
|
break;
|
|
case T_SelectStmt:
|
|
_outSelectStmt(str, obj);
|
|
break;
|
|
case T_ColumnDef:
|
|
_outColumnDef(str, obj);
|
|
break;
|
|
case T_TypeName:
|
|
_outTypeName(str, obj);
|
|
break;
|
|
case T_TypeCast:
|
|
_outTypeCast(str, obj);
|
|
break;
|
|
case T_IndexElem:
|
|
_outIndexElem(str, obj);
|
|
break;
|
|
case T_Query:
|
|
_outQuery(str, obj);
|
|
break;
|
|
case T_SortClause:
|
|
_outSortClause(str, obj);
|
|
break;
|
|
case T_GroupClause:
|
|
_outGroupClause(str, obj);
|
|
break;
|
|
case T_SetOperationStmt:
|
|
_outSetOperationStmt(str, obj);
|
|
break;
|
|
case T_Plan:
|
|
_outPlan(str, obj);
|
|
break;
|
|
case T_Result:
|
|
_outResult(str, obj);
|
|
break;
|
|
case T_Append:
|
|
_outAppend(str, obj);
|
|
break;
|
|
case T_Join:
|
|
_outJoin(str, obj);
|
|
break;
|
|
case T_NestLoop:
|
|
_outNestLoop(str, obj);
|
|
break;
|
|
case T_MergeJoin:
|
|
_outMergeJoin(str, obj);
|
|
break;
|
|
case T_HashJoin:
|
|
_outHashJoin(str, obj);
|
|
break;
|
|
case T_Scan:
|
|
_outScan(str, obj);
|
|
break;
|
|
case T_SeqScan:
|
|
_outSeqScan(str, obj);
|
|
break;
|
|
case T_IndexScan:
|
|
_outIndexScan(str, obj);
|
|
break;
|
|
case T_TidScan:
|
|
_outTidScan(str, obj);
|
|
break;
|
|
case T_SubqueryScan:
|
|
_outSubqueryScan(str, obj);
|
|
break;
|
|
case T_Material:
|
|
_outMaterial(str, obj);
|
|
break;
|
|
case T_Sort:
|
|
_outSort(str, obj);
|
|
break;
|
|
case T_Agg:
|
|
_outAgg(str, obj);
|
|
break;
|
|
case T_Group:
|
|
_outGroup(str, obj);
|
|
break;
|
|
case T_Unique:
|
|
_outUnique(str, obj);
|
|
break;
|
|
case T_SetOp:
|
|
_outSetOp(str, obj);
|
|
break;
|
|
case T_Limit:
|
|
_outLimit(str, obj);
|
|
break;
|
|
case T_Hash:
|
|
_outHash(str, obj);
|
|
break;
|
|
case T_SubPlan:
|
|
_outSubPlan(str, obj);
|
|
break;
|
|
case T_Resdom:
|
|
_outResdom(str, obj);
|
|
break;
|
|
case T_Fjoin:
|
|
_outFjoin(str, obj);
|
|
break;
|
|
case T_Expr:
|
|
_outExpr(str, obj);
|
|
break;
|
|
case T_Var:
|
|
_outVar(str, obj);
|
|
break;
|
|
case T_Const:
|
|
_outConst(str, obj);
|
|
break;
|
|
case T_Aggref:
|
|
_outAggref(str, obj);
|
|
break;
|
|
case T_SubLink:
|
|
_outSubLink(str, obj);
|
|
break;
|
|
case T_ArrayRef:
|
|
_outArrayRef(str, obj);
|
|
break;
|
|
case T_Func:
|
|
_outFunc(str, obj);
|
|
break;
|
|
case T_Oper:
|
|
_outOper(str, obj);
|
|
break;
|
|
case T_Param:
|
|
_outParam(str, obj);
|
|
break;
|
|
case T_FieldSelect:
|
|
_outFieldSelect(str, obj);
|
|
break;
|
|
case T_RelabelType:
|
|
_outRelabelType(str, obj);
|
|
break;
|
|
case T_RangeTblRef:
|
|
_outRangeTblRef(str, obj);
|
|
break;
|
|
case T_FromExpr:
|
|
_outFromExpr(str, obj);
|
|
break;
|
|
case T_JoinExpr:
|
|
_outJoinExpr(str, obj);
|
|
break;
|
|
case T_TargetEntry:
|
|
_outTargetEntry(str, obj);
|
|
break;
|
|
case T_Alias:
|
|
_outAlias(str, obj);
|
|
break;
|
|
case T_RangeTblEntry:
|
|
_outRangeTblEntry(str, obj);
|
|
break;
|
|
case T_Path:
|
|
_outPath(str, obj);
|
|
break;
|
|
case T_IndexPath:
|
|
_outIndexPath(str, obj);
|
|
break;
|
|
case T_TidPath:
|
|
_outTidPath(str, obj);
|
|
break;
|
|
case T_AppendPath:
|
|
_outAppendPath(str, obj);
|
|
break;
|
|
case T_NestPath:
|
|
_outNestPath(str, obj);
|
|
break;
|
|
case T_MergePath:
|
|
_outMergePath(str, obj);
|
|
break;
|
|
case T_HashPath:
|
|
_outHashPath(str, obj);
|
|
break;
|
|
case T_PathKeyItem:
|
|
_outPathKeyItem(str, obj);
|
|
break;
|
|
case T_RestrictInfo:
|
|
_outRestrictInfo(str, obj);
|
|
break;
|
|
case T_JoinInfo:
|
|
_outJoinInfo(str, obj);
|
|
break;
|
|
case T_Iter:
|
|
_outIter(str, obj);
|
|
break;
|
|
case T_Stream:
|
|
_outStream(str, obj);
|
|
break;
|
|
case T_A_Expr:
|
|
_outAExpr(str, obj);
|
|
break;
|
|
case T_RangeVar:
|
|
_outRangeVar(str, obj);
|
|
break;
|
|
case T_ColumnRef:
|
|
_outColumnRef(str, obj);
|
|
break;
|
|
case T_ParamRef:
|
|
_outParamRef(str, obj);
|
|
break;
|
|
case T_Ident:
|
|
_outIdent(str, obj);
|
|
break;
|
|
case T_A_Const:
|
|
_outAConst(str, obj);
|
|
break;
|
|
case T_ExprFieldSelect:
|
|
_outExprFieldSelect(str, obj);
|
|
break;
|
|
case T_Constraint:
|
|
_outConstraint(str, obj);
|
|
break;
|
|
case T_FkConstraint:
|
|
_outFkConstraint(str, obj);
|
|
break;
|
|
case T_CaseExpr:
|
|
_outCaseExpr(str, obj);
|
|
break;
|
|
case T_CaseWhen:
|
|
_outCaseWhen(str, obj);
|
|
break;
|
|
case T_NullTest:
|
|
_outNullTest(str, obj);
|
|
break;
|
|
case T_BooleanTest:
|
|
_outBooleanTest(str, obj);
|
|
break;
|
|
case T_FuncCall:
|
|
_outFuncCall(str, obj);
|
|
break;
|
|
|
|
default:
|
|
elog(WARNING, "_outNode: don't know how to print type %d ",
|
|
nodeTag(obj));
|
|
break;
|
|
}
|
|
appendStringInfoChar(str, '}');
|
|
}
|
|
}
|
|
|
|
/*
|
|
* nodeToString -
|
|
* returns the ascii representation of the Node as a palloc'd string
|
|
*/
|
|
char *
|
|
nodeToString(void *obj)
|
|
{
|
|
StringInfoData str;
|
|
|
|
/* see stringinfo.h for an explanation of this maneuver */
|
|
initStringInfo(&str);
|
|
_outNode(&str, obj);
|
|
return str.data;
|
|
}
|