LCOV - code coverage report
Current view: top level - src/backend/utils/adt - ruleutils.c (source / functions) Coverage Total Hit
Test: Code coverage Lines: 0.0 % 6650 0
Test Date: 2026-01-26 10:56:24 Functions: 0.0 % 170 0
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
Branches: 0.0 % 4178 0

             Branch data     Line data    Source code
       1                 :             : /*-------------------------------------------------------------------------
       2                 :             :  *
       3                 :             :  * ruleutils.c
       4                 :             :  *        Functions to convert stored expressions/querytrees back to
       5                 :             :  *        source text
       6                 :             :  *
       7                 :             :  * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
       8                 :             :  * Portions Copyright (c) 1994, Regents of the University of California
       9                 :             :  *
      10                 :             :  *
      11                 :             :  * IDENTIFICATION
      12                 :             :  *        src/backend/utils/adt/ruleutils.c
      13                 :             :  *
      14                 :             :  *-------------------------------------------------------------------------
      15                 :             :  */
      16                 :             : #include "postgres.h"
      17                 :             : 
      18                 :             : #include <ctype.h>
      19                 :             : #include <unistd.h>
      20                 :             : #include <fcntl.h>
      21                 :             : 
      22                 :             : #include "access/amapi.h"
      23                 :             : #include "access/htup_details.h"
      24                 :             : #include "access/relation.h"
      25                 :             : #include "access/table.h"
      26                 :             : #include "catalog/pg_aggregate.h"
      27                 :             : #include "catalog/pg_am.h"
      28                 :             : #include "catalog/pg_authid.h"
      29                 :             : #include "catalog/pg_collation.h"
      30                 :             : #include "catalog/pg_constraint.h"
      31                 :             : #include "catalog/pg_depend.h"
      32                 :             : #include "catalog/pg_language.h"
      33                 :             : #include "catalog/pg_opclass.h"
      34                 :             : #include "catalog/pg_operator.h"
      35                 :             : #include "catalog/pg_partitioned_table.h"
      36                 :             : #include "catalog/pg_proc.h"
      37                 :             : #include "catalog/pg_statistic_ext.h"
      38                 :             : #include "catalog/pg_trigger.h"
      39                 :             : #include "catalog/pg_type.h"
      40                 :             : #include "commands/defrem.h"
      41                 :             : #include "commands/tablespace.h"
      42                 :             : #include "common/keywords.h"
      43                 :             : #include "executor/spi.h"
      44                 :             : #include "funcapi.h"
      45                 :             : #include "mb/pg_wchar.h"
      46                 :             : #include "miscadmin.h"
      47                 :             : #include "nodes/makefuncs.h"
      48                 :             : #include "nodes/nodeFuncs.h"
      49                 :             : #include "nodes/pathnodes.h"
      50                 :             : #include "optimizer/optimizer.h"
      51                 :             : #include "parser/parse_agg.h"
      52                 :             : #include "parser/parse_func.h"
      53                 :             : #include "parser/parse_oper.h"
      54                 :             : #include "parser/parse_relation.h"
      55                 :             : #include "parser/parser.h"
      56                 :             : #include "parser/parsetree.h"
      57                 :             : #include "rewrite/rewriteHandler.h"
      58                 :             : #include "rewrite/rewriteManip.h"
      59                 :             : #include "rewrite/rewriteSupport.h"
      60                 :             : #include "utils/array.h"
      61                 :             : #include "utils/builtins.h"
      62                 :             : #include "utils/fmgroids.h"
      63                 :             : #include "utils/guc.h"
      64                 :             : #include "utils/hsearch.h"
      65                 :             : #include "utils/lsyscache.h"
      66                 :             : #include "utils/partcache.h"
      67                 :             : #include "utils/rel.h"
      68                 :             : #include "utils/ruleutils.h"
      69                 :             : #include "utils/snapmgr.h"
      70                 :             : #include "utils/syscache.h"
      71                 :             : #include "utils/typcache.h"
      72                 :             : #include "utils/varlena.h"
      73                 :             : #include "utils/xml.h"
      74                 :             : 
      75                 :             : /* ----------
      76                 :             :  * Pretty formatting constants
      77                 :             :  * ----------
      78                 :             :  */
      79                 :             : 
      80                 :             : /* Indent counts */
      81                 :             : #define PRETTYINDENT_STD                8
      82                 :             : #define PRETTYINDENT_JOIN               4
      83                 :             : #define PRETTYINDENT_VAR                4
      84                 :             : 
      85                 :             : #define PRETTYINDENT_LIMIT              40      /* wrap limit */
      86                 :             : 
      87                 :             : /* Pretty flags */
      88                 :             : #define PRETTYFLAG_PAREN                0x0001
      89                 :             : #define PRETTYFLAG_INDENT               0x0002
      90                 :             : #define PRETTYFLAG_SCHEMA               0x0004
      91                 :             : 
      92                 :             : /* Standard conversion of a "bool pretty" option to detailed flags */
      93                 :             : #define GET_PRETTY_FLAGS(pretty) \
      94                 :             :         ((pretty) ? (PRETTYFLAG_PAREN | PRETTYFLAG_INDENT | PRETTYFLAG_SCHEMA) \
      95                 :             :          : PRETTYFLAG_INDENT)
      96                 :             : 
      97                 :             : /* Default line length for pretty-print wrapping: 0 means wrap always */
      98                 :             : #define WRAP_COLUMN_DEFAULT             0
      99                 :             : 
     100                 :             : /* macros to test if pretty action needed */
     101                 :             : #define PRETTY_PAREN(context)   ((context)->prettyFlags & PRETTYFLAG_PAREN)
     102                 :             : #define PRETTY_INDENT(context)  ((context)->prettyFlags & PRETTYFLAG_INDENT)
     103                 :             : #define PRETTY_SCHEMA(context)  ((context)->prettyFlags & PRETTYFLAG_SCHEMA)
     104                 :             : 
     105                 :             : 
     106                 :             : /* ----------
     107                 :             :  * Local data types
     108                 :             :  * ----------
     109                 :             :  */
     110                 :             : 
     111                 :             : /* Context info needed for invoking a recursive querytree display routine */
     112                 :             : typedef struct
     113                 :             : {
     114                 :             :         StringInfo      buf;                    /* output buffer to append to */
     115                 :             :         List       *namespaces;         /* List of deparse_namespace nodes */
     116                 :             :         TupleDesc       resultDesc;             /* if top level of a view, the view's tupdesc */
     117                 :             :         List       *targetList;         /* Current query level's SELECT targetlist */
     118                 :             :         List       *windowClause;       /* Current query level's WINDOW clause */
     119                 :             :         int                     prettyFlags;    /* enabling of pretty-print functions */
     120                 :             :         int                     wrapColumn;             /* max line length, or -1 for no limit */
     121                 :             :         int                     indentLevel;    /* current indent level for pretty-print */
     122                 :             :         bool            varprefix;              /* true to print prefixes on Vars */
     123                 :             :         bool            colNamesVisible;        /* do we care about output column names? */
     124                 :             :         bool            inGroupBy;              /* deparsing GROUP BY clause? */
     125                 :             :         bool            varInOrderBy;   /* deparsing simple Var in ORDER BY? */
     126                 :             :         Bitmapset  *appendparents;      /* if not null, map child Vars of these relids
     127                 :             :                                                                  * back to the parent rel */
     128                 :             : } deparse_context;
     129                 :             : 
     130                 :             : /*
     131                 :             :  * Each level of query context around a subtree needs a level of Var namespace.
     132                 :             :  * A Var having varlevelsup=N refers to the N'th item (counting from 0) in
     133                 :             :  * the current context's namespaces list.
     134                 :             :  *
     135                 :             :  * rtable is the list of actual RTEs from the Query or PlannedStmt.
     136                 :             :  * rtable_names holds the alias name to be used for each RTE (either a C
     137                 :             :  * string, or NULL for nameless RTEs such as unnamed joins).
     138                 :             :  * rtable_columns holds the column alias names to be used for each RTE.
     139                 :             :  *
     140                 :             :  * subplans is a list of Plan trees for SubPlans and CTEs (it's only used
     141                 :             :  * in the PlannedStmt case).
     142                 :             :  * ctes is a list of CommonTableExpr nodes (only used in the Query case).
     143                 :             :  * appendrels, if not null (it's only used in the PlannedStmt case), is an
     144                 :             :  * array of AppendRelInfo nodes, indexed by child relid.  We use that to map
     145                 :             :  * child-table Vars to their inheritance parents.
     146                 :             :  *
     147                 :             :  * In some cases we need to make names of merged JOIN USING columns unique
     148                 :             :  * across the whole query, not only per-RTE.  If so, unique_using is true
     149                 :             :  * and using_names is a list of C strings representing names already assigned
     150                 :             :  * to USING columns.
     151                 :             :  *
     152                 :             :  * When deparsing plan trees, there is always just a single item in the
     153                 :             :  * deparse_namespace list (since a plan tree never contains Vars with
     154                 :             :  * varlevelsup > 0).  We store the Plan node that is the immediate
     155                 :             :  * parent of the expression to be deparsed, as well as a list of that
     156                 :             :  * Plan's ancestors.  In addition, we store its outer and inner subplan nodes,
     157                 :             :  * as well as their targetlists, and the index tlist if the current plan node
     158                 :             :  * might contain INDEX_VAR Vars.  (These fields could be derived on-the-fly
     159                 :             :  * from the current Plan node, but it seems notationally clearer to set them
     160                 :             :  * up as separate fields.)
     161                 :             :  */
     162                 :             : typedef struct
     163                 :             : {
     164                 :             :         List       *rtable;                     /* List of RangeTblEntry nodes */
     165                 :             :         List       *rtable_names;       /* Parallel list of names for RTEs */
     166                 :             :         List       *rtable_columns; /* Parallel list of deparse_columns structs */
     167                 :             :         List       *subplans;           /* List of Plan trees for SubPlans */
     168                 :             :         List       *ctes;                       /* List of CommonTableExpr nodes */
     169                 :             :         AppendRelInfo **appendrels; /* Array of AppendRelInfo nodes, or NULL */
     170                 :             :         char       *ret_old_alias;      /* alias for OLD in RETURNING list */
     171                 :             :         char       *ret_new_alias;      /* alias for NEW in RETURNING list */
     172                 :             :         /* Workspace for column alias assignment: */
     173                 :             :         bool            unique_using;   /* Are we making USING names globally unique */
     174                 :             :         List       *using_names;        /* List of assigned names for USING columns */
     175                 :             :         /* Remaining fields are used only when deparsing a Plan tree: */
     176                 :             :         Plan       *plan;                       /* immediate parent of current expression */
     177                 :             :         List       *ancestors;          /* ancestors of plan */
     178                 :             :         Plan       *outer_plan;         /* outer subnode, or NULL if none */
     179                 :             :         Plan       *inner_plan;         /* inner subnode, or NULL if none */
     180                 :             :         List       *outer_tlist;        /* referent for OUTER_VAR Vars */
     181                 :             :         List       *inner_tlist;        /* referent for INNER_VAR Vars */
     182                 :             :         List       *index_tlist;        /* referent for INDEX_VAR Vars */
     183                 :             :         /* Special namespace representing a function signature: */
     184                 :             :         char       *funcname;
     185                 :             :         int                     numargs;
     186                 :             :         char      **argnames;
     187                 :             : } deparse_namespace;
     188                 :             : 
     189                 :             : /*
     190                 :             :  * Per-relation data about column alias names.
     191                 :             :  *
     192                 :             :  * Selecting aliases is unreasonably complicated because of the need to dump
     193                 :             :  * rules/views whose underlying tables may have had columns added, deleted, or
     194                 :             :  * renamed since the query was parsed.  We must nonetheless print the rule/view
     195                 :             :  * in a form that can be reloaded and will produce the same results as before.
     196                 :             :  *
     197                 :             :  * For each RTE used in the query, we must assign column aliases that are
     198                 :             :  * unique within that RTE.  SQL does not require this of the original query,
     199                 :             :  * but due to factors such as *-expansion we need to be able to uniquely
     200                 :             :  * reference every column in a decompiled query.  As long as we qualify all
     201                 :             :  * column references, per-RTE uniqueness is sufficient for that.
     202                 :             :  *
     203                 :             :  * However, we can't ensure per-column name uniqueness for unnamed join RTEs,
     204                 :             :  * since they just inherit column names from their input RTEs, and we can't
     205                 :             :  * rename the columns at the join level.  Most of the time this isn't an issue
     206                 :             :  * because we don't need to reference the join's output columns as such; we
     207                 :             :  * can reference the input columns instead.  That approach can fail for merged
     208                 :             :  * JOIN USING columns, however, so when we have one of those in an unnamed
     209                 :             :  * join, we have to make that column's alias globally unique across the whole
     210                 :             :  * query to ensure it can be referenced unambiguously.
     211                 :             :  *
     212                 :             :  * Another problem is that a JOIN USING clause requires the columns to be
     213                 :             :  * merged to have the same aliases in both input RTEs, and that no other
     214                 :             :  * columns in those RTEs or their children conflict with the USING names.
     215                 :             :  * To handle that, we do USING-column alias assignment in a recursive
     216                 :             :  * traversal of the query's jointree.  When descending through a JOIN with
     217                 :             :  * USING, we preassign the USING column names to the child columns, overriding
     218                 :             :  * other rules for column alias assignment.  We also mark each RTE with a list
     219                 :             :  * of all USING column names selected for joins containing that RTE, so that
     220                 :             :  * when we assign other columns' aliases later, we can avoid conflicts.
     221                 :             :  *
     222                 :             :  * Another problem is that if a JOIN's input tables have had columns added or
     223                 :             :  * deleted since the query was parsed, we must generate a column alias list
     224                 :             :  * for the join that matches the current set of input columns --- otherwise, a
     225                 :             :  * change in the number of columns in the left input would throw off matching
     226                 :             :  * of aliases to columns of the right input.  Thus, positions in the printable
     227                 :             :  * column alias list are not necessarily one-for-one with varattnos of the
     228                 :             :  * JOIN, so we need a separate new_colnames[] array for printing purposes.
     229                 :             :  *
     230                 :             :  * Finally, when dealing with wide tables we risk O(N^2) costs in assigning
     231                 :             :  * non-duplicate column names.  We ameliorate that by using a hash table that
     232                 :             :  * holds all the strings appearing in colnames, new_colnames, and parentUsing.
     233                 :             :  */
     234                 :             : typedef struct
     235                 :             : {
     236                 :             :         /*
     237                 :             :          * colnames is an array containing column aliases to use for columns that
     238                 :             :          * existed when the query was parsed.  Dropped columns have NULL entries.
     239                 :             :          * This array can be directly indexed by varattno to get a Var's name.
     240                 :             :          *
     241                 :             :          * Non-NULL entries are guaranteed unique within the RTE, *except* when
     242                 :             :          * this is for an unnamed JOIN RTE.  In that case we merely copy up names
     243                 :             :          * from the two input RTEs.
     244                 :             :          *
     245                 :             :          * During the recursive descent in set_using_names(), forcible assignment
     246                 :             :          * of a child RTE's column name is represented by pre-setting that element
     247                 :             :          * of the child's colnames array.  So at that stage, NULL entries in this
     248                 :             :          * array just mean that no name has been preassigned, not necessarily that
     249                 :             :          * the column is dropped.
     250                 :             :          */
     251                 :             :         int                     num_cols;               /* length of colnames[] array */
     252                 :             :         char      **colnames;           /* array of C strings and NULLs */
     253                 :             : 
     254                 :             :         /*
     255                 :             :          * new_colnames is an array containing column aliases to use for columns
     256                 :             :          * that would exist if the query was re-parsed against the current
     257                 :             :          * definitions of its base tables.  This is what to print as the column
     258                 :             :          * alias list for the RTE.  This array does not include dropped columns,
     259                 :             :          * but it will include columns added since original parsing.  Indexes in
     260                 :             :          * it therefore have little to do with current varattno values.  As above,
     261                 :             :          * entries are unique unless this is for an unnamed JOIN RTE.  (In such an
     262                 :             :          * RTE, we never actually print this array, but we must compute it anyway
     263                 :             :          * for possible use in computing column names of upper joins.) The
     264                 :             :          * parallel array is_new_col marks which of these columns are new since
     265                 :             :          * original parsing.  Entries with is_new_col false must match the
     266                 :             :          * non-NULL colnames entries one-for-one.
     267                 :             :          */
     268                 :             :         int                     num_new_cols;   /* length of new_colnames[] array */
     269                 :             :         char      **new_colnames;       /* array of C strings */
     270                 :             :         bool       *is_new_col;         /* array of bool flags */
     271                 :             : 
     272                 :             :         /* This flag tells whether we should actually print a column alias list */
     273                 :             :         bool            printaliases;
     274                 :             : 
     275                 :             :         /* This list has all names used as USING names in joins above this RTE */
     276                 :             :         List       *parentUsing;        /* names assigned to parent merged columns */
     277                 :             : 
     278                 :             :         /*
     279                 :             :          * If this struct is for a JOIN RTE, we fill these fields during the
     280                 :             :          * set_using_names() pass to describe its relationship to its child RTEs.
     281                 :             :          *
     282                 :             :          * leftattnos and rightattnos are arrays with one entry per existing
     283                 :             :          * output column of the join (hence, indexable by join varattno).  For a
     284                 :             :          * simple reference to a column of the left child, leftattnos[i] is the
     285                 :             :          * child RTE's attno and rightattnos[i] is zero; and conversely for a
     286                 :             :          * column of the right child.  But for merged columns produced by JOIN
     287                 :             :          * USING/NATURAL JOIN, both leftattnos[i] and rightattnos[i] are nonzero.
     288                 :             :          * Note that a simple reference might be to a child RTE column that's been
     289                 :             :          * dropped; but that's OK since the column could not be used in the query.
     290                 :             :          *
     291                 :             :          * If it's a JOIN USING, usingNames holds the alias names selected for the
     292                 :             :          * merged columns (these might be different from the original USING list,
     293                 :             :          * if we had to modify names to achieve uniqueness).
     294                 :             :          */
     295                 :             :         int                     leftrti;                /* rangetable index of left child */
     296                 :             :         int                     rightrti;               /* rangetable index of right child */
     297                 :             :         int                *leftattnos;         /* left-child varattnos of join cols, or 0 */
     298                 :             :         int                *rightattnos;        /* right-child varattnos of join cols, or 0 */
     299                 :             :         List       *usingNames;         /* names assigned to merged columns */
     300                 :             : 
     301                 :             :         /*
     302                 :             :          * Hash table holding copies of all the strings appearing in this struct's
     303                 :             :          * colnames, new_colnames, and parentUsing.  We use a hash table only for
     304                 :             :          * sufficiently wide relations, and only during the colname-assignment
     305                 :             :          * functions set_relation_column_names and set_join_column_names;
     306                 :             :          * otherwise, names_hash is NULL.
     307                 :             :          */
     308                 :             :         HTAB       *names_hash;         /* entries are just strings */
     309                 :             : } deparse_columns;
     310                 :             : 
     311                 :             : /* This macro is analogous to rt_fetch(), but for deparse_columns structs */
     312                 :             : #define deparse_columns_fetch(rangetable_index, dpns) \
     313                 :             :         ((deparse_columns *) list_nth((dpns)->rtable_columns, (rangetable_index)-1))
     314                 :             : 
     315                 :             : /*
     316                 :             :  * Entry in set_rtable_names' hash table
     317                 :             :  */
     318                 :             : typedef struct
     319                 :             : {
     320                 :             :         char            name[NAMEDATALEN];      /* Hash key --- must be first */
     321                 :             :         int                     counter;                /* Largest addition used so far for name */
     322                 :             : } NameHashEntry;
     323                 :             : 
     324                 :             : /* Callback signature for resolve_special_varno() */
     325                 :             : typedef void (*rsv_callback) (Node *node, deparse_context *context,
     326                 :             :                                                           void *callback_arg);
     327                 :             : 
     328                 :             : 
     329                 :             : /* ----------
     330                 :             :  * Global data
     331                 :             :  * ----------
     332                 :             :  */
     333                 :             : static SPIPlanPtr plan_getrulebyoid = NULL;
     334                 :             : static const char *const query_getrulebyoid = "SELECT * FROM pg_catalog.pg_rewrite WHERE oid = $1";
     335                 :             : static SPIPlanPtr plan_getviewrule = NULL;
     336                 :             : static const char *const query_getviewrule = "SELECT * FROM pg_catalog.pg_rewrite WHERE ev_class = $1 AND rulename = $2";
     337                 :             : 
     338                 :             : /* GUC parameters */
     339                 :             : bool            quote_all_identifiers = false;
     340                 :             : 
     341                 :             : 
     342                 :             : /* ----------
     343                 :             :  * Local functions
     344                 :             :  *
     345                 :             :  * Most of these functions used to use fixed-size buffers to build their
     346                 :             :  * results.  Now, they take an (already initialized) StringInfo object
     347                 :             :  * as a parameter, and append their text output to its contents.
     348                 :             :  * ----------
     349                 :             :  */
     350                 :             : static char *deparse_expression_pretty(Node *expr, List *dpcontext,
     351                 :             :                                                                            bool forceprefix, bool showimplicit,
     352                 :             :                                                                            int prettyFlags, int startIndent);
     353                 :             : static char *pg_get_viewdef_worker(Oid viewoid,
     354                 :             :                                                                    int prettyFlags, int wrapColumn);
     355                 :             : static char *pg_get_triggerdef_worker(Oid trigid, bool pretty);
     356                 :             : static int      decompile_column_index_array(Datum column_index_array, Oid relId,
     357                 :             :                                                                                  bool withPeriod, StringInfo buf);
     358                 :             : static char *pg_get_ruledef_worker(Oid ruleoid, int prettyFlags);
     359                 :             : static char *pg_get_indexdef_worker(Oid indexrelid, int colno,
     360                 :             :                                                                         const Oid *excludeOps,
     361                 :             :                                                                         bool attrsOnly, bool keysOnly,
     362                 :             :                                                                         bool showTblSpc, bool inherits,
     363                 :             :                                                                         int prettyFlags, bool missing_ok);
     364                 :             : static char *pg_get_statisticsobj_worker(Oid statextid, bool columns_only,
     365                 :             :                                                                                  bool missing_ok);
     366                 :             : static char *pg_get_partkeydef_worker(Oid relid, int prettyFlags,
     367                 :             :                                                                           bool attrsOnly, bool missing_ok);
     368                 :             : static char *pg_get_constraintdef_worker(Oid constraintId, bool fullCommand,
     369                 :             :                                                                                  int prettyFlags, bool missing_ok);
     370                 :             : static text *pg_get_expr_worker(text *expr, Oid relid, int prettyFlags);
     371                 :             : static int      print_function_arguments(StringInfo buf, HeapTuple proctup,
     372                 :             :                                                                          bool print_table_args, bool print_defaults);
     373                 :             : static void print_function_rettype(StringInfo buf, HeapTuple proctup);
     374                 :             : static void print_function_trftypes(StringInfo buf, HeapTuple proctup);
     375                 :             : static void print_function_sqlbody(StringInfo buf, HeapTuple proctup);
     376                 :             : static void set_rtable_names(deparse_namespace *dpns, List *parent_namespaces,
     377                 :             :                                                          Bitmapset *rels_used);
     378                 :             : static void set_deparse_for_query(deparse_namespace *dpns, Query *query,
     379                 :             :                                                                   List *parent_namespaces);
     380                 :             : static void set_simple_column_names(deparse_namespace *dpns);
     381                 :             : static bool has_dangerous_join_using(deparse_namespace *dpns, Node *jtnode);
     382                 :             : static void set_using_names(deparse_namespace *dpns, Node *jtnode,
     383                 :             :                                                         List *parentUsing);
     384                 :             : static void set_relation_column_names(deparse_namespace *dpns,
     385                 :             :                                                                           RangeTblEntry *rte,
     386                 :             :                                                                           deparse_columns *colinfo);
     387                 :             : static void set_join_column_names(deparse_namespace *dpns, RangeTblEntry *rte,
     388                 :             :                                                                   deparse_columns *colinfo);
     389                 :             : static bool colname_is_unique(const char *colname, deparse_namespace *dpns,
     390                 :             :                                                           deparse_columns *colinfo);
     391                 :             : static char *make_colname_unique(char *colname, deparse_namespace *dpns,
     392                 :             :                                                                  deparse_columns *colinfo);
     393                 :             : static void expand_colnames_array_to(deparse_columns *colinfo, int n);
     394                 :             : static void build_colinfo_names_hash(deparse_columns *colinfo);
     395                 :             : static void add_to_names_hash(deparse_columns *colinfo, const char *name);
     396                 :             : static void destroy_colinfo_names_hash(deparse_columns *colinfo);
     397                 :             : static void identify_join_columns(JoinExpr *j, RangeTblEntry *jrte,
     398                 :             :                                                                   deparse_columns *colinfo);
     399                 :             : static char *get_rtable_name(int rtindex, deparse_context *context);
     400                 :             : static void set_deparse_plan(deparse_namespace *dpns, Plan *plan);
     401                 :             : static Plan *find_recursive_union(deparse_namespace *dpns,
     402                 :             :                                                                   WorkTableScan *wtscan);
     403                 :             : static void push_child_plan(deparse_namespace *dpns, Plan *plan,
     404                 :             :                                                         deparse_namespace *save_dpns);
     405                 :             : static void pop_child_plan(deparse_namespace *dpns,
     406                 :             :                                                    deparse_namespace *save_dpns);
     407                 :             : static void push_ancestor_plan(deparse_namespace *dpns, ListCell *ancestor_cell,
     408                 :             :                                                            deparse_namespace *save_dpns);
     409                 :             : static void pop_ancestor_plan(deparse_namespace *dpns,
     410                 :             :                                                           deparse_namespace *save_dpns);
     411                 :             : static void make_ruledef(StringInfo buf, HeapTuple ruletup, TupleDesc rulettc,
     412                 :             :                                                  int prettyFlags);
     413                 :             : static void make_viewdef(StringInfo buf, HeapTuple ruletup, TupleDesc rulettc,
     414                 :             :                                                  int prettyFlags, int wrapColumn);
     415                 :             : static void get_query_def(Query *query, StringInfo buf, List *parentnamespace,
     416                 :             :                                                   TupleDesc resultDesc, bool colNamesVisible,
     417                 :             :                                                   int prettyFlags, int wrapColumn, int startIndent);
     418                 :             : static void get_values_def(List *values_lists, deparse_context *context);
     419                 :             : static void get_with_clause(Query *query, deparse_context *context);
     420                 :             : static void get_select_query_def(Query *query, deparse_context *context);
     421                 :             : static void get_insert_query_def(Query *query, deparse_context *context);
     422                 :             : static void get_update_query_def(Query *query, deparse_context *context);
     423                 :             : static void get_update_query_targetlist_def(Query *query, List *targetList,
     424                 :             :                                                                                         deparse_context *context,
     425                 :             :                                                                                         RangeTblEntry *rte);
     426                 :             : static void get_delete_query_def(Query *query, deparse_context *context);
     427                 :             : static void get_merge_query_def(Query *query, deparse_context *context);
     428                 :             : static void get_utility_query_def(Query *query, deparse_context *context);
     429                 :             : static void get_basic_select_query(Query *query, deparse_context *context);
     430                 :             : static void get_target_list(List *targetList, deparse_context *context);
     431                 :             : static void get_returning_clause(Query *query, deparse_context *context);
     432                 :             : static void get_setop_query(Node *setOp, Query *query,
     433                 :             :                                                         deparse_context *context);
     434                 :             : static Node *get_rule_sortgroupclause(Index ref, List *tlist,
     435                 :             :                                                                           bool force_colno,
     436                 :             :                                                                           deparse_context *context);
     437                 :             : static void get_rule_groupingset(GroupingSet *gset, List *targetlist,
     438                 :             :                                                                  bool omit_parens, deparse_context *context);
     439                 :             : static void get_rule_orderby(List *orderList, List *targetList,
     440                 :             :                                                          bool force_colno, deparse_context *context);
     441                 :             : static void get_rule_windowclause(Query *query, deparse_context *context);
     442                 :             : static void get_rule_windowspec(WindowClause *wc, List *targetList,
     443                 :             :                                                                 deparse_context *context);
     444                 :             : static void get_window_frame_options(int frameOptions,
     445                 :             :                                                                          Node *startOffset, Node *endOffset,
     446                 :             :                                                                          deparse_context *context);
     447                 :             : static char *get_variable(Var *var, int levelsup, bool istoplevel,
     448                 :             :                                                   deparse_context *context);
     449                 :             : static void get_special_variable(Node *node, deparse_context *context,
     450                 :             :                                                                  void *callback_arg);
     451                 :             : static void resolve_special_varno(Node *node, deparse_context *context,
     452                 :             :                                                                   rsv_callback callback, void *callback_arg);
     453                 :             : static Node *find_param_referent(Param *param, deparse_context *context,
     454                 :             :                                                                  deparse_namespace **dpns_p, ListCell **ancestor_cell_p);
     455                 :             : static SubPlan *find_param_generator(Param *param, deparse_context *context,
     456                 :             :                                                                          int *column_p);
     457                 :             : static SubPlan *find_param_generator_initplan(Param *param, Plan *plan,
     458                 :             :                                                                                           int *column_p);
     459                 :             : static void get_parameter(Param *param, deparse_context *context);
     460                 :             : static const char *get_simple_binary_op_name(OpExpr *expr);
     461                 :             : static bool isSimpleNode(Node *node, Node *parentNode, int prettyFlags);
     462                 :             : static void appendContextKeyword(deparse_context *context, const char *str,
     463                 :             :                                                                  int indentBefore, int indentAfter, int indentPlus);
     464                 :             : static void removeStringInfoSpaces(StringInfo str);
     465                 :             : static void get_rule_expr(Node *node, deparse_context *context,
     466                 :             :                                                   bool showimplicit);
     467                 :             : static void get_rule_expr_toplevel(Node *node, deparse_context *context,
     468                 :             :                                                                    bool showimplicit);
     469                 :             : static void get_rule_list_toplevel(List *lst, deparse_context *context,
     470                 :             :                                                                    bool showimplicit);
     471                 :             : static void get_rule_expr_funccall(Node *node, deparse_context *context,
     472                 :             :                                                                    bool showimplicit);
     473                 :             : static bool looks_like_function(Node *node);
     474                 :             : static void get_oper_expr(OpExpr *expr, deparse_context *context);
     475                 :             : static void get_func_expr(FuncExpr *expr, deparse_context *context,
     476                 :             :                                                   bool showimplicit);
     477                 :             : static void get_agg_expr(Aggref *aggref, deparse_context *context,
     478                 :             :                                                  Aggref *original_aggref);
     479                 :             : static void get_agg_expr_helper(Aggref *aggref, deparse_context *context,
     480                 :             :                                                                 Aggref *original_aggref, const char *funcname,
     481                 :             :                                                                 const char *options, bool is_json_objectagg);
     482                 :             : static void get_agg_combine_expr(Node *node, deparse_context *context,
     483                 :             :                                                                  void *callback_arg);
     484                 :             : static void get_windowfunc_expr(WindowFunc *wfunc, deparse_context *context);
     485                 :             : static void get_windowfunc_expr_helper(WindowFunc *wfunc, deparse_context *context,
     486                 :             :                                                                            const char *funcname, const char *options,
     487                 :             :                                                                            bool is_json_objectagg);
     488                 :             : static bool get_func_sql_syntax(FuncExpr *expr, deparse_context *context);
     489                 :             : static void get_coercion_expr(Node *arg, deparse_context *context,
     490                 :             :                                                           Oid resulttype, int32 resulttypmod,
     491                 :             :                                                           Node *parentNode);
     492                 :             : static void get_const_expr(Const *constval, deparse_context *context,
     493                 :             :                                                    int showtype);
     494                 :             : static void get_const_collation(Const *constval, deparse_context *context);
     495                 :             : static void get_json_format(JsonFormat *format, StringInfo buf);
     496                 :             : static void get_json_returning(JsonReturning *returning, StringInfo buf,
     497                 :             :                                                            bool json_format_by_default);
     498                 :             : static void get_json_constructor(JsonConstructorExpr *ctor,
     499                 :             :                                                                  deparse_context *context, bool showimplicit);
     500                 :             : static void get_json_constructor_options(JsonConstructorExpr *ctor,
     501                 :             :                                                                                  StringInfo buf);
     502                 :             : static void get_json_agg_constructor(JsonConstructorExpr *ctor,
     503                 :             :                                                                          deparse_context *context,
     504                 :             :                                                                          const char *funcname,
     505                 :             :                                                                          bool is_json_objectagg);
     506                 :             : static void simple_quote_literal(StringInfo buf, const char *val);
     507                 :             : static void get_sublink_expr(SubLink *sublink, deparse_context *context);
     508                 :             : static void get_tablefunc(TableFunc *tf, deparse_context *context,
     509                 :             :                                                   bool showimplicit);
     510                 :             : static void get_from_clause(Query *query, const char *prefix,
     511                 :             :                                                         deparse_context *context);
     512                 :             : static void get_from_clause_item(Node *jtnode, Query *query,
     513                 :             :                                                                  deparse_context *context);
     514                 :             : static void get_rte_alias(RangeTblEntry *rte, int varno, bool use_as,
     515                 :             :                                                   deparse_context *context);
     516                 :             : static void get_column_alias_list(deparse_columns *colinfo,
     517                 :             :                                                                   deparse_context *context);
     518                 :             : static void get_from_clause_coldeflist(RangeTblFunction *rtfunc,
     519                 :             :                                                                            deparse_columns *colinfo,
     520                 :             :                                                                            deparse_context *context);
     521                 :             : static void get_tablesample_def(TableSampleClause *tablesample,
     522                 :             :                                                                 deparse_context *context);
     523                 :             : static void get_opclass_name(Oid opclass, Oid actual_datatype,
     524                 :             :                                                          StringInfo buf);
     525                 :             : static Node *processIndirection(Node *node, deparse_context *context);
     526                 :             : static void printSubscripts(SubscriptingRef *sbsref, deparse_context *context);
     527                 :             : static char *get_relation_name(Oid relid);
     528                 :             : static char *generate_relation_name(Oid relid, List *namespaces);
     529                 :             : static char *generate_qualified_relation_name(Oid relid);
     530                 :             : static char *generate_function_name(Oid funcid, int nargs,
     531                 :             :                                                                         List *argnames, Oid *argtypes,
     532                 :             :                                                                         bool has_variadic, bool *use_variadic_p,
     533                 :             :                                                                         bool inGroupBy);
     534                 :             : static char *generate_operator_name(Oid operid, Oid arg1, Oid arg2);
     535                 :             : static void add_cast_to(StringInfo buf, Oid typid);
     536                 :             : static char *generate_qualified_type_name(Oid typid);
     537                 :             : static text *string_to_text(char *str);
     538                 :             : static char *flatten_reloptions(Oid relid);
     539                 :             : static void get_reloptions(StringInfo buf, Datum reloptions);
     540                 :             : static void get_json_path_spec(Node *path_spec, deparse_context *context,
     541                 :             :                                                            bool showimplicit);
     542                 :             : static void get_json_table_columns(TableFunc *tf, JsonTablePathScan *scan,
     543                 :             :                                                                    deparse_context *context,
     544                 :             :                                                                    bool showimplicit);
     545                 :             : static void get_json_table_nested_columns(TableFunc *tf, JsonTablePlan *plan,
     546                 :             :                                                                                   deparse_context *context,
     547                 :             :                                                                                   bool showimplicit,
     548                 :             :                                                                                   bool needcomma);
     549                 :             : 
     550                 :             : #define only_marker(rte)  ((rte)->inh ? "" : "ONLY ")
     551                 :             : 
     552                 :             : 
     553                 :             : /* ----------
     554                 :             :  * pg_get_ruledef               - Do it all and return a text
     555                 :             :  *                                that could be used as a statement
     556                 :             :  *                                to recreate the rule
     557                 :             :  * ----------
     558                 :             :  */
     559                 :             : Datum
     560                 :           0 : pg_get_ruledef(PG_FUNCTION_ARGS)
     561                 :             : {
     562                 :           0 :         Oid                     ruleoid = PG_GETARG_OID(0);
     563                 :           0 :         int                     prettyFlags;
     564                 :           0 :         char       *res;
     565                 :             : 
     566                 :           0 :         prettyFlags = PRETTYFLAG_INDENT;
     567                 :             : 
     568                 :           0 :         res = pg_get_ruledef_worker(ruleoid, prettyFlags);
     569                 :             : 
     570         [ #  # ]:           0 :         if (res == NULL)
     571                 :           0 :                 PG_RETURN_NULL();
     572                 :             : 
     573                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
     574                 :           0 : }
     575                 :             : 
     576                 :             : 
     577                 :             : Datum
     578                 :           0 : pg_get_ruledef_ext(PG_FUNCTION_ARGS)
     579                 :             : {
     580                 :           0 :         Oid                     ruleoid = PG_GETARG_OID(0);
     581                 :           0 :         bool            pretty = PG_GETARG_BOOL(1);
     582                 :           0 :         int                     prettyFlags;
     583                 :           0 :         char       *res;
     584                 :             : 
     585                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(pretty);
     586                 :             : 
     587                 :           0 :         res = pg_get_ruledef_worker(ruleoid, prettyFlags);
     588                 :             : 
     589         [ #  # ]:           0 :         if (res == NULL)
     590                 :           0 :                 PG_RETURN_NULL();
     591                 :             : 
     592                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
     593                 :           0 : }
     594                 :             : 
     595                 :             : 
     596                 :             : static char *
     597                 :           0 : pg_get_ruledef_worker(Oid ruleoid, int prettyFlags)
     598                 :             : {
     599                 :           0 :         Datum           args[1];
     600                 :           0 :         char            nulls[1];
     601                 :           0 :         int                     spirc;
     602                 :           0 :         HeapTuple       ruletup;
     603                 :           0 :         TupleDesc       rulettc;
     604                 :           0 :         StringInfoData buf;
     605                 :             : 
     606                 :             :         /*
     607                 :             :          * Do this first so that string is alloc'd in outer context not SPI's.
     608                 :             :          */
     609                 :           0 :         initStringInfo(&buf);
     610                 :             : 
     611                 :             :         /*
     612                 :             :          * Connect to SPI manager
     613                 :             :          */
     614                 :           0 :         SPI_connect();
     615                 :             : 
     616                 :             :         /*
     617                 :             :          * On the first call prepare the plan to lookup pg_rewrite. We read
     618                 :             :          * pg_rewrite over the SPI manager instead of using the syscache to be
     619                 :             :          * checked for read access on pg_rewrite.
     620                 :             :          */
     621         [ #  # ]:           0 :         if (plan_getrulebyoid == NULL)
     622                 :             :         {
     623                 :           0 :                 Oid                     argtypes[1];
     624                 :           0 :                 SPIPlanPtr      plan;
     625                 :             : 
     626                 :           0 :                 argtypes[0] = OIDOID;
     627                 :           0 :                 plan = SPI_prepare(query_getrulebyoid, 1, argtypes);
     628         [ #  # ]:           0 :                 if (plan == NULL)
     629   [ #  #  #  # ]:           0 :                         elog(ERROR, "SPI_prepare failed for \"%s\"", query_getrulebyoid);
     630                 :           0 :                 SPI_keepplan(plan);
     631                 :           0 :                 plan_getrulebyoid = plan;
     632                 :           0 :         }
     633                 :             : 
     634                 :             :         /*
     635                 :             :          * Get the pg_rewrite tuple for this rule
     636                 :             :          */
     637                 :           0 :         args[0] = ObjectIdGetDatum(ruleoid);
     638                 :           0 :         nulls[0] = ' ';
     639                 :           0 :         spirc = SPI_execute_plan(plan_getrulebyoid, args, nulls, true, 0);
     640         [ #  # ]:           0 :         if (spirc != SPI_OK_SELECT)
     641   [ #  #  #  # ]:           0 :                 elog(ERROR, "failed to get pg_rewrite tuple for rule %u", ruleoid);
     642         [ #  # ]:           0 :         if (SPI_processed != 1)
     643                 :             :         {
     644                 :             :                 /*
     645                 :             :                  * There is no tuple data available here, just keep the output buffer
     646                 :             :                  * empty.
     647                 :             :                  */
     648                 :           0 :         }
     649                 :             :         else
     650                 :             :         {
     651                 :             :                 /*
     652                 :             :                  * Get the rule's definition and put it into executor's memory
     653                 :             :                  */
     654                 :           0 :                 ruletup = SPI_tuptable->vals[0];
     655                 :           0 :                 rulettc = SPI_tuptable->tupdesc;
     656                 :           0 :                 make_ruledef(&buf, ruletup, rulettc, prettyFlags);
     657                 :             :         }
     658                 :             : 
     659                 :             :         /*
     660                 :             :          * Disconnect from SPI manager
     661                 :             :          */
     662         [ #  # ]:           0 :         if (SPI_finish() != SPI_OK_FINISH)
     663   [ #  #  #  # ]:           0 :                 elog(ERROR, "SPI_finish failed");
     664                 :             : 
     665         [ #  # ]:           0 :         if (buf.len == 0)
     666                 :           0 :                 return NULL;
     667                 :             : 
     668                 :           0 :         return buf.data;
     669                 :           0 : }
     670                 :             : 
     671                 :             : 
     672                 :             : /* ----------
     673                 :             :  * pg_get_viewdef               - Mainly the same thing, but we
     674                 :             :  *                                only return the SELECT part of a view
     675                 :             :  * ----------
     676                 :             :  */
     677                 :             : Datum
     678                 :           0 : pg_get_viewdef(PG_FUNCTION_ARGS)
     679                 :             : {
     680                 :             :         /* By OID */
     681                 :           0 :         Oid                     viewoid = PG_GETARG_OID(0);
     682                 :           0 :         int                     prettyFlags;
     683                 :           0 :         char       *res;
     684                 :             : 
     685                 :           0 :         prettyFlags = PRETTYFLAG_INDENT;
     686                 :             : 
     687                 :           0 :         res = pg_get_viewdef_worker(viewoid, prettyFlags, WRAP_COLUMN_DEFAULT);
     688                 :             : 
     689         [ #  # ]:           0 :         if (res == NULL)
     690                 :           0 :                 PG_RETURN_NULL();
     691                 :             : 
     692                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
     693                 :           0 : }
     694                 :             : 
     695                 :             : 
     696                 :             : Datum
     697                 :           0 : pg_get_viewdef_ext(PG_FUNCTION_ARGS)
     698                 :             : {
     699                 :             :         /* By OID */
     700                 :           0 :         Oid                     viewoid = PG_GETARG_OID(0);
     701                 :           0 :         bool            pretty = PG_GETARG_BOOL(1);
     702                 :           0 :         int                     prettyFlags;
     703                 :           0 :         char       *res;
     704                 :             : 
     705                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(pretty);
     706                 :             : 
     707                 :           0 :         res = pg_get_viewdef_worker(viewoid, prettyFlags, WRAP_COLUMN_DEFAULT);
     708                 :             : 
     709         [ #  # ]:           0 :         if (res == NULL)
     710                 :           0 :                 PG_RETURN_NULL();
     711                 :             : 
     712                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
     713                 :           0 : }
     714                 :             : 
     715                 :             : Datum
     716                 :           0 : pg_get_viewdef_wrap(PG_FUNCTION_ARGS)
     717                 :             : {
     718                 :             :         /* By OID */
     719                 :           0 :         Oid                     viewoid = PG_GETARG_OID(0);
     720                 :           0 :         int                     wrap = PG_GETARG_INT32(1);
     721                 :           0 :         int                     prettyFlags;
     722                 :           0 :         char       *res;
     723                 :             : 
     724                 :             :         /* calling this implies we want pretty printing */
     725                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(true);
     726                 :             : 
     727                 :           0 :         res = pg_get_viewdef_worker(viewoid, prettyFlags, wrap);
     728                 :             : 
     729         [ #  # ]:           0 :         if (res == NULL)
     730                 :           0 :                 PG_RETURN_NULL();
     731                 :             : 
     732                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
     733                 :           0 : }
     734                 :             : 
     735                 :             : Datum
     736                 :           0 : pg_get_viewdef_name(PG_FUNCTION_ARGS)
     737                 :             : {
     738                 :             :         /* By qualified name */
     739                 :           0 :         text       *viewname = PG_GETARG_TEXT_PP(0);
     740                 :           0 :         int                     prettyFlags;
     741                 :           0 :         RangeVar   *viewrel;
     742                 :           0 :         Oid                     viewoid;
     743                 :           0 :         char       *res;
     744                 :             : 
     745                 :           0 :         prettyFlags = PRETTYFLAG_INDENT;
     746                 :             : 
     747                 :             :         /* Look up view name.  Can't lock it - we might not have privileges. */
     748                 :           0 :         viewrel = makeRangeVarFromNameList(textToQualifiedNameList(viewname));
     749                 :           0 :         viewoid = RangeVarGetRelid(viewrel, NoLock, false);
     750                 :             : 
     751                 :           0 :         res = pg_get_viewdef_worker(viewoid, prettyFlags, WRAP_COLUMN_DEFAULT);
     752                 :             : 
     753         [ #  # ]:           0 :         if (res == NULL)
     754                 :           0 :                 PG_RETURN_NULL();
     755                 :             : 
     756                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
     757                 :           0 : }
     758                 :             : 
     759                 :             : 
     760                 :             : Datum
     761                 :           0 : pg_get_viewdef_name_ext(PG_FUNCTION_ARGS)
     762                 :             : {
     763                 :             :         /* By qualified name */
     764                 :           0 :         text       *viewname = PG_GETARG_TEXT_PP(0);
     765                 :           0 :         bool            pretty = PG_GETARG_BOOL(1);
     766                 :           0 :         int                     prettyFlags;
     767                 :           0 :         RangeVar   *viewrel;
     768                 :           0 :         Oid                     viewoid;
     769                 :           0 :         char       *res;
     770                 :             : 
     771                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(pretty);
     772                 :             : 
     773                 :             :         /* Look up view name.  Can't lock it - we might not have privileges. */
     774                 :           0 :         viewrel = makeRangeVarFromNameList(textToQualifiedNameList(viewname));
     775                 :           0 :         viewoid = RangeVarGetRelid(viewrel, NoLock, false);
     776                 :             : 
     777                 :           0 :         res = pg_get_viewdef_worker(viewoid, prettyFlags, WRAP_COLUMN_DEFAULT);
     778                 :             : 
     779         [ #  # ]:           0 :         if (res == NULL)
     780                 :           0 :                 PG_RETURN_NULL();
     781                 :             : 
     782                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
     783                 :           0 : }
     784                 :             : 
     785                 :             : /*
     786                 :             :  * Common code for by-OID and by-name variants of pg_get_viewdef
     787                 :             :  */
     788                 :             : static char *
     789                 :           0 : pg_get_viewdef_worker(Oid viewoid, int prettyFlags, int wrapColumn)
     790                 :             : {
     791                 :           0 :         Datum           args[2];
     792                 :           0 :         char            nulls[2];
     793                 :           0 :         int                     spirc;
     794                 :           0 :         HeapTuple       ruletup;
     795                 :           0 :         TupleDesc       rulettc;
     796                 :           0 :         StringInfoData buf;
     797                 :             : 
     798                 :             :         /*
     799                 :             :          * Do this first so that string is alloc'd in outer context not SPI's.
     800                 :             :          */
     801                 :           0 :         initStringInfo(&buf);
     802                 :             : 
     803                 :             :         /*
     804                 :             :          * Connect to SPI manager
     805                 :             :          */
     806                 :           0 :         SPI_connect();
     807                 :             : 
     808                 :             :         /*
     809                 :             :          * On the first call prepare the plan to lookup pg_rewrite. We read
     810                 :             :          * pg_rewrite over the SPI manager instead of using the syscache to be
     811                 :             :          * checked for read access on pg_rewrite.
     812                 :             :          */
     813         [ #  # ]:           0 :         if (plan_getviewrule == NULL)
     814                 :             :         {
     815                 :           0 :                 Oid                     argtypes[2];
     816                 :           0 :                 SPIPlanPtr      plan;
     817                 :             : 
     818                 :           0 :                 argtypes[0] = OIDOID;
     819                 :           0 :                 argtypes[1] = NAMEOID;
     820                 :           0 :                 plan = SPI_prepare(query_getviewrule, 2, argtypes);
     821         [ #  # ]:           0 :                 if (plan == NULL)
     822   [ #  #  #  # ]:           0 :                         elog(ERROR, "SPI_prepare failed for \"%s\"", query_getviewrule);
     823                 :           0 :                 SPI_keepplan(plan);
     824                 :           0 :                 plan_getviewrule = plan;
     825                 :           0 :         }
     826                 :             : 
     827                 :             :         /*
     828                 :             :          * Get the pg_rewrite tuple for the view's SELECT rule
     829                 :             :          */
     830                 :           0 :         args[0] = ObjectIdGetDatum(viewoid);
     831                 :           0 :         args[1] = DirectFunctionCall1(namein, CStringGetDatum(ViewSelectRuleName));
     832                 :           0 :         nulls[0] = ' ';
     833                 :           0 :         nulls[1] = ' ';
     834                 :           0 :         spirc = SPI_execute_plan(plan_getviewrule, args, nulls, true, 0);
     835         [ #  # ]:           0 :         if (spirc != SPI_OK_SELECT)
     836   [ #  #  #  # ]:           0 :                 elog(ERROR, "failed to get pg_rewrite tuple for view %u", viewoid);
     837         [ #  # ]:           0 :         if (SPI_processed != 1)
     838                 :             :         {
     839                 :             :                 /*
     840                 :             :                  * There is no tuple data available here, just keep the output buffer
     841                 :             :                  * empty.
     842                 :             :                  */
     843                 :           0 :         }
     844                 :             :         else
     845                 :             :         {
     846                 :             :                 /*
     847                 :             :                  * Get the rule's definition and put it into executor's memory
     848                 :             :                  */
     849                 :           0 :                 ruletup = SPI_tuptable->vals[0];
     850                 :           0 :                 rulettc = SPI_tuptable->tupdesc;
     851                 :           0 :                 make_viewdef(&buf, ruletup, rulettc, prettyFlags, wrapColumn);
     852                 :             :         }
     853                 :             : 
     854                 :             :         /*
     855                 :             :          * Disconnect from SPI manager
     856                 :             :          */
     857         [ #  # ]:           0 :         if (SPI_finish() != SPI_OK_FINISH)
     858   [ #  #  #  # ]:           0 :                 elog(ERROR, "SPI_finish failed");
     859                 :             : 
     860         [ #  # ]:           0 :         if (buf.len == 0)
     861                 :           0 :                 return NULL;
     862                 :             : 
     863                 :           0 :         return buf.data;
     864                 :           0 : }
     865                 :             : 
     866                 :             : /* ----------
     867                 :             :  * pg_get_triggerdef            - Get the definition of a trigger
     868                 :             :  * ----------
     869                 :             :  */
     870                 :             : Datum
     871                 :           0 : pg_get_triggerdef(PG_FUNCTION_ARGS)
     872                 :             : {
     873                 :           0 :         Oid                     trigid = PG_GETARG_OID(0);
     874                 :           0 :         char       *res;
     875                 :             : 
     876                 :           0 :         res = pg_get_triggerdef_worker(trigid, false);
     877                 :             : 
     878         [ #  # ]:           0 :         if (res == NULL)
     879                 :           0 :                 PG_RETURN_NULL();
     880                 :             : 
     881                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
     882                 :           0 : }
     883                 :             : 
     884                 :             : Datum
     885                 :           0 : pg_get_triggerdef_ext(PG_FUNCTION_ARGS)
     886                 :             : {
     887                 :           0 :         Oid                     trigid = PG_GETARG_OID(0);
     888                 :           0 :         bool            pretty = PG_GETARG_BOOL(1);
     889                 :           0 :         char       *res;
     890                 :             : 
     891                 :           0 :         res = pg_get_triggerdef_worker(trigid, pretty);
     892                 :             : 
     893         [ #  # ]:           0 :         if (res == NULL)
     894                 :           0 :                 PG_RETURN_NULL();
     895                 :             : 
     896                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
     897                 :           0 : }
     898                 :             : 
     899                 :             : static char *
     900                 :           0 : pg_get_triggerdef_worker(Oid trigid, bool pretty)
     901                 :             : {
     902                 :           0 :         HeapTuple       ht_trig;
     903                 :           0 :         Form_pg_trigger trigrec;
     904                 :           0 :         StringInfoData buf;
     905                 :           0 :         Relation        tgrel;
     906                 :           0 :         ScanKeyData skey[1];
     907                 :           0 :         SysScanDesc tgscan;
     908                 :           0 :         int                     findx = 0;
     909                 :           0 :         char       *tgname;
     910                 :           0 :         char       *tgoldtable;
     911                 :           0 :         char       *tgnewtable;
     912                 :           0 :         Datum           value;
     913                 :           0 :         bool            isnull;
     914                 :             : 
     915                 :             :         /*
     916                 :             :          * Fetch the pg_trigger tuple by the Oid of the trigger
     917                 :             :          */
     918                 :           0 :         tgrel = table_open(TriggerRelationId, AccessShareLock);
     919                 :             : 
     920                 :           0 :         ScanKeyInit(&skey[0],
     921                 :             :                                 Anum_pg_trigger_oid,
     922                 :             :                                 BTEqualStrategyNumber, F_OIDEQ,
     923                 :           0 :                                 ObjectIdGetDatum(trigid));
     924                 :             : 
     925                 :           0 :         tgscan = systable_beginscan(tgrel, TriggerOidIndexId, true,
     926                 :           0 :                                                                 NULL, 1, skey);
     927                 :             : 
     928                 :           0 :         ht_trig = systable_getnext(tgscan);
     929                 :             : 
     930         [ #  # ]:           0 :         if (!HeapTupleIsValid(ht_trig))
     931                 :             :         {
     932                 :           0 :                 systable_endscan(tgscan);
     933                 :           0 :                 table_close(tgrel, AccessShareLock);
     934                 :           0 :                 return NULL;
     935                 :             :         }
     936                 :             : 
     937                 :           0 :         trigrec = (Form_pg_trigger) GETSTRUCT(ht_trig);
     938                 :             : 
     939                 :             :         /*
     940                 :             :          * Start the trigger definition. Note that the trigger's name should never
     941                 :             :          * be schema-qualified, but the trigger rel's name may be.
     942                 :             :          */
     943                 :           0 :         initStringInfo(&buf);
     944                 :             : 
     945                 :           0 :         tgname = NameStr(trigrec->tgname);
     946                 :           0 :         appendStringInfo(&buf, "CREATE %sTRIGGER %s ",
     947                 :           0 :                                          OidIsValid(trigrec->tgconstraint) ? "CONSTRAINT " : "",
     948                 :           0 :                                          quote_identifier(tgname));
     949                 :             : 
     950         [ #  # ]:           0 :         if (TRIGGER_FOR_BEFORE(trigrec->tgtype))
     951                 :           0 :                 appendStringInfoString(&buf, "BEFORE");
     952         [ #  # ]:           0 :         else if (TRIGGER_FOR_AFTER(trigrec->tgtype))
     953                 :           0 :                 appendStringInfoString(&buf, "AFTER");
     954         [ #  # ]:           0 :         else if (TRIGGER_FOR_INSTEAD(trigrec->tgtype))
     955                 :           0 :                 appendStringInfoString(&buf, "INSTEAD OF");
     956                 :             :         else
     957   [ #  #  #  # ]:           0 :                 elog(ERROR, "unexpected tgtype value: %d", trigrec->tgtype);
     958                 :             : 
     959         [ #  # ]:           0 :         if (TRIGGER_FOR_INSERT(trigrec->tgtype))
     960                 :             :         {
     961                 :           0 :                 appendStringInfoString(&buf, " INSERT");
     962                 :           0 :                 findx++;
     963                 :           0 :         }
     964         [ #  # ]:           0 :         if (TRIGGER_FOR_DELETE(trigrec->tgtype))
     965                 :             :         {
     966         [ #  # ]:           0 :                 if (findx > 0)
     967                 :           0 :                         appendStringInfoString(&buf, " OR DELETE");
     968                 :             :                 else
     969                 :           0 :                         appendStringInfoString(&buf, " DELETE");
     970                 :           0 :                 findx++;
     971                 :           0 :         }
     972         [ #  # ]:           0 :         if (TRIGGER_FOR_UPDATE(trigrec->tgtype))
     973                 :             :         {
     974         [ #  # ]:           0 :                 if (findx > 0)
     975                 :           0 :                         appendStringInfoString(&buf, " OR UPDATE");
     976                 :             :                 else
     977                 :           0 :                         appendStringInfoString(&buf, " UPDATE");
     978                 :           0 :                 findx++;
     979                 :             :                 /* tgattr is first var-width field, so OK to access directly */
     980         [ #  # ]:           0 :                 if (trigrec->tgattr.dim1 > 0)
     981                 :             :                 {
     982                 :           0 :                         int                     i;
     983                 :             : 
     984                 :           0 :                         appendStringInfoString(&buf, " OF ");
     985         [ #  # ]:           0 :                         for (i = 0; i < trigrec->tgattr.dim1; i++)
     986                 :             :                         {
     987                 :           0 :                                 char       *attname;
     988                 :             : 
     989         [ #  # ]:           0 :                                 if (i > 0)
     990                 :           0 :                                         appendStringInfoString(&buf, ", ");
     991                 :           0 :                                 attname = get_attname(trigrec->tgrelid,
     992                 :           0 :                                                                           trigrec->tgattr.values[i], false);
     993                 :           0 :                                 appendStringInfoString(&buf, quote_identifier(attname));
     994                 :           0 :                         }
     995                 :           0 :                 }
     996                 :           0 :         }
     997         [ #  # ]:           0 :         if (TRIGGER_FOR_TRUNCATE(trigrec->tgtype))
     998                 :             :         {
     999         [ #  # ]:           0 :                 if (findx > 0)
    1000                 :           0 :                         appendStringInfoString(&buf, " OR TRUNCATE");
    1001                 :             :                 else
    1002                 :           0 :                         appendStringInfoString(&buf, " TRUNCATE");
    1003                 :           0 :                 findx++;
    1004                 :           0 :         }
    1005                 :             : 
    1006                 :             :         /*
    1007                 :             :          * In non-pretty mode, always schema-qualify the target table name for
    1008                 :             :          * safety.  In pretty mode, schema-qualify only if not visible.
    1009                 :             :          */
    1010                 :           0 :         appendStringInfo(&buf, " ON %s ",
    1011         [ #  # ]:           0 :                                          pretty ?
    1012                 :           0 :                                          generate_relation_name(trigrec->tgrelid, NIL) :
    1013                 :           0 :                                          generate_qualified_relation_name(trigrec->tgrelid));
    1014                 :             : 
    1015         [ #  # ]:           0 :         if (OidIsValid(trigrec->tgconstraint))
    1016                 :             :         {
    1017         [ #  # ]:           0 :                 if (OidIsValid(trigrec->tgconstrrelid))
    1018                 :           0 :                         appendStringInfo(&buf, "FROM %s ",
    1019                 :           0 :                                                          generate_relation_name(trigrec->tgconstrrelid, NIL));
    1020         [ #  # ]:           0 :                 if (!trigrec->tgdeferrable)
    1021                 :           0 :                         appendStringInfoString(&buf, "NOT ");
    1022                 :           0 :                 appendStringInfoString(&buf, "DEFERRABLE INITIALLY ");
    1023         [ #  # ]:           0 :                 if (trigrec->tginitdeferred)
    1024                 :           0 :                         appendStringInfoString(&buf, "DEFERRED ");
    1025                 :             :                 else
    1026                 :           0 :                         appendStringInfoString(&buf, "IMMEDIATE ");
    1027                 :           0 :         }
    1028                 :             : 
    1029                 :           0 :         value = fastgetattr(ht_trig, Anum_pg_trigger_tgoldtable,
    1030                 :           0 :                                                 tgrel->rd_att, &isnull);
    1031         [ #  # ]:           0 :         if (!isnull)
    1032                 :           0 :                 tgoldtable = NameStr(*DatumGetName(value));
    1033                 :             :         else
    1034                 :           0 :                 tgoldtable = NULL;
    1035                 :           0 :         value = fastgetattr(ht_trig, Anum_pg_trigger_tgnewtable,
    1036                 :           0 :                                                 tgrel->rd_att, &isnull);
    1037         [ #  # ]:           0 :         if (!isnull)
    1038                 :           0 :                 tgnewtable = NameStr(*DatumGetName(value));
    1039                 :             :         else
    1040                 :           0 :                 tgnewtable = NULL;
    1041   [ #  #  #  # ]:           0 :         if (tgoldtable != NULL || tgnewtable != NULL)
    1042                 :             :         {
    1043                 :           0 :                 appendStringInfoString(&buf, "REFERENCING ");
    1044         [ #  # ]:           0 :                 if (tgoldtable != NULL)
    1045                 :           0 :                         appendStringInfo(&buf, "OLD TABLE AS %s ",
    1046                 :           0 :                                                          quote_identifier(tgoldtable));
    1047         [ #  # ]:           0 :                 if (tgnewtable != NULL)
    1048                 :           0 :                         appendStringInfo(&buf, "NEW TABLE AS %s ",
    1049                 :           0 :                                                          quote_identifier(tgnewtable));
    1050                 :           0 :         }
    1051                 :             : 
    1052         [ #  # ]:           0 :         if (TRIGGER_FOR_ROW(trigrec->tgtype))
    1053                 :           0 :                 appendStringInfoString(&buf, "FOR EACH ROW ");
    1054                 :             :         else
    1055                 :           0 :                 appendStringInfoString(&buf, "FOR EACH STATEMENT ");
    1056                 :             : 
    1057                 :             :         /* If the trigger has a WHEN qualification, add that */
    1058                 :           0 :         value = fastgetattr(ht_trig, Anum_pg_trigger_tgqual,
    1059                 :           0 :                                                 tgrel->rd_att, &isnull);
    1060         [ #  # ]:           0 :         if (!isnull)
    1061                 :             :         {
    1062                 :           0 :                 Node       *qual;
    1063                 :           0 :                 char            relkind;
    1064                 :           0 :                 deparse_context context;
    1065                 :           0 :                 deparse_namespace dpns;
    1066                 :           0 :                 RangeTblEntry *oldrte;
    1067                 :           0 :                 RangeTblEntry *newrte;
    1068                 :             : 
    1069                 :           0 :                 appendStringInfoString(&buf, "WHEN (");
    1070                 :             : 
    1071                 :           0 :                 qual = stringToNode(TextDatumGetCString(value));
    1072                 :             : 
    1073                 :           0 :                 relkind = get_rel_relkind(trigrec->tgrelid);
    1074                 :             : 
    1075                 :             :                 /* Build minimal OLD and NEW RTEs for the rel */
    1076                 :           0 :                 oldrte = makeNode(RangeTblEntry);
    1077                 :           0 :                 oldrte->rtekind = RTE_RELATION;
    1078                 :           0 :                 oldrte->relid = trigrec->tgrelid;
    1079                 :           0 :                 oldrte->relkind = relkind;
    1080                 :           0 :                 oldrte->rellockmode = AccessShareLock;
    1081                 :           0 :                 oldrte->alias = makeAlias("old", NIL);
    1082                 :           0 :                 oldrte->eref = oldrte->alias;
    1083                 :           0 :                 oldrte->lateral = false;
    1084                 :           0 :                 oldrte->inh = false;
    1085                 :           0 :                 oldrte->inFromCl = true;
    1086                 :             : 
    1087                 :           0 :                 newrte = makeNode(RangeTblEntry);
    1088                 :           0 :                 newrte->rtekind = RTE_RELATION;
    1089                 :           0 :                 newrte->relid = trigrec->tgrelid;
    1090                 :           0 :                 newrte->relkind = relkind;
    1091                 :           0 :                 newrte->rellockmode = AccessShareLock;
    1092                 :           0 :                 newrte->alias = makeAlias("new", NIL);
    1093                 :           0 :                 newrte->eref = newrte->alias;
    1094                 :           0 :                 newrte->lateral = false;
    1095                 :           0 :                 newrte->inh = false;
    1096                 :           0 :                 newrte->inFromCl = true;
    1097                 :             : 
    1098                 :             :                 /* Build two-element rtable */
    1099                 :           0 :                 memset(&dpns, 0, sizeof(dpns));
    1100                 :           0 :                 dpns.rtable = list_make2(oldrte, newrte);
    1101                 :           0 :                 dpns.subplans = NIL;
    1102                 :           0 :                 dpns.ctes = NIL;
    1103                 :           0 :                 dpns.appendrels = NULL;
    1104                 :           0 :                 set_rtable_names(&dpns, NIL, NULL);
    1105                 :           0 :                 set_simple_column_names(&dpns);
    1106                 :             : 
    1107                 :             :                 /* Set up context with one-deep namespace stack */
    1108                 :           0 :                 context.buf = &buf;
    1109                 :           0 :                 context.namespaces = list_make1(&dpns);
    1110                 :           0 :                 context.resultDesc = NULL;
    1111                 :           0 :                 context.targetList = NIL;
    1112                 :           0 :                 context.windowClause = NIL;
    1113                 :           0 :                 context.varprefix = true;
    1114                 :           0 :                 context.prettyFlags = GET_PRETTY_FLAGS(pretty);
    1115                 :           0 :                 context.wrapColumn = WRAP_COLUMN_DEFAULT;
    1116                 :           0 :                 context.indentLevel = PRETTYINDENT_STD;
    1117                 :           0 :                 context.colNamesVisible = true;
    1118                 :           0 :                 context.inGroupBy = false;
    1119                 :           0 :                 context.varInOrderBy = false;
    1120                 :           0 :                 context.appendparents = NULL;
    1121                 :             : 
    1122                 :           0 :                 get_rule_expr(qual, &context, false);
    1123                 :             : 
    1124                 :           0 :                 appendStringInfoString(&buf, ") ");
    1125                 :           0 :         }
    1126                 :             : 
    1127                 :           0 :         appendStringInfo(&buf, "EXECUTE FUNCTION %s(",
    1128                 :           0 :                                          generate_function_name(trigrec->tgfoid, 0,
    1129                 :             :                                                                                         NIL, NULL,
    1130                 :             :                                                                                         false, NULL, false));
    1131                 :             : 
    1132         [ #  # ]:           0 :         if (trigrec->tgnargs > 0)
    1133                 :             :         {
    1134                 :           0 :                 char       *p;
    1135                 :           0 :                 int                     i;
    1136                 :             : 
    1137                 :           0 :                 value = fastgetattr(ht_trig, Anum_pg_trigger_tgargs,
    1138                 :           0 :                                                         tgrel->rd_att, &isnull);
    1139         [ #  # ]:           0 :                 if (isnull)
    1140   [ #  #  #  # ]:           0 :                         elog(ERROR, "tgargs is null for trigger %u", trigid);
    1141                 :           0 :                 p = (char *) VARDATA_ANY(DatumGetByteaPP(value));
    1142         [ #  # ]:           0 :                 for (i = 0; i < trigrec->tgnargs; i++)
    1143                 :             :                 {
    1144         [ #  # ]:           0 :                         if (i > 0)
    1145                 :           0 :                                 appendStringInfoString(&buf, ", ");
    1146                 :           0 :                         simple_quote_literal(&buf, p);
    1147                 :             :                         /* advance p to next string embedded in tgargs */
    1148         [ #  # ]:           0 :                         while (*p)
    1149                 :           0 :                                 p++;
    1150                 :           0 :                         p++;
    1151                 :           0 :                 }
    1152                 :           0 :         }
    1153                 :             : 
    1154                 :             :         /* We deliberately do not put semi-colon at end */
    1155                 :           0 :         appendStringInfoChar(&buf, ')');
    1156                 :             : 
    1157                 :             :         /* Clean up */
    1158                 :           0 :         systable_endscan(tgscan);
    1159                 :             : 
    1160                 :           0 :         table_close(tgrel, AccessShareLock);
    1161                 :             : 
    1162                 :           0 :         return buf.data;
    1163                 :           0 : }
    1164                 :             : 
    1165                 :             : /* ----------
    1166                 :             :  * pg_get_indexdef                      - Get the definition of an index
    1167                 :             :  *
    1168                 :             :  * In the extended version, there is a colno argument as well as pretty bool.
    1169                 :             :  *      if colno == 0, we want a complete index definition.
    1170                 :             :  *      if colno > 0, we only want the Nth index key's variable or expression.
    1171                 :             :  *
    1172                 :             :  * Note that the SQL-function versions of this omit any info about the
    1173                 :             :  * index tablespace; this is intentional because pg_dump wants it that way.
    1174                 :             :  * However pg_get_indexdef_string() includes the index tablespace.
    1175                 :             :  * ----------
    1176                 :             :  */
    1177                 :             : Datum
    1178                 :           0 : pg_get_indexdef(PG_FUNCTION_ARGS)
    1179                 :             : {
    1180                 :           0 :         Oid                     indexrelid = PG_GETARG_OID(0);
    1181                 :           0 :         int                     prettyFlags;
    1182                 :           0 :         char       *res;
    1183                 :             : 
    1184                 :           0 :         prettyFlags = PRETTYFLAG_INDENT;
    1185                 :             : 
    1186                 :           0 :         res = pg_get_indexdef_worker(indexrelid, 0, NULL,
    1187                 :             :                                                                  false, false,
    1188                 :             :                                                                  false, false,
    1189                 :           0 :                                                                  prettyFlags, true);
    1190                 :             : 
    1191         [ #  # ]:           0 :         if (res == NULL)
    1192                 :           0 :                 PG_RETURN_NULL();
    1193                 :             : 
    1194                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
    1195                 :           0 : }
    1196                 :             : 
    1197                 :             : Datum
    1198                 :           0 : pg_get_indexdef_ext(PG_FUNCTION_ARGS)
    1199                 :             : {
    1200                 :           0 :         Oid                     indexrelid = PG_GETARG_OID(0);
    1201                 :           0 :         int32           colno = PG_GETARG_INT32(1);
    1202                 :           0 :         bool            pretty = PG_GETARG_BOOL(2);
    1203                 :           0 :         int                     prettyFlags;
    1204                 :           0 :         char       *res;
    1205                 :             : 
    1206                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(pretty);
    1207                 :             : 
    1208                 :           0 :         res = pg_get_indexdef_worker(indexrelid, colno, NULL,
    1209                 :           0 :                                                                  colno != 0, false,
    1210                 :             :                                                                  false, false,
    1211                 :           0 :                                                                  prettyFlags, true);
    1212                 :             : 
    1213         [ #  # ]:           0 :         if (res == NULL)
    1214                 :           0 :                 PG_RETURN_NULL();
    1215                 :             : 
    1216                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
    1217                 :           0 : }
    1218                 :             : 
    1219                 :             : /*
    1220                 :             :  * Internal version for use by ALTER TABLE.
    1221                 :             :  * Includes a tablespace clause in the result.
    1222                 :             :  * Returns a palloc'd C string; no pretty-printing.
    1223                 :             :  */
    1224                 :             : char *
    1225                 :           0 : pg_get_indexdef_string(Oid indexrelid)
    1226                 :             : {
    1227                 :           0 :         return pg_get_indexdef_worker(indexrelid, 0, NULL,
    1228                 :             :                                                                   false, false,
    1229                 :             :                                                                   true, true,
    1230                 :             :                                                                   0, false);
    1231                 :             : }
    1232                 :             : 
    1233                 :             : /* Internal version that just reports the key-column definitions */
    1234                 :             : char *
    1235                 :           0 : pg_get_indexdef_columns(Oid indexrelid, bool pretty)
    1236                 :             : {
    1237                 :           0 :         int                     prettyFlags;
    1238                 :             : 
    1239                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(pretty);
    1240                 :             : 
    1241                 :           0 :         return pg_get_indexdef_worker(indexrelid, 0, NULL,
    1242                 :             :                                                                   true, true,
    1243                 :             :                                                                   false, false,
    1244                 :           0 :                                                                   prettyFlags, false);
    1245                 :           0 : }
    1246                 :             : 
    1247                 :             : /* Internal version, extensible with flags to control its behavior */
    1248                 :             : char *
    1249                 :           0 : pg_get_indexdef_columns_extended(Oid indexrelid, bits16 flags)
    1250                 :             : {
    1251                 :           0 :         bool            pretty = ((flags & RULE_INDEXDEF_PRETTY) != 0);
    1252                 :           0 :         bool            keys_only = ((flags & RULE_INDEXDEF_KEYS_ONLY) != 0);
    1253                 :           0 :         int                     prettyFlags;
    1254                 :             : 
    1255                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(pretty);
    1256                 :             : 
    1257                 :           0 :         return pg_get_indexdef_worker(indexrelid, 0, NULL,
    1258                 :           0 :                                                                   true, keys_only,
    1259                 :             :                                                                   false, false,
    1260                 :           0 :                                                                   prettyFlags, false);
    1261                 :           0 : }
    1262                 :             : 
    1263                 :             : /*
    1264                 :             :  * Internal workhorse to decompile an index definition.
    1265                 :             :  *
    1266                 :             :  * This is now used for exclusion constraints as well: if excludeOps is not
    1267                 :             :  * NULL then it points to an array of exclusion operator OIDs.
    1268                 :             :  */
    1269                 :             : static char *
    1270                 :           0 : pg_get_indexdef_worker(Oid indexrelid, int colno,
    1271                 :             :                                            const Oid *excludeOps,
    1272                 :             :                                            bool attrsOnly, bool keysOnly,
    1273                 :             :                                            bool showTblSpc, bool inherits,
    1274                 :             :                                            int prettyFlags, bool missing_ok)
    1275                 :             : {
    1276                 :             :         /* might want a separate isConstraint parameter later */
    1277                 :           0 :         bool            isConstraint = (excludeOps != NULL);
    1278                 :           0 :         HeapTuple       ht_idx;
    1279                 :           0 :         HeapTuple       ht_idxrel;
    1280                 :           0 :         HeapTuple       ht_am;
    1281                 :           0 :         Form_pg_index idxrec;
    1282                 :           0 :         Form_pg_class idxrelrec;
    1283                 :           0 :         Form_pg_am      amrec;
    1284                 :           0 :         const IndexAmRoutine *amroutine;
    1285                 :           0 :         List       *indexprs;
    1286                 :           0 :         ListCell   *indexpr_item;
    1287                 :           0 :         List       *context;
    1288                 :           0 :         Oid                     indrelid;
    1289                 :           0 :         int                     keyno;
    1290                 :           0 :         Datum           indcollDatum;
    1291                 :           0 :         Datum           indclassDatum;
    1292                 :           0 :         Datum           indoptionDatum;
    1293                 :           0 :         oidvector  *indcollation;
    1294                 :           0 :         oidvector  *indclass;
    1295                 :           0 :         int2vector *indoption;
    1296                 :           0 :         StringInfoData buf;
    1297                 :           0 :         char       *str;
    1298                 :           0 :         char       *sep;
    1299                 :             : 
    1300                 :             :         /*
    1301                 :             :          * Fetch the pg_index tuple by the Oid of the index
    1302                 :             :          */
    1303                 :           0 :         ht_idx = SearchSysCache1(INDEXRELID, ObjectIdGetDatum(indexrelid));
    1304         [ #  # ]:           0 :         if (!HeapTupleIsValid(ht_idx))
    1305                 :             :         {
    1306         [ #  # ]:           0 :                 if (missing_ok)
    1307                 :           0 :                         return NULL;
    1308   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for index %u", indexrelid);
    1309                 :           0 :         }
    1310                 :           0 :         idxrec = (Form_pg_index) GETSTRUCT(ht_idx);
    1311                 :             : 
    1312                 :           0 :         indrelid = idxrec->indrelid;
    1313         [ #  # ]:           0 :         Assert(indexrelid == idxrec->indexrelid);
    1314                 :             : 
    1315                 :             :         /* Must get indcollation, indclass, and indoption the hard way */
    1316                 :           0 :         indcollDatum = SysCacheGetAttrNotNull(INDEXRELID, ht_idx,
    1317                 :             :                                                                                   Anum_pg_index_indcollation);
    1318                 :           0 :         indcollation = (oidvector *) DatumGetPointer(indcollDatum);
    1319                 :             : 
    1320                 :           0 :         indclassDatum = SysCacheGetAttrNotNull(INDEXRELID, ht_idx,
    1321                 :             :                                                                                    Anum_pg_index_indclass);
    1322                 :           0 :         indclass = (oidvector *) DatumGetPointer(indclassDatum);
    1323                 :             : 
    1324                 :           0 :         indoptionDatum = SysCacheGetAttrNotNull(INDEXRELID, ht_idx,
    1325                 :             :                                                                                         Anum_pg_index_indoption);
    1326                 :           0 :         indoption = (int2vector *) DatumGetPointer(indoptionDatum);
    1327                 :             : 
    1328                 :             :         /*
    1329                 :             :          * Fetch the pg_class tuple of the index relation
    1330                 :             :          */
    1331                 :           0 :         ht_idxrel = SearchSysCache1(RELOID, ObjectIdGetDatum(indexrelid));
    1332         [ #  # ]:           0 :         if (!HeapTupleIsValid(ht_idxrel))
    1333   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for relation %u", indexrelid);
    1334                 :           0 :         idxrelrec = (Form_pg_class) GETSTRUCT(ht_idxrel);
    1335                 :             : 
    1336                 :             :         /*
    1337                 :             :          * Fetch the pg_am tuple of the index' access method
    1338                 :             :          */
    1339                 :           0 :         ht_am = SearchSysCache1(AMOID, ObjectIdGetDatum(idxrelrec->relam));
    1340         [ #  # ]:           0 :         if (!HeapTupleIsValid(ht_am))
    1341   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for access method %u",
    1342                 :             :                          idxrelrec->relam);
    1343                 :           0 :         amrec = (Form_pg_am) GETSTRUCT(ht_am);
    1344                 :             : 
    1345                 :             :         /* Fetch the index AM's API struct */
    1346                 :           0 :         amroutine = GetIndexAmRoutine(amrec->amhandler);
    1347                 :             : 
    1348                 :             :         /*
    1349                 :             :          * Get the index expressions, if any.  (NOTE: we do not use the relcache
    1350                 :             :          * versions of the expressions and predicate, because we want to display
    1351                 :             :          * non-const-folded expressions.)
    1352                 :             :          */
    1353         [ #  # ]:           0 :         if (!heap_attisnull(ht_idx, Anum_pg_index_indexprs, NULL))
    1354                 :             :         {
    1355                 :           0 :                 Datum           exprsDatum;
    1356                 :           0 :                 char       *exprsString;
    1357                 :             : 
    1358                 :           0 :                 exprsDatum = SysCacheGetAttrNotNull(INDEXRELID, ht_idx,
    1359                 :             :                                                                                         Anum_pg_index_indexprs);
    1360                 :           0 :                 exprsString = TextDatumGetCString(exprsDatum);
    1361                 :           0 :                 indexprs = (List *) stringToNode(exprsString);
    1362                 :           0 :                 pfree(exprsString);
    1363                 :           0 :         }
    1364                 :             :         else
    1365                 :           0 :                 indexprs = NIL;
    1366                 :             : 
    1367                 :           0 :         indexpr_item = list_head(indexprs);
    1368                 :             : 
    1369                 :           0 :         context = deparse_context_for(get_relation_name(indrelid), indrelid);
    1370                 :             : 
    1371                 :             :         /*
    1372                 :             :          * Start the index definition.  Note that the index's name should never be
    1373                 :             :          * schema-qualified, but the indexed rel's name may be.
    1374                 :             :          */
    1375                 :           0 :         initStringInfo(&buf);
    1376                 :             : 
    1377         [ #  # ]:           0 :         if (!attrsOnly)
    1378                 :             :         {
    1379         [ #  # ]:           0 :                 if (!isConstraint)
    1380                 :           0 :                         appendStringInfo(&buf, "CREATE %sINDEX %s ON %s%s USING %s (",
    1381                 :           0 :                                                          idxrec->indisunique ? "UNIQUE " : "",
    1382                 :           0 :                                                          quote_identifier(NameStr(idxrelrec->relname)),
    1383                 :           0 :                                                          idxrelrec->relkind == RELKIND_PARTITIONED_INDEX
    1384         [ #  # ]:           0 :                                                          && !inherits ? "ONLY " : "",
    1385         [ #  # ]:           0 :                                                          (prettyFlags & PRETTYFLAG_SCHEMA) ?
    1386                 :           0 :                                                          generate_relation_name(indrelid, NIL) :
    1387                 :           0 :                                                          generate_qualified_relation_name(indrelid),
    1388                 :           0 :                                                          quote_identifier(NameStr(amrec->amname)));
    1389                 :             :                 else                                    /* currently, must be EXCLUDE constraint */
    1390                 :           0 :                         appendStringInfo(&buf, "EXCLUDE USING %s (",
    1391                 :           0 :                                                          quote_identifier(NameStr(amrec->amname)));
    1392                 :           0 :         }
    1393                 :             : 
    1394                 :             :         /*
    1395                 :             :          * Report the indexed attributes
    1396                 :             :          */
    1397                 :           0 :         sep = "";
    1398         [ #  # ]:           0 :         for (keyno = 0; keyno < idxrec->indnatts; keyno++)
    1399                 :             :         {
    1400                 :           0 :                 AttrNumber      attnum = idxrec->indkey.values[keyno];
    1401                 :           0 :                 Oid                     keycoltype;
    1402                 :           0 :                 Oid                     keycolcollation;
    1403                 :             : 
    1404                 :             :                 /*
    1405                 :             :                  * Ignore non-key attributes if told to.
    1406                 :             :                  */
    1407   [ #  #  #  # ]:           0 :                 if (keysOnly && keyno >= idxrec->indnkeyatts)
    1408                 :           0 :                         break;
    1409                 :             : 
    1410                 :             :                 /* Otherwise, print INCLUDE to divide key and non-key attrs. */
    1411   [ #  #  #  # ]:           0 :                 if (!colno && keyno == idxrec->indnkeyatts)
    1412                 :             :                 {
    1413                 :           0 :                         appendStringInfoString(&buf, ") INCLUDE (");
    1414                 :           0 :                         sep = "";
    1415                 :           0 :                 }
    1416                 :             : 
    1417         [ #  # ]:           0 :                 if (!colno)
    1418                 :           0 :                         appendStringInfoString(&buf, sep);
    1419                 :           0 :                 sep = ", ";
    1420                 :             : 
    1421         [ #  # ]:           0 :                 if (attnum != 0)
    1422                 :             :                 {
    1423                 :             :                         /* Simple index column */
    1424                 :           0 :                         char       *attname;
    1425                 :           0 :                         int32           keycoltypmod;
    1426                 :             : 
    1427                 :           0 :                         attname = get_attname(indrelid, attnum, false);
    1428   [ #  #  #  # ]:           0 :                         if (!colno || colno == keyno + 1)
    1429                 :           0 :                                 appendStringInfoString(&buf, quote_identifier(attname));
    1430                 :           0 :                         get_atttypetypmodcoll(indrelid, attnum,
    1431                 :             :                                                                   &keycoltype, &keycoltypmod,
    1432                 :             :                                                                   &keycolcollation);
    1433                 :           0 :                 }
    1434                 :             :                 else
    1435                 :             :                 {
    1436                 :             :                         /* expressional index */
    1437                 :           0 :                         Node       *indexkey;
    1438                 :             : 
    1439         [ #  # ]:           0 :                         if (indexpr_item == NULL)
    1440   [ #  #  #  # ]:           0 :                                 elog(ERROR, "too few entries in indexprs list");
    1441                 :           0 :                         indexkey = (Node *) lfirst(indexpr_item);
    1442                 :           0 :                         indexpr_item = lnext(indexprs, indexpr_item);
    1443                 :             :                         /* Deparse */
    1444                 :           0 :                         str = deparse_expression_pretty(indexkey, context, false, false,
    1445                 :           0 :                                                                                         prettyFlags, 0);
    1446   [ #  #  #  # ]:           0 :                         if (!colno || colno == keyno + 1)
    1447                 :             :                         {
    1448                 :             :                                 /* Need parens if it's not a bare function call */
    1449         [ #  # ]:           0 :                                 if (looks_like_function(indexkey))
    1450                 :           0 :                                         appendStringInfoString(&buf, str);
    1451                 :             :                                 else
    1452                 :           0 :                                         appendStringInfo(&buf, "(%s)", str);
    1453                 :           0 :                         }
    1454                 :           0 :                         keycoltype = exprType(indexkey);
    1455                 :           0 :                         keycolcollation = exprCollation(indexkey);
    1456                 :           0 :                 }
    1457                 :             : 
    1458                 :             :                 /* Print additional decoration for (selected) key columns */
    1459   [ #  #  #  #  :           0 :                 if (!attrsOnly && keyno < idxrec->indnkeyatts &&
                   #  # ]
    1460         [ #  # ]:           0 :                         (!colno || colno == keyno + 1))
    1461                 :             :                 {
    1462                 :           0 :                         int16           opt = indoption->values[keyno];
    1463                 :           0 :                         Oid                     indcoll = indcollation->values[keyno];
    1464                 :           0 :                         Datum           attoptions = get_attoptions(indexrelid, keyno + 1);
    1465                 :           0 :                         bool            has_options = attoptions != (Datum) 0;
    1466                 :             : 
    1467                 :             :                         /* Add collation, if not default for column */
    1468   [ #  #  #  # ]:           0 :                         if (OidIsValid(indcoll) && indcoll != keycolcollation)
    1469                 :           0 :                                 appendStringInfo(&buf, " COLLATE %s",
    1470                 :           0 :                                                                  generate_collation_name((indcoll)));
    1471                 :             : 
    1472                 :             :                         /* Add the operator class name, if not default */
    1473                 :           0 :                         get_opclass_name(indclass->values[keyno],
    1474         [ #  # ]:           0 :                                                          has_options ? InvalidOid : keycoltype, &buf);
    1475                 :             : 
    1476         [ #  # ]:           0 :                         if (has_options)
    1477                 :             :                         {
    1478                 :           0 :                                 appendStringInfoString(&buf, " (");
    1479                 :           0 :                                 get_reloptions(&buf, attoptions);
    1480                 :           0 :                                 appendStringInfoChar(&buf, ')');
    1481                 :           0 :                         }
    1482                 :             : 
    1483                 :             :                         /* Add options if relevant */
    1484         [ #  # ]:           0 :                         if (amroutine->amcanorder)
    1485                 :             :                         {
    1486                 :             :                                 /* if it supports sort ordering, report DESC and NULLS opts */
    1487         [ #  # ]:           0 :                                 if (opt & INDOPTION_DESC)
    1488                 :             :                                 {
    1489                 :           0 :                                         appendStringInfoString(&buf, " DESC");
    1490                 :             :                                         /* NULLS FIRST is the default in this case */
    1491         [ #  # ]:           0 :                                         if (!(opt & INDOPTION_NULLS_FIRST))
    1492                 :           0 :                                                 appendStringInfoString(&buf, " NULLS LAST");
    1493                 :           0 :                                 }
    1494                 :             :                                 else
    1495                 :             :                                 {
    1496         [ #  # ]:           0 :                                         if (opt & INDOPTION_NULLS_FIRST)
    1497                 :           0 :                                                 appendStringInfoString(&buf, " NULLS FIRST");
    1498                 :             :                                 }
    1499                 :           0 :                         }
    1500                 :             : 
    1501                 :             :                         /* Add the exclusion operator if relevant */
    1502         [ #  # ]:           0 :                         if (excludeOps != NULL)
    1503                 :           0 :                                 appendStringInfo(&buf, " WITH %s",
    1504                 :           0 :                                                                  generate_operator_name(excludeOps[keyno],
    1505                 :           0 :                                                                                                                 keycoltype,
    1506                 :           0 :                                                                                                                 keycoltype));
    1507                 :           0 :                 }
    1508      [ #  #  # ]:           0 :         }
    1509                 :             : 
    1510         [ #  # ]:           0 :         if (!attrsOnly)
    1511                 :             :         {
    1512                 :           0 :                 appendStringInfoChar(&buf, ')');
    1513                 :             : 
    1514         [ #  # ]:           0 :                 if (idxrec->indnullsnotdistinct)
    1515                 :           0 :                         appendStringInfoString(&buf, " NULLS NOT DISTINCT");
    1516                 :             : 
    1517                 :             :                 /*
    1518                 :             :                  * If it has options, append "WITH (options)"
    1519                 :             :                  */
    1520                 :           0 :                 str = flatten_reloptions(indexrelid);
    1521         [ #  # ]:           0 :                 if (str)
    1522                 :             :                 {
    1523                 :           0 :                         appendStringInfo(&buf, " WITH (%s)", str);
    1524                 :           0 :                         pfree(str);
    1525                 :           0 :                 }
    1526                 :             : 
    1527                 :             :                 /*
    1528                 :             :                  * Print tablespace, but only if requested
    1529                 :             :                  */
    1530         [ #  # ]:           0 :                 if (showTblSpc)
    1531                 :             :                 {
    1532                 :           0 :                         Oid                     tblspc;
    1533                 :             : 
    1534                 :           0 :                         tblspc = get_rel_tablespace(indexrelid);
    1535         [ #  # ]:           0 :                         if (OidIsValid(tblspc))
    1536                 :             :                         {
    1537         [ #  # ]:           0 :                                 if (isConstraint)
    1538                 :           0 :                                         appendStringInfoString(&buf, " USING INDEX");
    1539                 :           0 :                                 appendStringInfo(&buf, " TABLESPACE %s",
    1540                 :           0 :                                                                  quote_identifier(get_tablespace_name(tblspc)));
    1541                 :           0 :                         }
    1542                 :           0 :                 }
    1543                 :             : 
    1544                 :             :                 /*
    1545                 :             :                  * If it's a partial index, decompile and append the predicate
    1546                 :             :                  */
    1547         [ #  # ]:           0 :                 if (!heap_attisnull(ht_idx, Anum_pg_index_indpred, NULL))
    1548                 :             :                 {
    1549                 :           0 :                         Node       *node;
    1550                 :           0 :                         Datum           predDatum;
    1551                 :           0 :                         char       *predString;
    1552                 :             : 
    1553                 :             :                         /* Convert text string to node tree */
    1554                 :           0 :                         predDatum = SysCacheGetAttrNotNull(INDEXRELID, ht_idx,
    1555                 :             :                                                                                            Anum_pg_index_indpred);
    1556                 :           0 :                         predString = TextDatumGetCString(predDatum);
    1557                 :           0 :                         node = (Node *) stringToNode(predString);
    1558                 :           0 :                         pfree(predString);
    1559                 :             : 
    1560                 :             :                         /* Deparse */
    1561                 :           0 :                         str = deparse_expression_pretty(node, context, false, false,
    1562                 :           0 :                                                                                         prettyFlags, 0);
    1563         [ #  # ]:           0 :                         if (isConstraint)
    1564                 :           0 :                                 appendStringInfo(&buf, " WHERE (%s)", str);
    1565                 :             :                         else
    1566                 :           0 :                                 appendStringInfo(&buf, " WHERE %s", str);
    1567                 :           0 :                 }
    1568                 :           0 :         }
    1569                 :             : 
    1570                 :             :         /* Clean up */
    1571                 :           0 :         ReleaseSysCache(ht_idx);
    1572                 :           0 :         ReleaseSysCache(ht_idxrel);
    1573                 :           0 :         ReleaseSysCache(ht_am);
    1574                 :             : 
    1575                 :           0 :         return buf.data;
    1576                 :           0 : }
    1577                 :             : 
    1578                 :             : /* ----------
    1579                 :             :  * pg_get_querydef
    1580                 :             :  *
    1581                 :             :  * Public entry point to deparse one query parsetree.
    1582                 :             :  * The pretty flags are determined by GET_PRETTY_FLAGS(pretty).
    1583                 :             :  *
    1584                 :             :  * The result is a palloc'd C string.
    1585                 :             :  * ----------
    1586                 :             :  */
    1587                 :             : char *
    1588                 :           0 : pg_get_querydef(Query *query, bool pretty)
    1589                 :             : {
    1590                 :           0 :         StringInfoData buf;
    1591                 :           0 :         int                     prettyFlags;
    1592                 :             : 
    1593                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(pretty);
    1594                 :             : 
    1595                 :           0 :         initStringInfo(&buf);
    1596                 :             : 
    1597                 :           0 :         get_query_def(query, &buf, NIL, NULL, true,
    1598                 :           0 :                                   prettyFlags, WRAP_COLUMN_DEFAULT, 0);
    1599                 :             : 
    1600                 :           0 :         return buf.data;
    1601                 :           0 : }
    1602                 :             : 
    1603                 :             : /*
    1604                 :             :  * pg_get_statisticsobjdef
    1605                 :             :  *              Get the definition of an extended statistics object
    1606                 :             :  */
    1607                 :             : Datum
    1608                 :           0 : pg_get_statisticsobjdef(PG_FUNCTION_ARGS)
    1609                 :             : {
    1610                 :           0 :         Oid                     statextid = PG_GETARG_OID(0);
    1611                 :           0 :         char       *res;
    1612                 :             : 
    1613                 :           0 :         res = pg_get_statisticsobj_worker(statextid, false, true);
    1614                 :             : 
    1615         [ #  # ]:           0 :         if (res == NULL)
    1616                 :           0 :                 PG_RETURN_NULL();
    1617                 :             : 
    1618                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
    1619                 :           0 : }
    1620                 :             : 
    1621                 :             : /*
    1622                 :             :  * Internal version for use by ALTER TABLE.
    1623                 :             :  * Returns a palloc'd C string; no pretty-printing.
    1624                 :             :  */
    1625                 :             : char *
    1626                 :           0 : pg_get_statisticsobjdef_string(Oid statextid)
    1627                 :             : {
    1628                 :           0 :         return pg_get_statisticsobj_worker(statextid, false, false);
    1629                 :             : }
    1630                 :             : 
    1631                 :             : /*
    1632                 :             :  * pg_get_statisticsobjdef_columns
    1633                 :             :  *              Get columns and expressions for an extended statistics object
    1634                 :             :  */
    1635                 :             : Datum
    1636                 :           0 : pg_get_statisticsobjdef_columns(PG_FUNCTION_ARGS)
    1637                 :             : {
    1638                 :           0 :         Oid                     statextid = PG_GETARG_OID(0);
    1639                 :           0 :         char       *res;
    1640                 :             : 
    1641                 :           0 :         res = pg_get_statisticsobj_worker(statextid, true, true);
    1642                 :             : 
    1643         [ #  # ]:           0 :         if (res == NULL)
    1644                 :           0 :                 PG_RETURN_NULL();
    1645                 :             : 
    1646                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
    1647                 :           0 : }
    1648                 :             : 
    1649                 :             : /*
    1650                 :             :  * Internal workhorse to decompile an extended statistics object.
    1651                 :             :  */
    1652                 :             : static char *
    1653                 :           0 : pg_get_statisticsobj_worker(Oid statextid, bool columns_only, bool missing_ok)
    1654                 :             : {
    1655                 :           0 :         Form_pg_statistic_ext statextrec;
    1656                 :           0 :         HeapTuple       statexttup;
    1657                 :           0 :         StringInfoData buf;
    1658                 :           0 :         int                     colno;
    1659                 :           0 :         char       *nsp;
    1660                 :           0 :         ArrayType  *arr;
    1661                 :           0 :         char       *enabled;
    1662                 :           0 :         Datum           datum;
    1663                 :           0 :         bool            ndistinct_enabled;
    1664                 :           0 :         bool            dependencies_enabled;
    1665                 :           0 :         bool            mcv_enabled;
    1666                 :           0 :         int                     i;
    1667                 :           0 :         List       *context;
    1668                 :           0 :         ListCell   *lc;
    1669                 :           0 :         List       *exprs = NIL;
    1670                 :           0 :         bool            has_exprs;
    1671                 :           0 :         int                     ncolumns;
    1672                 :             : 
    1673                 :           0 :         statexttup = SearchSysCache1(STATEXTOID, ObjectIdGetDatum(statextid));
    1674                 :             : 
    1675         [ #  # ]:           0 :         if (!HeapTupleIsValid(statexttup))
    1676                 :             :         {
    1677         [ #  # ]:           0 :                 if (missing_ok)
    1678                 :           0 :                         return NULL;
    1679   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for statistics object %u", statextid);
    1680                 :           0 :         }
    1681                 :             : 
    1682                 :             :         /* has the statistics expressions? */
    1683                 :           0 :         has_exprs = !heap_attisnull(statexttup, Anum_pg_statistic_ext_stxexprs, NULL);
    1684                 :             : 
    1685                 :           0 :         statextrec = (Form_pg_statistic_ext) GETSTRUCT(statexttup);
    1686                 :             : 
    1687                 :             :         /*
    1688                 :             :          * Get the statistics expressions, if any.  (NOTE: we do not use the
    1689                 :             :          * relcache versions of the expressions, because we want to display
    1690                 :             :          * non-const-folded expressions.)
    1691                 :             :          */
    1692         [ #  # ]:           0 :         if (has_exprs)
    1693                 :             :         {
    1694                 :           0 :                 Datum           exprsDatum;
    1695                 :           0 :                 char       *exprsString;
    1696                 :             : 
    1697                 :           0 :                 exprsDatum = SysCacheGetAttrNotNull(STATEXTOID, statexttup,
    1698                 :             :                                                                                         Anum_pg_statistic_ext_stxexprs);
    1699                 :           0 :                 exprsString = TextDatumGetCString(exprsDatum);
    1700                 :           0 :                 exprs = (List *) stringToNode(exprsString);
    1701                 :           0 :                 pfree(exprsString);
    1702                 :           0 :         }
    1703                 :             :         else
    1704                 :           0 :                 exprs = NIL;
    1705                 :             : 
    1706                 :             :         /* count the number of columns (attributes and expressions) */
    1707                 :           0 :         ncolumns = statextrec->stxkeys.dim1 + list_length(exprs);
    1708                 :             : 
    1709                 :           0 :         initStringInfo(&buf);
    1710                 :             : 
    1711         [ #  # ]:           0 :         if (!columns_only)
    1712                 :             :         {
    1713                 :           0 :                 nsp = get_namespace_name_or_temp(statextrec->stxnamespace);
    1714                 :           0 :                 appendStringInfo(&buf, "CREATE STATISTICS %s",
    1715                 :           0 :                                                  quote_qualified_identifier(nsp,
    1716                 :           0 :                                                                                                         NameStr(statextrec->stxname)));
    1717                 :             : 
    1718                 :             :                 /*
    1719                 :             :                  * Decode the stxkind column so that we know which stats types to
    1720                 :             :                  * print.
    1721                 :             :                  */
    1722                 :           0 :                 datum = SysCacheGetAttrNotNull(STATEXTOID, statexttup,
    1723                 :             :                                                                            Anum_pg_statistic_ext_stxkind);
    1724                 :           0 :                 arr = DatumGetArrayTypeP(datum);
    1725         [ #  # ]:           0 :                 if (ARR_NDIM(arr) != 1 ||
    1726                 :           0 :                         ARR_HASNULL(arr) ||
    1727                 :           0 :                         ARR_ELEMTYPE(arr) != CHAROID)
    1728   [ #  #  #  # ]:           0 :                         elog(ERROR, "stxkind is not a 1-D char array");
    1729         [ #  # ]:           0 :                 enabled = (char *) ARR_DATA_PTR(arr);
    1730                 :             : 
    1731                 :           0 :                 ndistinct_enabled = false;
    1732                 :           0 :                 dependencies_enabled = false;
    1733                 :           0 :                 mcv_enabled = false;
    1734                 :             : 
    1735         [ #  # ]:           0 :                 for (i = 0; i < ARR_DIMS(arr)[0]; i++)
    1736                 :             :                 {
    1737         [ #  # ]:           0 :                         if (enabled[i] == STATS_EXT_NDISTINCT)
    1738                 :           0 :                                 ndistinct_enabled = true;
    1739         [ #  # ]:           0 :                         else if (enabled[i] == STATS_EXT_DEPENDENCIES)
    1740                 :           0 :                                 dependencies_enabled = true;
    1741         [ #  # ]:           0 :                         else if (enabled[i] == STATS_EXT_MCV)
    1742                 :           0 :                                 mcv_enabled = true;
    1743                 :             : 
    1744                 :             :                         /* ignore STATS_EXT_EXPRESSIONS (it's built automatically) */
    1745                 :           0 :                 }
    1746                 :             : 
    1747                 :             :                 /*
    1748                 :             :                  * If any option is disabled, then we'll need to append the types
    1749                 :             :                  * clause to show which options are enabled.  We omit the types clause
    1750                 :             :                  * on purpose when all options are enabled, so a pg_dump/pg_restore
    1751                 :             :                  * will create all statistics types on a newer postgres version, if
    1752                 :             :                  * the statistics had all options enabled on the original version.
    1753                 :             :                  *
    1754                 :             :                  * But if the statistics is defined on just a single column, it has to
    1755                 :             :                  * be an expression statistics. In that case we don't need to specify
    1756                 :             :                  * kinds.
    1757                 :             :                  */
    1758   [ #  #  #  #  :           0 :                 if ((!ndistinct_enabled || !dependencies_enabled || !mcv_enabled) &&
                   #  # ]
    1759                 :           0 :                         (ncolumns > 1))
    1760                 :             :                 {
    1761                 :           0 :                         bool            gotone = false;
    1762                 :             : 
    1763                 :           0 :                         appendStringInfoString(&buf, " (");
    1764                 :             : 
    1765         [ #  # ]:           0 :                         if (ndistinct_enabled)
    1766                 :             :                         {
    1767                 :           0 :                                 appendStringInfoString(&buf, "ndistinct");
    1768                 :           0 :                                 gotone = true;
    1769                 :           0 :                         }
    1770                 :             : 
    1771         [ #  # ]:           0 :                         if (dependencies_enabled)
    1772                 :             :                         {
    1773                 :           0 :                                 appendStringInfo(&buf, "%sdependencies", gotone ? ", " : "");
    1774                 :           0 :                                 gotone = true;
    1775                 :           0 :                         }
    1776                 :             : 
    1777         [ #  # ]:           0 :                         if (mcv_enabled)
    1778                 :           0 :                                 appendStringInfo(&buf, "%smcv", gotone ? ", " : "");
    1779                 :             : 
    1780                 :           0 :                         appendStringInfoChar(&buf, ')');
    1781                 :           0 :                 }
    1782                 :             : 
    1783                 :           0 :                 appendStringInfoString(&buf, " ON ");
    1784                 :           0 :         }
    1785                 :             : 
    1786                 :             :         /* decode simple column references */
    1787         [ #  # ]:           0 :         for (colno = 0; colno < statextrec->stxkeys.dim1; colno++)
    1788                 :             :         {
    1789                 :           0 :                 AttrNumber      attnum = statextrec->stxkeys.values[colno];
    1790                 :           0 :                 char       *attname;
    1791                 :             : 
    1792         [ #  # ]:           0 :                 if (colno > 0)
    1793                 :           0 :                         appendStringInfoString(&buf, ", ");
    1794                 :             : 
    1795                 :           0 :                 attname = get_attname(statextrec->stxrelid, attnum, false);
    1796                 :             : 
    1797                 :           0 :                 appendStringInfoString(&buf, quote_identifier(attname));
    1798                 :           0 :         }
    1799                 :             : 
    1800                 :           0 :         context = deparse_context_for(get_relation_name(statextrec->stxrelid),
    1801                 :           0 :                                                                   statextrec->stxrelid);
    1802                 :             : 
    1803   [ #  #  #  #  :           0 :         foreach(lc, exprs)
                   #  # ]
    1804                 :             :         {
    1805                 :           0 :                 Node       *expr = (Node *) lfirst(lc);
    1806                 :           0 :                 char       *str;
    1807                 :           0 :                 int                     prettyFlags = PRETTYFLAG_PAREN;
    1808                 :             : 
    1809                 :           0 :                 str = deparse_expression_pretty(expr, context, false, false,
    1810                 :           0 :                                                                                 prettyFlags, 0);
    1811                 :             : 
    1812         [ #  # ]:           0 :                 if (colno > 0)
    1813                 :           0 :                         appendStringInfoString(&buf, ", ");
    1814                 :             : 
    1815                 :             :                 /* Need parens if it's not a bare function call */
    1816         [ #  # ]:           0 :                 if (looks_like_function(expr))
    1817                 :           0 :                         appendStringInfoString(&buf, str);
    1818                 :             :                 else
    1819                 :           0 :                         appendStringInfo(&buf, "(%s)", str);
    1820                 :             : 
    1821                 :           0 :                 colno++;
    1822                 :           0 :         }
    1823                 :             : 
    1824         [ #  # ]:           0 :         if (!columns_only)
    1825                 :           0 :                 appendStringInfo(&buf, " FROM %s",
    1826                 :           0 :                                                  generate_relation_name(statextrec->stxrelid, NIL));
    1827                 :             : 
    1828                 :           0 :         ReleaseSysCache(statexttup);
    1829                 :             : 
    1830                 :           0 :         return buf.data;
    1831                 :           0 : }
    1832                 :             : 
    1833                 :             : /*
    1834                 :             :  * Generate text array of expressions for statistics object.
    1835                 :             :  */
    1836                 :             : Datum
    1837                 :           0 : pg_get_statisticsobjdef_expressions(PG_FUNCTION_ARGS)
    1838                 :             : {
    1839                 :           0 :         Oid                     statextid = PG_GETARG_OID(0);
    1840                 :           0 :         Form_pg_statistic_ext statextrec;
    1841                 :           0 :         HeapTuple       statexttup;
    1842                 :           0 :         Datum           datum;
    1843                 :           0 :         List       *context;
    1844                 :           0 :         ListCell   *lc;
    1845                 :           0 :         List       *exprs = NIL;
    1846                 :           0 :         bool            has_exprs;
    1847                 :           0 :         char       *tmp;
    1848                 :           0 :         ArrayBuildState *astate = NULL;
    1849                 :             : 
    1850                 :           0 :         statexttup = SearchSysCache1(STATEXTOID, ObjectIdGetDatum(statextid));
    1851                 :             : 
    1852         [ #  # ]:           0 :         if (!HeapTupleIsValid(statexttup))
    1853                 :           0 :                 PG_RETURN_NULL();
    1854                 :             : 
    1855                 :             :         /* Does the stats object have expressions? */
    1856                 :           0 :         has_exprs = !heap_attisnull(statexttup, Anum_pg_statistic_ext_stxexprs, NULL);
    1857                 :             : 
    1858                 :             :         /* no expressions? we're done */
    1859         [ #  # ]:           0 :         if (!has_exprs)
    1860                 :             :         {
    1861                 :           0 :                 ReleaseSysCache(statexttup);
    1862                 :           0 :                 PG_RETURN_NULL();
    1863                 :           0 :         }
    1864                 :             : 
    1865                 :           0 :         statextrec = (Form_pg_statistic_ext) GETSTRUCT(statexttup);
    1866                 :             : 
    1867                 :             :         /*
    1868                 :             :          * Get the statistics expressions, and deparse them into text values.
    1869                 :             :          */
    1870                 :           0 :         datum = SysCacheGetAttrNotNull(STATEXTOID, statexttup,
    1871                 :             :                                                                    Anum_pg_statistic_ext_stxexprs);
    1872                 :           0 :         tmp = TextDatumGetCString(datum);
    1873                 :           0 :         exprs = (List *) stringToNode(tmp);
    1874                 :           0 :         pfree(tmp);
    1875                 :             : 
    1876                 :           0 :         context = deparse_context_for(get_relation_name(statextrec->stxrelid),
    1877                 :           0 :                                                                   statextrec->stxrelid);
    1878                 :             : 
    1879   [ #  #  #  #  :           0 :         foreach(lc, exprs)
                   #  # ]
    1880                 :             :         {
    1881                 :           0 :                 Node       *expr = (Node *) lfirst(lc);
    1882                 :           0 :                 char       *str;
    1883                 :           0 :                 int                     prettyFlags = PRETTYFLAG_INDENT;
    1884                 :             : 
    1885                 :           0 :                 str = deparse_expression_pretty(expr, context, false, false,
    1886                 :           0 :                                                                                 prettyFlags, 0);
    1887                 :             : 
    1888                 :           0 :                 astate = accumArrayResult(astate,
    1889                 :           0 :                                                                   PointerGetDatum(cstring_to_text(str)),
    1890                 :             :                                                                   false,
    1891                 :             :                                                                   TEXTOID,
    1892                 :           0 :                                                                   CurrentMemoryContext);
    1893                 :           0 :         }
    1894                 :             : 
    1895                 :           0 :         ReleaseSysCache(statexttup);
    1896                 :             : 
    1897                 :           0 :         PG_RETURN_DATUM(makeArrayResult(astate, CurrentMemoryContext));
    1898                 :           0 : }
    1899                 :             : 
    1900                 :             : /*
    1901                 :             :  * pg_get_partkeydef
    1902                 :             :  *
    1903                 :             :  * Returns the partition key specification, ie, the following:
    1904                 :             :  *
    1905                 :             :  * { RANGE | LIST | HASH } (column opt_collation opt_opclass [, ...])
    1906                 :             :  */
    1907                 :             : Datum
    1908                 :           0 : pg_get_partkeydef(PG_FUNCTION_ARGS)
    1909                 :             : {
    1910                 :           0 :         Oid                     relid = PG_GETARG_OID(0);
    1911                 :           0 :         char       *res;
    1912                 :             : 
    1913                 :           0 :         res = pg_get_partkeydef_worker(relid, PRETTYFLAG_INDENT, false, true);
    1914                 :             : 
    1915         [ #  # ]:           0 :         if (res == NULL)
    1916                 :           0 :                 PG_RETURN_NULL();
    1917                 :             : 
    1918                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
    1919                 :           0 : }
    1920                 :             : 
    1921                 :             : /* Internal version that just reports the column definitions */
    1922                 :             : char *
    1923                 :           0 : pg_get_partkeydef_columns(Oid relid, bool pretty)
    1924                 :             : {
    1925                 :           0 :         int                     prettyFlags;
    1926                 :             : 
    1927                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(pretty);
    1928                 :             : 
    1929                 :           0 :         return pg_get_partkeydef_worker(relid, prettyFlags, true, false);
    1930                 :           0 : }
    1931                 :             : 
    1932                 :             : /*
    1933                 :             :  * Internal workhorse to decompile a partition key definition.
    1934                 :             :  */
    1935                 :             : static char *
    1936                 :           0 : pg_get_partkeydef_worker(Oid relid, int prettyFlags,
    1937                 :             :                                                  bool attrsOnly, bool missing_ok)
    1938                 :             : {
    1939                 :           0 :         Form_pg_partitioned_table form;
    1940                 :           0 :         HeapTuple       tuple;
    1941                 :           0 :         oidvector  *partclass;
    1942                 :           0 :         oidvector  *partcollation;
    1943                 :           0 :         List       *partexprs;
    1944                 :           0 :         ListCell   *partexpr_item;
    1945                 :           0 :         List       *context;
    1946                 :           0 :         Datum           datum;
    1947                 :           0 :         StringInfoData buf;
    1948                 :           0 :         int                     keyno;
    1949                 :           0 :         char       *str;
    1950                 :           0 :         char       *sep;
    1951                 :             : 
    1952                 :           0 :         tuple = SearchSysCache1(PARTRELID, ObjectIdGetDatum(relid));
    1953         [ #  # ]:           0 :         if (!HeapTupleIsValid(tuple))
    1954                 :             :         {
    1955         [ #  # ]:           0 :                 if (missing_ok)
    1956                 :           0 :                         return NULL;
    1957   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for partition key of %u", relid);
    1958                 :           0 :         }
    1959                 :             : 
    1960                 :           0 :         form = (Form_pg_partitioned_table) GETSTRUCT(tuple);
    1961                 :             : 
    1962         [ #  # ]:           0 :         Assert(form->partrelid == relid);
    1963                 :             : 
    1964                 :             :         /* Must get partclass and partcollation the hard way */
    1965                 :           0 :         datum = SysCacheGetAttrNotNull(PARTRELID, tuple,
    1966                 :             :                                                                    Anum_pg_partitioned_table_partclass);
    1967                 :           0 :         partclass = (oidvector *) DatumGetPointer(datum);
    1968                 :             : 
    1969                 :           0 :         datum = SysCacheGetAttrNotNull(PARTRELID, tuple,
    1970                 :             :                                                                    Anum_pg_partitioned_table_partcollation);
    1971                 :           0 :         partcollation = (oidvector *) DatumGetPointer(datum);
    1972                 :             : 
    1973                 :             : 
    1974                 :             :         /*
    1975                 :             :          * Get the expressions, if any.  (NOTE: we do not use the relcache
    1976                 :             :          * versions of the expressions, because we want to display
    1977                 :             :          * non-const-folded expressions.)
    1978                 :             :          */
    1979         [ #  # ]:           0 :         if (!heap_attisnull(tuple, Anum_pg_partitioned_table_partexprs, NULL))
    1980                 :             :         {
    1981                 :           0 :                 Datum           exprsDatum;
    1982                 :           0 :                 char       *exprsString;
    1983                 :             : 
    1984                 :           0 :                 exprsDatum = SysCacheGetAttrNotNull(PARTRELID, tuple,
    1985                 :             :                                                                                         Anum_pg_partitioned_table_partexprs);
    1986                 :           0 :                 exprsString = TextDatumGetCString(exprsDatum);
    1987                 :           0 :                 partexprs = (List *) stringToNode(exprsString);
    1988                 :             : 
    1989         [ #  # ]:           0 :                 if (!IsA(partexprs, List))
    1990   [ #  #  #  # ]:           0 :                         elog(ERROR, "unexpected node type found in partexprs: %d",
    1991                 :             :                                  (int) nodeTag(partexprs));
    1992                 :             : 
    1993                 :           0 :                 pfree(exprsString);
    1994                 :           0 :         }
    1995                 :             :         else
    1996                 :           0 :                 partexprs = NIL;
    1997                 :             : 
    1998                 :           0 :         partexpr_item = list_head(partexprs);
    1999                 :           0 :         context = deparse_context_for(get_relation_name(relid), relid);
    2000                 :             : 
    2001                 :           0 :         initStringInfo(&buf);
    2002                 :             : 
    2003   [ #  #  #  # ]:           0 :         switch (form->partstrat)
    2004                 :             :         {
    2005                 :             :                 case PARTITION_STRATEGY_HASH:
    2006         [ #  # ]:           0 :                         if (!attrsOnly)
    2007                 :           0 :                                 appendStringInfoString(&buf, "HASH");
    2008                 :           0 :                         break;
    2009                 :             :                 case PARTITION_STRATEGY_LIST:
    2010         [ #  # ]:           0 :                         if (!attrsOnly)
    2011                 :           0 :                                 appendStringInfoString(&buf, "LIST");
    2012                 :           0 :                         break;
    2013                 :             :                 case PARTITION_STRATEGY_RANGE:
    2014         [ #  # ]:           0 :                         if (!attrsOnly)
    2015                 :           0 :                                 appendStringInfoString(&buf, "RANGE");
    2016                 :           0 :                         break;
    2017                 :             :                 default:
    2018   [ #  #  #  # ]:           0 :                         elog(ERROR, "unexpected partition strategy: %d",
    2019                 :             :                                  (int) form->partstrat);
    2020                 :           0 :         }
    2021                 :             : 
    2022         [ #  # ]:           0 :         if (!attrsOnly)
    2023                 :           0 :                 appendStringInfoString(&buf, " (");
    2024                 :           0 :         sep = "";
    2025         [ #  # ]:           0 :         for (keyno = 0; keyno < form->partnatts; keyno++)
    2026                 :             :         {
    2027                 :           0 :                 AttrNumber      attnum = form->partattrs.values[keyno];
    2028                 :           0 :                 Oid                     keycoltype;
    2029                 :           0 :                 Oid                     keycolcollation;
    2030                 :           0 :                 Oid                     partcoll;
    2031                 :             : 
    2032                 :           0 :                 appendStringInfoString(&buf, sep);
    2033                 :           0 :                 sep = ", ";
    2034         [ #  # ]:           0 :                 if (attnum != 0)
    2035                 :             :                 {
    2036                 :             :                         /* Simple attribute reference */
    2037                 :           0 :                         char       *attname;
    2038                 :           0 :                         int32           keycoltypmod;
    2039                 :             : 
    2040                 :           0 :                         attname = get_attname(relid, attnum, false);
    2041                 :           0 :                         appendStringInfoString(&buf, quote_identifier(attname));
    2042                 :           0 :                         get_atttypetypmodcoll(relid, attnum,
    2043                 :             :                                                                   &keycoltype, &keycoltypmod,
    2044                 :             :                                                                   &keycolcollation);
    2045                 :           0 :                 }
    2046                 :             :                 else
    2047                 :             :                 {
    2048                 :             :                         /* Expression */
    2049                 :           0 :                         Node       *partkey;
    2050                 :             : 
    2051         [ #  # ]:           0 :                         if (partexpr_item == NULL)
    2052   [ #  #  #  # ]:           0 :                                 elog(ERROR, "too few entries in partexprs list");
    2053                 :           0 :                         partkey = (Node *) lfirst(partexpr_item);
    2054                 :           0 :                         partexpr_item = lnext(partexprs, partexpr_item);
    2055                 :             : 
    2056                 :             :                         /* Deparse */
    2057                 :           0 :                         str = deparse_expression_pretty(partkey, context, false, false,
    2058                 :           0 :                                                                                         prettyFlags, 0);
    2059                 :             :                         /* Need parens if it's not a bare function call */
    2060         [ #  # ]:           0 :                         if (looks_like_function(partkey))
    2061                 :           0 :                                 appendStringInfoString(&buf, str);
    2062                 :             :                         else
    2063                 :           0 :                                 appendStringInfo(&buf, "(%s)", str);
    2064                 :             : 
    2065                 :           0 :                         keycoltype = exprType(partkey);
    2066                 :           0 :                         keycolcollation = exprCollation(partkey);
    2067                 :           0 :                 }
    2068                 :             : 
    2069                 :             :                 /* Add collation, if not default for column */
    2070                 :           0 :                 partcoll = partcollation->values[keyno];
    2071   [ #  #  #  #  :           0 :                 if (!attrsOnly && OidIsValid(partcoll) && partcoll != keycolcollation)
                   #  # ]
    2072                 :           0 :                         appendStringInfo(&buf, " COLLATE %s",
    2073                 :           0 :                                                          generate_collation_name((partcoll)));
    2074                 :             : 
    2075                 :             :                 /* Add the operator class name, if not default */
    2076         [ #  # ]:           0 :                 if (!attrsOnly)
    2077                 :           0 :                         get_opclass_name(partclass->values[keyno], keycoltype, &buf);
    2078                 :           0 :         }
    2079                 :             : 
    2080         [ #  # ]:           0 :         if (!attrsOnly)
    2081                 :           0 :                 appendStringInfoChar(&buf, ')');
    2082                 :             : 
    2083                 :             :         /* Clean up */
    2084                 :           0 :         ReleaseSysCache(tuple);
    2085                 :             : 
    2086                 :           0 :         return buf.data;
    2087                 :           0 : }
    2088                 :             : 
    2089                 :             : /*
    2090                 :             :  * pg_get_partition_constraintdef
    2091                 :             :  *
    2092                 :             :  * Returns partition constraint expression as a string for the input relation
    2093                 :             :  */
    2094                 :             : Datum
    2095                 :           0 : pg_get_partition_constraintdef(PG_FUNCTION_ARGS)
    2096                 :             : {
    2097                 :           0 :         Oid                     relationId = PG_GETARG_OID(0);
    2098                 :           0 :         Expr       *constr_expr;
    2099                 :           0 :         int                     prettyFlags;
    2100                 :           0 :         List       *context;
    2101                 :           0 :         char       *consrc;
    2102                 :             : 
    2103                 :           0 :         constr_expr = get_partition_qual_relid(relationId);
    2104                 :             : 
    2105                 :             :         /* Quick exit if no partition constraint */
    2106         [ #  # ]:           0 :         if (constr_expr == NULL)
    2107                 :           0 :                 PG_RETURN_NULL();
    2108                 :             : 
    2109                 :             :         /*
    2110                 :             :          * Deparse and return the constraint expression.
    2111                 :             :          */
    2112                 :           0 :         prettyFlags = PRETTYFLAG_INDENT;
    2113                 :           0 :         context = deparse_context_for(get_relation_name(relationId), relationId);
    2114                 :           0 :         consrc = deparse_expression_pretty((Node *) constr_expr, context, false,
    2115                 :           0 :                                                                            false, prettyFlags, 0);
    2116                 :             : 
    2117                 :           0 :         PG_RETURN_TEXT_P(string_to_text(consrc));
    2118                 :           0 : }
    2119                 :             : 
    2120                 :             : /*
    2121                 :             :  * pg_get_partconstrdef_string
    2122                 :             :  *
    2123                 :             :  * Returns the partition constraint as a C-string for the input relation, with
    2124                 :             :  * the given alias.  No pretty-printing.
    2125                 :             :  */
    2126                 :             : char *
    2127                 :           0 : pg_get_partconstrdef_string(Oid partitionId, char *aliasname)
    2128                 :             : {
    2129                 :           0 :         Expr       *constr_expr;
    2130                 :           0 :         List       *context;
    2131                 :             : 
    2132                 :           0 :         constr_expr = get_partition_qual_relid(partitionId);
    2133                 :           0 :         context = deparse_context_for(aliasname, partitionId);
    2134                 :             : 
    2135                 :           0 :         return deparse_expression((Node *) constr_expr, context, true, false);
    2136                 :           0 : }
    2137                 :             : 
    2138                 :             : /*
    2139                 :             :  * pg_get_constraintdef
    2140                 :             :  *
    2141                 :             :  * Returns the definition for the constraint, ie, everything that needs to
    2142                 :             :  * appear after "ALTER TABLE ... ADD CONSTRAINT <constraintname>".
    2143                 :             :  */
    2144                 :             : Datum
    2145                 :           0 : pg_get_constraintdef(PG_FUNCTION_ARGS)
    2146                 :             : {
    2147                 :           0 :         Oid                     constraintId = PG_GETARG_OID(0);
    2148                 :           0 :         int                     prettyFlags;
    2149                 :           0 :         char       *res;
    2150                 :             : 
    2151                 :           0 :         prettyFlags = PRETTYFLAG_INDENT;
    2152                 :             : 
    2153                 :           0 :         res = pg_get_constraintdef_worker(constraintId, false, prettyFlags, true);
    2154                 :             : 
    2155         [ #  # ]:           0 :         if (res == NULL)
    2156                 :           0 :                 PG_RETURN_NULL();
    2157                 :             : 
    2158                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
    2159                 :           0 : }
    2160                 :             : 
    2161                 :             : Datum
    2162                 :           0 : pg_get_constraintdef_ext(PG_FUNCTION_ARGS)
    2163                 :             : {
    2164                 :           0 :         Oid                     constraintId = PG_GETARG_OID(0);
    2165                 :           0 :         bool            pretty = PG_GETARG_BOOL(1);
    2166                 :           0 :         int                     prettyFlags;
    2167                 :           0 :         char       *res;
    2168                 :             : 
    2169                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(pretty);
    2170                 :             : 
    2171                 :           0 :         res = pg_get_constraintdef_worker(constraintId, false, prettyFlags, true);
    2172                 :             : 
    2173         [ #  # ]:           0 :         if (res == NULL)
    2174                 :           0 :                 PG_RETURN_NULL();
    2175                 :             : 
    2176                 :           0 :         PG_RETURN_TEXT_P(string_to_text(res));
    2177                 :           0 : }
    2178                 :             : 
    2179                 :             : /*
    2180                 :             :  * Internal version that returns a full ALTER TABLE ... ADD CONSTRAINT command
    2181                 :             :  */
    2182                 :             : char *
    2183                 :           0 : pg_get_constraintdef_command(Oid constraintId)
    2184                 :             : {
    2185                 :           0 :         return pg_get_constraintdef_worker(constraintId, true, 0, false);
    2186                 :             : }
    2187                 :             : 
    2188                 :             : /*
    2189                 :             :  * As of 9.4, we now use an MVCC snapshot for this.
    2190                 :             :  */
    2191                 :             : static char *
    2192                 :           0 : pg_get_constraintdef_worker(Oid constraintId, bool fullCommand,
    2193                 :             :                                                         int prettyFlags, bool missing_ok)
    2194                 :             : {
    2195                 :           0 :         HeapTuple       tup;
    2196                 :           0 :         Form_pg_constraint conForm;
    2197                 :           0 :         StringInfoData buf;
    2198                 :           0 :         SysScanDesc scandesc;
    2199                 :           0 :         ScanKeyData scankey[1];
    2200                 :           0 :         Snapshot        snapshot = RegisterSnapshot(GetTransactionSnapshot());
    2201                 :           0 :         Relation        relation = table_open(ConstraintRelationId, AccessShareLock);
    2202                 :             : 
    2203                 :           0 :         ScanKeyInit(&scankey[0],
    2204                 :             :                                 Anum_pg_constraint_oid,
    2205                 :             :                                 BTEqualStrategyNumber, F_OIDEQ,
    2206                 :           0 :                                 ObjectIdGetDatum(constraintId));
    2207                 :             : 
    2208                 :           0 :         scandesc = systable_beginscan(relation,
    2209                 :             :                                                                   ConstraintOidIndexId,
    2210                 :             :                                                                   true,
    2211                 :           0 :                                                                   snapshot,
    2212                 :             :                                                                   1,
    2213                 :           0 :                                                                   scankey);
    2214                 :             : 
    2215                 :             :         /*
    2216                 :             :          * We later use the tuple with SysCacheGetAttr() as if we had obtained it
    2217                 :             :          * via SearchSysCache, which works fine.
    2218                 :             :          */
    2219                 :           0 :         tup = systable_getnext(scandesc);
    2220                 :             : 
    2221                 :           0 :         UnregisterSnapshot(snapshot);
    2222                 :             : 
    2223         [ #  # ]:           0 :         if (!HeapTupleIsValid(tup))
    2224                 :             :         {
    2225         [ #  # ]:           0 :                 if (missing_ok)
    2226                 :             :                 {
    2227                 :           0 :                         systable_endscan(scandesc);
    2228                 :           0 :                         table_close(relation, AccessShareLock);
    2229                 :           0 :                         return NULL;
    2230                 :             :                 }
    2231   [ #  #  #  # ]:           0 :                 elog(ERROR, "could not find tuple for constraint %u", constraintId);
    2232                 :           0 :         }
    2233                 :             : 
    2234                 :           0 :         conForm = (Form_pg_constraint) GETSTRUCT(tup);
    2235                 :             : 
    2236                 :           0 :         initStringInfo(&buf);
    2237                 :             : 
    2238         [ #  # ]:           0 :         if (fullCommand)
    2239                 :             :         {
    2240         [ #  # ]:           0 :                 if (OidIsValid(conForm->conrelid))
    2241                 :             :                 {
    2242                 :             :                         /*
    2243                 :             :                          * Currently, callers want ALTER TABLE (without ONLY) for CHECK
    2244                 :             :                          * constraints, and other types of constraints don't inherit
    2245                 :             :                          * anyway so it doesn't matter whether we say ONLY or not. Someday
    2246                 :             :                          * we might need to let callers specify whether to put ONLY in the
    2247                 :             :                          * command.
    2248                 :             :                          */
    2249                 :           0 :                         appendStringInfo(&buf, "ALTER TABLE %s ADD CONSTRAINT %s ",
    2250                 :           0 :                                                          generate_qualified_relation_name(conForm->conrelid),
    2251                 :           0 :                                                          quote_identifier(NameStr(conForm->conname)));
    2252                 :           0 :                 }
    2253                 :             :                 else
    2254                 :             :                 {
    2255                 :             :                         /* Must be a domain constraint */
    2256         [ #  # ]:           0 :                         Assert(OidIsValid(conForm->contypid));
    2257                 :           0 :                         appendStringInfo(&buf, "ALTER DOMAIN %s ADD CONSTRAINT %s ",
    2258                 :           0 :                                                          generate_qualified_type_name(conForm->contypid),
    2259                 :           0 :                                                          quote_identifier(NameStr(conForm->conname)));
    2260                 :             :                 }
    2261                 :           0 :         }
    2262                 :             : 
    2263   [ #  #  #  #  :           0 :         switch (conForm->contype)
                #  #  # ]
    2264                 :             :         {
    2265                 :             :                 case CONSTRAINT_FOREIGN:
    2266                 :             :                         {
    2267                 :           0 :                                 Datum           val;
    2268                 :           0 :                                 bool            isnull;
    2269                 :           0 :                                 const char *string;
    2270                 :             : 
    2271                 :             :                                 /* Start off the constraint definition */
    2272                 :           0 :                                 appendStringInfoString(&buf, "FOREIGN KEY (");
    2273                 :             : 
    2274                 :             :                                 /* Fetch and build referencing-column list */
    2275                 :           0 :                                 val = SysCacheGetAttrNotNull(CONSTROID, tup,
    2276                 :             :                                                                                          Anum_pg_constraint_conkey);
    2277                 :             : 
    2278                 :             :                                 /* If it is a temporal foreign key then it uses PERIOD. */
    2279                 :           0 :                                 decompile_column_index_array(val, conForm->conrelid, conForm->conperiod, &buf);
    2280                 :             : 
    2281                 :             :                                 /* add foreign relation name */
    2282                 :           0 :                                 appendStringInfo(&buf, ") REFERENCES %s(",
    2283                 :           0 :                                                                  generate_relation_name(conForm->confrelid,
    2284                 :             :                                                                                                                 NIL));
    2285                 :             : 
    2286                 :             :                                 /* Fetch and build referenced-column list */
    2287                 :           0 :                                 val = SysCacheGetAttrNotNull(CONSTROID, tup,
    2288                 :             :                                                                                          Anum_pg_constraint_confkey);
    2289                 :             : 
    2290                 :           0 :                                 decompile_column_index_array(val, conForm->confrelid, conForm->conperiod, &buf);
    2291                 :             : 
    2292                 :           0 :                                 appendStringInfoChar(&buf, ')');
    2293                 :             : 
    2294                 :             :                                 /* Add match type */
    2295   [ #  #  #  # ]:           0 :                                 switch (conForm->confmatchtype)
    2296                 :             :                                 {
    2297                 :             :                                         case FKCONSTR_MATCH_FULL:
    2298                 :           0 :                                                 string = " MATCH FULL";
    2299                 :           0 :                                                 break;
    2300                 :             :                                         case FKCONSTR_MATCH_PARTIAL:
    2301                 :           0 :                                                 string = " MATCH PARTIAL";
    2302                 :           0 :                                                 break;
    2303                 :             :                                         case FKCONSTR_MATCH_SIMPLE:
    2304                 :           0 :                                                 string = "";
    2305                 :           0 :                                                 break;
    2306                 :             :                                         default:
    2307   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "unrecognized confmatchtype: %d",
    2308                 :             :                                                          conForm->confmatchtype);
    2309                 :           0 :                                                 string = "";  /* keep compiler quiet */
    2310                 :           0 :                                                 break;
    2311                 :             :                                 }
    2312                 :           0 :                                 appendStringInfoString(&buf, string);
    2313                 :             : 
    2314                 :             :                                 /* Add ON UPDATE and ON DELETE clauses, if needed */
    2315   [ #  #  #  #  :           0 :                                 switch (conForm->confupdtype)
                   #  # ]
    2316                 :             :                                 {
    2317                 :             :                                         case FKCONSTR_ACTION_NOACTION:
    2318                 :           0 :                                                 string = NULL;  /* suppress default */
    2319                 :           0 :                                                 break;
    2320                 :             :                                         case FKCONSTR_ACTION_RESTRICT:
    2321                 :           0 :                                                 string = "RESTRICT";
    2322                 :           0 :                                                 break;
    2323                 :             :                                         case FKCONSTR_ACTION_CASCADE:
    2324                 :           0 :                                                 string = "CASCADE";
    2325                 :           0 :                                                 break;
    2326                 :             :                                         case FKCONSTR_ACTION_SETNULL:
    2327                 :           0 :                                                 string = "SET NULL";
    2328                 :           0 :                                                 break;
    2329                 :             :                                         case FKCONSTR_ACTION_SETDEFAULT:
    2330                 :           0 :                                                 string = "SET DEFAULT";
    2331                 :           0 :                                                 break;
    2332                 :             :                                         default:
    2333   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "unrecognized confupdtype: %d",
    2334                 :             :                                                          conForm->confupdtype);
    2335                 :           0 :                                                 string = NULL;  /* keep compiler quiet */
    2336                 :           0 :                                                 break;
    2337                 :             :                                 }
    2338         [ #  # ]:           0 :                                 if (string)
    2339                 :           0 :                                         appendStringInfo(&buf, " ON UPDATE %s", string);
    2340                 :             : 
    2341   [ #  #  #  #  :           0 :                                 switch (conForm->confdeltype)
                   #  # ]
    2342                 :             :                                 {
    2343                 :             :                                         case FKCONSTR_ACTION_NOACTION:
    2344                 :           0 :                                                 string = NULL;  /* suppress default */
    2345                 :           0 :                                                 break;
    2346                 :             :                                         case FKCONSTR_ACTION_RESTRICT:
    2347                 :           0 :                                                 string = "RESTRICT";
    2348                 :           0 :                                                 break;
    2349                 :             :                                         case FKCONSTR_ACTION_CASCADE:
    2350                 :           0 :                                                 string = "CASCADE";
    2351                 :           0 :                                                 break;
    2352                 :             :                                         case FKCONSTR_ACTION_SETNULL:
    2353                 :           0 :                                                 string = "SET NULL";
    2354                 :           0 :                                                 break;
    2355                 :             :                                         case FKCONSTR_ACTION_SETDEFAULT:
    2356                 :           0 :                                                 string = "SET DEFAULT";
    2357                 :           0 :                                                 break;
    2358                 :             :                                         default:
    2359   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "unrecognized confdeltype: %d",
    2360                 :             :                                                          conForm->confdeltype);
    2361                 :           0 :                                                 string = NULL;  /* keep compiler quiet */
    2362                 :           0 :                                                 break;
    2363                 :             :                                 }
    2364         [ #  # ]:           0 :                                 if (string)
    2365                 :           0 :                                         appendStringInfo(&buf, " ON DELETE %s", string);
    2366                 :             : 
    2367                 :             :                                 /*
    2368                 :             :                                  * Add columns specified to SET NULL or SET DEFAULT if
    2369                 :             :                                  * provided.
    2370                 :             :                                  */
    2371                 :           0 :                                 val = SysCacheGetAttr(CONSTROID, tup,
    2372                 :             :                                                                           Anum_pg_constraint_confdelsetcols, &isnull);
    2373         [ #  # ]:           0 :                                 if (!isnull)
    2374                 :             :                                 {
    2375                 :           0 :                                         appendStringInfoString(&buf, " (");
    2376                 :           0 :                                         decompile_column_index_array(val, conForm->conrelid, false, &buf);
    2377                 :           0 :                                         appendStringInfoChar(&buf, ')');
    2378                 :           0 :                                 }
    2379                 :             : 
    2380                 :             :                                 break;
    2381                 :           0 :                         }
    2382                 :             :                 case CONSTRAINT_PRIMARY:
    2383                 :             :                 case CONSTRAINT_UNIQUE:
    2384                 :             :                         {
    2385                 :           0 :                                 Datum           val;
    2386                 :           0 :                                 Oid                     indexId;
    2387                 :           0 :                                 int                     keyatts;
    2388                 :           0 :                                 HeapTuple       indtup;
    2389                 :             : 
    2390                 :             :                                 /* Start off the constraint definition */
    2391         [ #  # ]:           0 :                                 if (conForm->contype == CONSTRAINT_PRIMARY)
    2392                 :           0 :                                         appendStringInfoString(&buf, "PRIMARY KEY ");
    2393                 :             :                                 else
    2394                 :           0 :                                         appendStringInfoString(&buf, "UNIQUE ");
    2395                 :             : 
    2396                 :           0 :                                 indexId = conForm->conindid;
    2397                 :             : 
    2398                 :           0 :                                 indtup = SearchSysCache1(INDEXRELID, ObjectIdGetDatum(indexId));
    2399         [ #  # ]:           0 :                                 if (!HeapTupleIsValid(indtup))
    2400   [ #  #  #  # ]:           0 :                                         elog(ERROR, "cache lookup failed for index %u", indexId);
    2401   [ #  #  #  # ]:           0 :                                 if (conForm->contype == CONSTRAINT_UNIQUE &&
    2402                 :           0 :                                         ((Form_pg_index) GETSTRUCT(indtup))->indnullsnotdistinct)
    2403                 :           0 :                                         appendStringInfoString(&buf, "NULLS NOT DISTINCT ");
    2404                 :             : 
    2405                 :           0 :                                 appendStringInfoChar(&buf, '(');
    2406                 :             : 
    2407                 :             :                                 /* Fetch and build target column list */
    2408                 :           0 :                                 val = SysCacheGetAttrNotNull(CONSTROID, tup,
    2409                 :             :                                                                                          Anum_pg_constraint_conkey);
    2410                 :             : 
    2411                 :           0 :                                 keyatts = decompile_column_index_array(val, conForm->conrelid, false, &buf);
    2412         [ #  # ]:           0 :                                 if (conForm->conperiod)
    2413                 :           0 :                                         appendStringInfoString(&buf, " WITHOUT OVERLAPS");
    2414                 :             : 
    2415                 :           0 :                                 appendStringInfoChar(&buf, ')');
    2416                 :             : 
    2417                 :             :                                 /* Build including column list (from pg_index.indkeys) */
    2418                 :           0 :                                 val = SysCacheGetAttrNotNull(INDEXRELID, indtup,
    2419                 :             :                                                                                          Anum_pg_index_indnatts);
    2420         [ #  # ]:           0 :                                 if (DatumGetInt32(val) > keyatts)
    2421                 :             :                                 {
    2422                 :           0 :                                         Datum           cols;
    2423                 :           0 :                                         Datum      *keys;
    2424                 :           0 :                                         int                     nKeys;
    2425                 :           0 :                                         int                     j;
    2426                 :             : 
    2427                 :           0 :                                         appendStringInfoString(&buf, " INCLUDE (");
    2428                 :             : 
    2429                 :           0 :                                         cols = SysCacheGetAttrNotNull(INDEXRELID, indtup,
    2430                 :             :                                                                                                   Anum_pg_index_indkey);
    2431                 :             : 
    2432                 :           0 :                                         deconstruct_array_builtin(DatumGetArrayTypeP(cols), INT2OID,
    2433                 :             :                                                                                           &keys, NULL, &nKeys);
    2434                 :             : 
    2435         [ #  # ]:           0 :                                         for (j = keyatts; j < nKeys; j++)
    2436                 :             :                                         {
    2437                 :           0 :                                                 char       *colName;
    2438                 :             : 
    2439                 :           0 :                                                 colName = get_attname(conForm->conrelid,
    2440                 :           0 :                                                                                           DatumGetInt16(keys[j]), false);
    2441         [ #  # ]:           0 :                                                 if (j > keyatts)
    2442                 :           0 :                                                         appendStringInfoString(&buf, ", ");
    2443                 :           0 :                                                 appendStringInfoString(&buf, quote_identifier(colName));
    2444                 :           0 :                                         }
    2445                 :             : 
    2446                 :           0 :                                         appendStringInfoChar(&buf, ')');
    2447                 :           0 :                                 }
    2448                 :           0 :                                 ReleaseSysCache(indtup);
    2449                 :             : 
    2450                 :             :                                 /* XXX why do we only print these bits if fullCommand? */
    2451   [ #  #  #  # ]:           0 :                                 if (fullCommand && OidIsValid(indexId))
    2452                 :             :                                 {
    2453                 :           0 :                                         char       *options = flatten_reloptions(indexId);
    2454                 :           0 :                                         Oid                     tblspc;
    2455                 :             : 
    2456         [ #  # ]:           0 :                                         if (options)
    2457                 :             :                                         {
    2458                 :           0 :                                                 appendStringInfo(&buf, " WITH (%s)", options);
    2459                 :           0 :                                                 pfree(options);
    2460                 :           0 :                                         }
    2461                 :             : 
    2462                 :             :                                         /*
    2463                 :             :                                          * Print the tablespace, unless it's the database default.
    2464                 :             :                                          * This is to help ALTER TABLE usage of this facility,
    2465                 :             :                                          * which needs this behavior to recreate exact catalog
    2466                 :             :                                          * state.
    2467                 :             :                                          */
    2468                 :           0 :                                         tblspc = get_rel_tablespace(indexId);
    2469         [ #  # ]:           0 :                                         if (OidIsValid(tblspc))
    2470                 :           0 :                                                 appendStringInfo(&buf, " USING INDEX TABLESPACE %s",
    2471                 :           0 :                                                                                  quote_identifier(get_tablespace_name(tblspc)));
    2472                 :           0 :                                 }
    2473                 :             : 
    2474                 :             :                                 break;
    2475                 :           0 :                         }
    2476                 :             :                 case CONSTRAINT_CHECK:
    2477                 :             :                         {
    2478                 :           0 :                                 Datum           val;
    2479                 :           0 :                                 char       *conbin;
    2480                 :           0 :                                 char       *consrc;
    2481                 :           0 :                                 Node       *expr;
    2482                 :           0 :                                 List       *context;
    2483                 :             : 
    2484                 :             :                                 /* Fetch constraint expression in parsetree form */
    2485                 :           0 :                                 val = SysCacheGetAttrNotNull(CONSTROID, tup,
    2486                 :             :                                                                                          Anum_pg_constraint_conbin);
    2487                 :             : 
    2488                 :           0 :                                 conbin = TextDatumGetCString(val);
    2489                 :           0 :                                 expr = stringToNode(conbin);
    2490                 :             : 
    2491                 :             :                                 /* Set up deparsing context for Var nodes in constraint */
    2492         [ #  # ]:           0 :                                 if (conForm->conrelid != InvalidOid)
    2493                 :             :                                 {
    2494                 :             :                                         /* relation constraint */
    2495                 :           0 :                                         context = deparse_context_for(get_relation_name(conForm->conrelid),
    2496                 :           0 :                                                                                                   conForm->conrelid);
    2497                 :           0 :                                 }
    2498                 :             :                                 else
    2499                 :             :                                 {
    2500                 :             :                                         /* domain constraint --- can't have Vars */
    2501                 :           0 :                                         context = NIL;
    2502                 :             :                                 }
    2503                 :             : 
    2504                 :           0 :                                 consrc = deparse_expression_pretty(expr, context, false, false,
    2505                 :           0 :                                                                                                    prettyFlags, 0);
    2506                 :             : 
    2507                 :             :                                 /*
    2508                 :             :                                  * Now emit the constraint definition, adding NO INHERIT if
    2509                 :             :                                  * necessary.
    2510                 :             :                                  *
    2511                 :             :                                  * There are cases where the constraint expression will be
    2512                 :             :                                  * fully parenthesized and we don't need the outer parens ...
    2513                 :             :                                  * but there are other cases where we do need 'em.  Be
    2514                 :             :                                  * conservative for now.
    2515                 :             :                                  *
    2516                 :             :                                  * Note that simply checking for leading '(' and trailing ')'
    2517                 :             :                                  * would NOT be good enough, consider "(x > 0) AND (y > 0)".
    2518                 :             :                                  */
    2519                 :           0 :                                 appendStringInfo(&buf, "CHECK (%s)%s",
    2520                 :           0 :                                                                  consrc,
    2521                 :           0 :                                                                  conForm->connoinherit ? " NO INHERIT" : "");
    2522                 :             :                                 break;
    2523                 :           0 :                         }
    2524                 :             :                 case CONSTRAINT_NOTNULL:
    2525                 :             :                         {
    2526         [ #  # ]:           0 :                                 if (conForm->conrelid)
    2527                 :             :                                 {
    2528                 :           0 :                                         AttrNumber      attnum;
    2529                 :             : 
    2530                 :           0 :                                         attnum = extractNotNullColumn(tup);
    2531                 :             : 
    2532                 :           0 :                                         appendStringInfo(&buf, "NOT NULL %s",
    2533                 :           0 :                                                                          quote_identifier(get_attname(conForm->conrelid,
    2534                 :           0 :                                                                                                                                   attnum, false)));
    2535         [ #  # ]:           0 :                                         if (((Form_pg_constraint) GETSTRUCT(tup))->connoinherit)
    2536                 :           0 :                                                 appendStringInfoString(&buf, " NO INHERIT");
    2537                 :           0 :                                 }
    2538         [ #  # ]:           0 :                                 else if (conForm->contypid)
    2539                 :             :                                 {
    2540                 :             :                                         /* conkey is null for domain not-null constraints */
    2541                 :           0 :                                         appendStringInfoString(&buf, "NOT NULL");
    2542                 :           0 :                                 }
    2543                 :           0 :                                 break;
    2544                 :             :                         }
    2545                 :             : 
    2546                 :             :                 case CONSTRAINT_TRIGGER:
    2547                 :             : 
    2548                 :             :                         /*
    2549                 :             :                          * There isn't an ALTER TABLE syntax for creating a user-defined
    2550                 :             :                          * constraint trigger, but it seems better to print something than
    2551                 :             :                          * throw an error; if we throw error then this function couldn't
    2552                 :             :                          * safely be applied to all rows of pg_constraint.
    2553                 :             :                          */
    2554                 :           0 :                         appendStringInfoString(&buf, "TRIGGER");
    2555                 :           0 :                         break;
    2556                 :             :                 case CONSTRAINT_EXCLUSION:
    2557                 :             :                         {
    2558                 :           0 :                                 Oid                     indexOid = conForm->conindid;
    2559                 :           0 :                                 Datum           val;
    2560                 :           0 :                                 Datum      *elems;
    2561                 :           0 :                                 int                     nElems;
    2562                 :           0 :                                 int                     i;
    2563                 :           0 :                                 Oid                *operators;
    2564                 :             : 
    2565                 :             :                                 /* Extract operator OIDs from the pg_constraint tuple */
    2566                 :           0 :                                 val = SysCacheGetAttrNotNull(CONSTROID, tup,
    2567                 :             :                                                                                          Anum_pg_constraint_conexclop);
    2568                 :             : 
    2569                 :           0 :                                 deconstruct_array_builtin(DatumGetArrayTypeP(val), OIDOID,
    2570                 :             :                                                                                   &elems, NULL, &nElems);
    2571                 :             : 
    2572                 :           0 :                                 operators = (Oid *) palloc(nElems * sizeof(Oid));
    2573         [ #  # ]:           0 :                                 for (i = 0; i < nElems; i++)
    2574                 :           0 :                                         operators[i] = DatumGetObjectId(elems[i]);
    2575                 :             : 
    2576                 :             :                                 /* pg_get_indexdef_worker does the rest */
    2577                 :             :                                 /* suppress tablespace because pg_dump wants it that way */
    2578                 :           0 :                                 appendStringInfoString(&buf,
    2579                 :           0 :                                                                            pg_get_indexdef_worker(indexOid,
    2580                 :             :                                                                                                                           0,
    2581                 :           0 :                                                                                                                           operators,
    2582                 :             :                                                                                                                           false,
    2583                 :             :                                                                                                                           false,
    2584                 :             :                                                                                                                           false,
    2585                 :             :                                                                                                                           false,
    2586                 :           0 :                                                                                                                           prettyFlags,
    2587                 :             :                                                                                                                           false));
    2588                 :             :                                 break;
    2589                 :           0 :                         }
    2590                 :             :                 default:
    2591   [ #  #  #  # ]:           0 :                         elog(ERROR, "invalid constraint type \"%c\"", conForm->contype);
    2592                 :           0 :                         break;
    2593                 :             :         }
    2594                 :             : 
    2595         [ #  # ]:           0 :         if (conForm->condeferrable)
    2596                 :           0 :                 appendStringInfoString(&buf, " DEFERRABLE");
    2597         [ #  # ]:           0 :         if (conForm->condeferred)
    2598                 :           0 :                 appendStringInfoString(&buf, " INITIALLY DEFERRED");
    2599                 :             : 
    2600                 :             :         /* Validated status is irrelevant when the constraint is NOT ENFORCED. */
    2601         [ #  # ]:           0 :         if (!conForm->conenforced)
    2602                 :           0 :                 appendStringInfoString(&buf, " NOT ENFORCED");
    2603         [ #  # ]:           0 :         else if (!conForm->convalidated)
    2604                 :           0 :                 appendStringInfoString(&buf, " NOT VALID");
    2605                 :             : 
    2606                 :             :         /* Cleanup */
    2607                 :           0 :         systable_endscan(scandesc);
    2608                 :           0 :         table_close(relation, AccessShareLock);
    2609                 :             : 
    2610                 :           0 :         return buf.data;
    2611                 :           0 : }
    2612                 :             : 
    2613                 :             : 
    2614                 :             : /*
    2615                 :             :  * Convert an int16[] Datum into a comma-separated list of column names
    2616                 :             :  * for the indicated relation; append the list to buf.  Returns the number
    2617                 :             :  * of keys.
    2618                 :             :  */
    2619                 :             : static int
    2620                 :           0 : decompile_column_index_array(Datum column_index_array, Oid relId,
    2621                 :             :                                                          bool withPeriod, StringInfo buf)
    2622                 :             : {
    2623                 :           0 :         Datum      *keys;
    2624                 :           0 :         int                     nKeys;
    2625                 :           0 :         int                     j;
    2626                 :             : 
    2627                 :             :         /* Extract data from array of int16 */
    2628                 :           0 :         deconstruct_array_builtin(DatumGetArrayTypeP(column_index_array), INT2OID,
    2629                 :             :                                                           &keys, NULL, &nKeys);
    2630                 :             : 
    2631         [ #  # ]:           0 :         for (j = 0; j < nKeys; j++)
    2632                 :             :         {
    2633                 :           0 :                 char       *colName;
    2634                 :             : 
    2635                 :           0 :                 colName = get_attname(relId, DatumGetInt16(keys[j]), false);
    2636                 :             : 
    2637         [ #  # ]:           0 :                 if (j == 0)
    2638                 :           0 :                         appendStringInfoString(buf, quote_identifier(colName));
    2639                 :             :                 else
    2640                 :           0 :                         appendStringInfo(buf, ", %s%s",
    2641         [ #  # ]:           0 :                                                          (withPeriod && j == nKeys - 1) ? "PERIOD " : "",
    2642                 :           0 :                                                          quote_identifier(colName));
    2643                 :           0 :         }
    2644                 :             : 
    2645                 :           0 :         return nKeys;
    2646                 :           0 : }
    2647                 :             : 
    2648                 :             : 
    2649                 :             : /* ----------
    2650                 :             :  * pg_get_expr                  - Decompile an expression tree
    2651                 :             :  *
    2652                 :             :  * Input: an expression tree in nodeToString form, and a relation OID
    2653                 :             :  *
    2654                 :             :  * Output: reverse-listed expression
    2655                 :             :  *
    2656                 :             :  * Currently, the expression can only refer to a single relation, namely
    2657                 :             :  * the one specified by the second parameter.  This is sufficient for
    2658                 :             :  * partial indexes, column default expressions, etc.  We also support
    2659                 :             :  * Var-free expressions, for which the OID can be InvalidOid.
    2660                 :             :  *
    2661                 :             :  * If the OID is nonzero but not actually valid, don't throw an error,
    2662                 :             :  * just return NULL.  This is a bit questionable, but it's what we've
    2663                 :             :  * done historically, and it can help avoid unwanted failures when
    2664                 :             :  * examining catalog entries for just-deleted relations.
    2665                 :             :  *
    2666                 :             :  * We expect this function to work, or throw a reasonably clean error,
    2667                 :             :  * for any node tree that can appear in a catalog pg_node_tree column.
    2668                 :             :  * Query trees, such as those appearing in pg_rewrite.ev_action, are
    2669                 :             :  * not supported.  Nor are expressions in more than one relation, which
    2670                 :             :  * can appear in places like pg_rewrite.ev_qual.
    2671                 :             :  * ----------
    2672                 :             :  */
    2673                 :             : Datum
    2674                 :           0 : pg_get_expr(PG_FUNCTION_ARGS)
    2675                 :             : {
    2676                 :           0 :         text       *expr = PG_GETARG_TEXT_PP(0);
    2677                 :           0 :         Oid                     relid = PG_GETARG_OID(1);
    2678                 :           0 :         text       *result;
    2679                 :           0 :         int                     prettyFlags;
    2680                 :             : 
    2681                 :           0 :         prettyFlags = PRETTYFLAG_INDENT;
    2682                 :             : 
    2683                 :           0 :         result = pg_get_expr_worker(expr, relid, prettyFlags);
    2684         [ #  # ]:           0 :         if (result)
    2685                 :           0 :                 PG_RETURN_TEXT_P(result);
    2686                 :             :         else
    2687                 :           0 :                 PG_RETURN_NULL();
    2688         [ #  # ]:           0 : }
    2689                 :             : 
    2690                 :             : Datum
    2691                 :           0 : pg_get_expr_ext(PG_FUNCTION_ARGS)
    2692                 :             : {
    2693                 :           0 :         text       *expr = PG_GETARG_TEXT_PP(0);
    2694                 :           0 :         Oid                     relid = PG_GETARG_OID(1);
    2695                 :           0 :         bool            pretty = PG_GETARG_BOOL(2);
    2696                 :           0 :         text       *result;
    2697                 :           0 :         int                     prettyFlags;
    2698                 :             : 
    2699                 :           0 :         prettyFlags = GET_PRETTY_FLAGS(pretty);
    2700                 :             : 
    2701                 :           0 :         result = pg_get_expr_worker(expr, relid, prettyFlags);
    2702         [ #  # ]:           0 :         if (result)
    2703                 :           0 :                 PG_RETURN_TEXT_P(result);
    2704                 :             :         else
    2705                 :           0 :                 PG_RETURN_NULL();
    2706         [ #  # ]:           0 : }
    2707                 :             : 
    2708                 :             : static text *
    2709                 :           0 : pg_get_expr_worker(text *expr, Oid relid, int prettyFlags)
    2710                 :             : {
    2711                 :           0 :         Node       *node;
    2712                 :           0 :         Node       *tst;
    2713                 :           0 :         Relids          relids;
    2714                 :           0 :         List       *context;
    2715                 :           0 :         char       *exprstr;
    2716                 :           0 :         Relation        rel = NULL;
    2717                 :           0 :         char       *str;
    2718                 :             : 
    2719                 :             :         /* Convert input pg_node_tree (really TEXT) object to C string */
    2720                 :           0 :         exprstr = text_to_cstring(expr);
    2721                 :             : 
    2722                 :             :         /* Convert expression to node tree */
    2723                 :           0 :         node = (Node *) stringToNode(exprstr);
    2724                 :             : 
    2725                 :           0 :         pfree(exprstr);
    2726                 :             : 
    2727                 :             :         /*
    2728                 :             :          * Throw error if the input is a querytree rather than an expression tree.
    2729                 :             :          * While we could support queries here, there seems no very good reason
    2730                 :             :          * to.  In most such catalog columns, we'll see a List of Query nodes, or
    2731                 :             :          * even nested Lists, so drill down to a non-List node before checking.
    2732                 :             :          */
    2733                 :           0 :         tst = node;
    2734   [ #  #  #  # ]:           0 :         while (tst && IsA(tst, List))
    2735                 :           0 :                 tst = linitial((List *) tst);
    2736   [ #  #  #  # ]:           0 :         if (tst && IsA(tst, Query))
    2737   [ #  #  #  # ]:           0 :                 ereport(ERROR,
    2738                 :             :                                 (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
    2739                 :             :                                  errmsg("input is a query, not an expression")));
    2740                 :             : 
    2741                 :             :         /*
    2742                 :             :          * Throw error if the expression contains Vars we won't be able to
    2743                 :             :          * deparse.
    2744                 :             :          */
    2745                 :           0 :         relids = pull_varnos(NULL, node);
    2746         [ #  # ]:           0 :         if (OidIsValid(relid))
    2747                 :             :         {
    2748         [ #  # ]:           0 :                 if (!bms_is_subset(relids, bms_make_singleton(1)))
    2749   [ #  #  #  # ]:           0 :                         ereport(ERROR,
    2750                 :             :                                         (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
    2751                 :             :                                          errmsg("expression contains variables of more than one relation")));
    2752                 :           0 :         }
    2753                 :             :         else
    2754                 :             :         {
    2755         [ #  # ]:           0 :                 if (!bms_is_empty(relids))
    2756   [ #  #  #  # ]:           0 :                         ereport(ERROR,
    2757                 :             :                                         (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
    2758                 :             :                                          errmsg("expression contains variables")));
    2759                 :             :         }
    2760                 :             : 
    2761                 :             :         /*
    2762                 :             :          * Prepare deparse context if needed.  If we are deparsing with a relid,
    2763                 :             :          * we need to transiently open and lock the rel, to make sure it won't go
    2764                 :             :          * away underneath us.  (set_relation_column_names would lock it anyway,
    2765                 :             :          * so this isn't really introducing any new behavior.)
    2766                 :             :          */
    2767         [ #  # ]:           0 :         if (OidIsValid(relid))
    2768                 :             :         {
    2769                 :           0 :                 rel = try_relation_open(relid, AccessShareLock);
    2770         [ #  # ]:           0 :                 if (rel == NULL)
    2771                 :           0 :                         return NULL;
    2772                 :           0 :                 context = deparse_context_for(RelationGetRelationName(rel), relid);
    2773                 :           0 :         }
    2774                 :             :         else
    2775                 :           0 :                 context = NIL;
    2776                 :             : 
    2777                 :             :         /* Deparse */
    2778                 :           0 :         str = deparse_expression_pretty(node, context, false, false,
    2779                 :           0 :                                                                         prettyFlags, 0);
    2780                 :             : 
    2781         [ #  # ]:           0 :         if (rel != NULL)
    2782                 :           0 :                 relation_close(rel, AccessShareLock);
    2783                 :             : 
    2784                 :           0 :         return string_to_text(str);
    2785                 :           0 : }
    2786                 :             : 
    2787                 :             : 
    2788                 :             : /* ----------
    2789                 :             :  * pg_get_userbyid              - Get a user name by roleid and
    2790                 :             :  *                                fallback to 'unknown (OID=n)'
    2791                 :             :  * ----------
    2792                 :             :  */
    2793                 :             : Datum
    2794                 :           0 : pg_get_userbyid(PG_FUNCTION_ARGS)
    2795                 :             : {
    2796                 :           0 :         Oid                     roleid = PG_GETARG_OID(0);
    2797                 :           0 :         Name            result;
    2798                 :           0 :         HeapTuple       roletup;
    2799                 :           0 :         Form_pg_authid role_rec;
    2800                 :             : 
    2801                 :             :         /*
    2802                 :             :          * Allocate space for the result
    2803                 :             :          */
    2804                 :           0 :         result = (Name) palloc(NAMEDATALEN);
    2805                 :           0 :         memset(NameStr(*result), 0, NAMEDATALEN);
    2806                 :             : 
    2807                 :             :         /*
    2808                 :             :          * Get the pg_authid entry and print the result
    2809                 :             :          */
    2810                 :           0 :         roletup = SearchSysCache1(AUTHOID, ObjectIdGetDatum(roleid));
    2811         [ #  # ]:           0 :         if (HeapTupleIsValid(roletup))
    2812                 :             :         {
    2813                 :           0 :                 role_rec = (Form_pg_authid) GETSTRUCT(roletup);
    2814                 :           0 :                 *result = role_rec->rolname;
    2815                 :           0 :                 ReleaseSysCache(roletup);
    2816                 :           0 :         }
    2817                 :             :         else
    2818                 :           0 :                 sprintf(NameStr(*result), "unknown (OID=%u)", roleid);
    2819                 :             : 
    2820                 :           0 :         PG_RETURN_NAME(result);
    2821                 :           0 : }
    2822                 :             : 
    2823                 :             : 
    2824                 :             : /*
    2825                 :             :  * pg_get_serial_sequence
    2826                 :             :  *              Get the name of the sequence used by an identity or serial column,
    2827                 :             :  *              formatted suitably for passing to setval, nextval or currval.
    2828                 :             :  *              First parameter is not treated as double-quoted, second parameter
    2829                 :             :  *              is --- see documentation for reason.
    2830                 :             :  */
    2831                 :             : Datum
    2832                 :           0 : pg_get_serial_sequence(PG_FUNCTION_ARGS)
    2833                 :             : {
    2834                 :           0 :         text       *tablename = PG_GETARG_TEXT_PP(0);
    2835                 :           0 :         text       *columnname = PG_GETARG_TEXT_PP(1);
    2836                 :           0 :         RangeVar   *tablerv;
    2837                 :           0 :         Oid                     tableOid;
    2838                 :           0 :         char       *column;
    2839                 :           0 :         AttrNumber      attnum;
    2840                 :           0 :         Oid                     sequenceId = InvalidOid;
    2841                 :           0 :         Relation        depRel;
    2842                 :           0 :         ScanKeyData key[3];
    2843                 :           0 :         SysScanDesc scan;
    2844                 :           0 :         HeapTuple       tup;
    2845                 :             : 
    2846                 :             :         /* Look up table name.  Can't lock it - we might not have privileges. */
    2847                 :           0 :         tablerv = makeRangeVarFromNameList(textToQualifiedNameList(tablename));
    2848                 :           0 :         tableOid = RangeVarGetRelid(tablerv, NoLock, false);
    2849                 :             : 
    2850                 :             :         /* Get the number of the column */
    2851                 :           0 :         column = text_to_cstring(columnname);
    2852                 :             : 
    2853                 :           0 :         attnum = get_attnum(tableOid, column);
    2854         [ #  # ]:           0 :         if (attnum == InvalidAttrNumber)
    2855   [ #  #  #  # ]:           0 :                 ereport(ERROR,
    2856                 :             :                                 (errcode(ERRCODE_UNDEFINED_COLUMN),
    2857                 :             :                                  errmsg("column \"%s\" of relation \"%s\" does not exist",
    2858                 :             :                                                 column, tablerv->relname)));
    2859                 :             : 
    2860                 :             :         /* Search the dependency table for the dependent sequence */
    2861                 :           0 :         depRel = table_open(DependRelationId, AccessShareLock);
    2862                 :             : 
    2863                 :           0 :         ScanKeyInit(&key[0],
    2864                 :             :                                 Anum_pg_depend_refclassid,
    2865                 :             :                                 BTEqualStrategyNumber, F_OIDEQ,
    2866                 :           0 :                                 ObjectIdGetDatum(RelationRelationId));
    2867                 :           0 :         ScanKeyInit(&key[1],
    2868                 :             :                                 Anum_pg_depend_refobjid,
    2869                 :             :                                 BTEqualStrategyNumber, F_OIDEQ,
    2870                 :           0 :                                 ObjectIdGetDatum(tableOid));
    2871                 :           0 :         ScanKeyInit(&key[2],
    2872                 :             :                                 Anum_pg_depend_refobjsubid,
    2873                 :             :                                 BTEqualStrategyNumber, F_INT4EQ,
    2874                 :           0 :                                 Int32GetDatum(attnum));
    2875                 :             : 
    2876                 :           0 :         scan = systable_beginscan(depRel, DependReferenceIndexId, true,
    2877                 :           0 :                                                           NULL, 3, key);
    2878                 :             : 
    2879         [ #  # ]:           0 :         while (HeapTupleIsValid(tup = systable_getnext(scan)))
    2880                 :             :         {
    2881                 :           0 :                 Form_pg_depend deprec = (Form_pg_depend) GETSTRUCT(tup);
    2882                 :             : 
    2883                 :             :                 /*
    2884                 :             :                  * Look for an auto dependency (serial column) or internal dependency
    2885                 :             :                  * (identity column) of a sequence on a column.  (We need the relkind
    2886                 :             :                  * test because indexes can also have auto dependencies on columns.)
    2887                 :             :                  */
    2888         [ #  # ]:           0 :                 if (deprec->classid == RelationRelationId &&
    2889         [ #  # ]:           0 :                         deprec->objsubid == 0 &&
    2890         [ #  # ]:           0 :                         (deprec->deptype == DEPENDENCY_AUTO ||
    2891         [ #  # ]:           0 :                          deprec->deptype == DEPENDENCY_INTERNAL) &&
    2892                 :           0 :                         get_rel_relkind(deprec->objid) == RELKIND_SEQUENCE)
    2893                 :             :                 {
    2894                 :           0 :                         sequenceId = deprec->objid;
    2895                 :           0 :                         break;
    2896                 :             :                 }
    2897         [ #  # ]:           0 :         }
    2898                 :             : 
    2899                 :           0 :         systable_endscan(scan);
    2900                 :           0 :         table_close(depRel, AccessShareLock);
    2901                 :             : 
    2902         [ #  # ]:           0 :         if (OidIsValid(sequenceId))
    2903                 :             :         {
    2904                 :           0 :                 char       *result;
    2905                 :             : 
    2906                 :           0 :                 result = generate_qualified_relation_name(sequenceId);
    2907                 :             : 
    2908                 :           0 :                 PG_RETURN_TEXT_P(string_to_text(result));
    2909                 :           0 :         }
    2910                 :             : 
    2911                 :           0 :         PG_RETURN_NULL();
    2912                 :           0 : }
    2913                 :             : 
    2914                 :             : 
    2915                 :             : /*
    2916                 :             :  * pg_get_functiondef
    2917                 :             :  *              Returns the complete "CREATE OR REPLACE FUNCTION ..." statement for
    2918                 :             :  *              the specified function.
    2919                 :             :  *
    2920                 :             :  * Note: if you change the output format of this function, be careful not
    2921                 :             :  * to break psql's rules (in \ef and \sf) for identifying the start of the
    2922                 :             :  * function body.  To wit: the function body starts on a line that begins with
    2923                 :             :  * "AS ", "BEGIN ", or "RETURN ", and no preceding line will look like that.
    2924                 :             :  */
    2925                 :             : Datum
    2926                 :           0 : pg_get_functiondef(PG_FUNCTION_ARGS)
    2927                 :             : {
    2928                 :           0 :         Oid                     funcid = PG_GETARG_OID(0);
    2929                 :           0 :         StringInfoData buf;
    2930                 :           0 :         StringInfoData dq;
    2931                 :           0 :         HeapTuple       proctup;
    2932                 :           0 :         Form_pg_proc proc;
    2933                 :           0 :         bool            isfunction;
    2934                 :           0 :         Datum           tmp;
    2935                 :           0 :         bool            isnull;
    2936                 :           0 :         const char *prosrc;
    2937                 :           0 :         const char *name;
    2938                 :           0 :         const char *nsp;
    2939                 :           0 :         float4          procost;
    2940                 :           0 :         int                     oldlen;
    2941                 :             : 
    2942                 :           0 :         initStringInfo(&buf);
    2943                 :             : 
    2944                 :             :         /* Look up the function */
    2945                 :           0 :         proctup = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
    2946         [ #  # ]:           0 :         if (!HeapTupleIsValid(proctup))
    2947                 :           0 :                 PG_RETURN_NULL();
    2948                 :             : 
    2949                 :           0 :         proc = (Form_pg_proc) GETSTRUCT(proctup);
    2950                 :           0 :         name = NameStr(proc->proname);
    2951                 :             : 
    2952         [ #  # ]:           0 :         if (proc->prokind == PROKIND_AGGREGATE)
    2953   [ #  #  #  # ]:           0 :                 ereport(ERROR,
    2954                 :             :                                 (errcode(ERRCODE_WRONG_OBJECT_TYPE),
    2955                 :             :                                  errmsg("\"%s\" is an aggregate function", name)));
    2956                 :             : 
    2957                 :           0 :         isfunction = (proc->prokind != PROKIND_PROCEDURE);
    2958                 :             : 
    2959                 :             :         /*
    2960                 :             :          * We always qualify the function name, to ensure the right function gets
    2961                 :             :          * replaced.
    2962                 :             :          */
    2963                 :           0 :         nsp = get_namespace_name_or_temp(proc->pronamespace);
    2964                 :           0 :         appendStringInfo(&buf, "CREATE OR REPLACE %s %s(",
    2965                 :           0 :                                          isfunction ? "FUNCTION" : "PROCEDURE",
    2966                 :           0 :                                          quote_qualified_identifier(nsp, name));
    2967                 :           0 :         (void) print_function_arguments(&buf, proctup, false, true);
    2968                 :           0 :         appendStringInfoString(&buf, ")\n");
    2969         [ #  # ]:           0 :         if (isfunction)
    2970                 :             :         {
    2971                 :           0 :                 appendStringInfoString(&buf, " RETURNS ");
    2972                 :           0 :                 print_function_rettype(&buf, proctup);
    2973                 :           0 :                 appendStringInfoChar(&buf, '\n');
    2974                 :           0 :         }
    2975                 :             : 
    2976                 :           0 :         print_function_trftypes(&buf, proctup);
    2977                 :             : 
    2978                 :           0 :         appendStringInfo(&buf, " LANGUAGE %s\n",
    2979                 :           0 :                                          quote_identifier(get_language_name(proc->prolang, false)));
    2980                 :             : 
    2981                 :             :         /* Emit some miscellaneous options on one line */
    2982                 :           0 :         oldlen = buf.len;
    2983                 :             : 
    2984         [ #  # ]:           0 :         if (proc->prokind == PROKIND_WINDOW)
    2985                 :           0 :                 appendStringInfoString(&buf, " WINDOW");
    2986      [ #  #  # ]:           0 :         switch (proc->provolatile)
    2987                 :             :         {
    2988                 :             :                 case PROVOLATILE_IMMUTABLE:
    2989                 :           0 :                         appendStringInfoString(&buf, " IMMUTABLE");
    2990                 :           0 :                         break;
    2991                 :             :                 case PROVOLATILE_STABLE:
    2992                 :           0 :                         appendStringInfoString(&buf, " STABLE");
    2993                 :           0 :                         break;
    2994                 :             :                 case PROVOLATILE_VOLATILE:
    2995                 :             :                         break;
    2996                 :             :         }
    2997                 :             : 
    2998      [ #  #  # ]:           0 :         switch (proc->proparallel)
    2999                 :             :         {
    3000                 :             :                 case PROPARALLEL_SAFE:
    3001                 :           0 :                         appendStringInfoString(&buf, " PARALLEL SAFE");
    3002                 :           0 :                         break;
    3003                 :             :                 case PROPARALLEL_RESTRICTED:
    3004                 :           0 :                         appendStringInfoString(&buf, " PARALLEL RESTRICTED");
    3005                 :           0 :                         break;
    3006                 :             :                 case PROPARALLEL_UNSAFE:
    3007                 :             :                         break;
    3008                 :             :         }
    3009                 :             : 
    3010         [ #  # ]:           0 :         if (proc->proisstrict)
    3011                 :           0 :                 appendStringInfoString(&buf, " STRICT");
    3012         [ #  # ]:           0 :         if (proc->prosecdef)
    3013                 :           0 :                 appendStringInfoString(&buf, " SECURITY DEFINER");
    3014         [ #  # ]:           0 :         if (proc->proleakproof)
    3015                 :           0 :                 appendStringInfoString(&buf, " LEAKPROOF");
    3016                 :             : 
    3017                 :             :         /* This code for the default cost and rows should match functioncmds.c */
    3018   [ #  #  #  # ]:           0 :         if (proc->prolang == INTERNALlanguageId ||
    3019                 :           0 :                 proc->prolang == ClanguageId)
    3020                 :           0 :                 procost = 1;
    3021                 :             :         else
    3022                 :           0 :                 procost = 100;
    3023         [ #  # ]:           0 :         if (proc->procost != procost)
    3024                 :           0 :                 appendStringInfo(&buf, " COST %g", proc->procost);
    3025                 :             : 
    3026   [ #  #  #  # ]:           0 :         if (proc->prorows > 0 && proc->prorows != 1000)
    3027                 :           0 :                 appendStringInfo(&buf, " ROWS %g", proc->prorows);
    3028                 :             : 
    3029         [ #  # ]:           0 :         if (proc->prosupport)
    3030                 :             :         {
    3031                 :           0 :                 Oid                     argtypes[1];
    3032                 :             : 
    3033                 :             :                 /*
    3034                 :             :                  * We should qualify the support function's name if it wouldn't be
    3035                 :             :                  * resolved by lookup in the current search path.
    3036                 :             :                  */
    3037                 :           0 :                 argtypes[0] = INTERNALOID;
    3038                 :           0 :                 appendStringInfo(&buf, " SUPPORT %s",
    3039                 :           0 :                                                  generate_function_name(proc->prosupport, 1,
    3040                 :           0 :                                                                                                 NIL, argtypes,
    3041                 :             :                                                                                                 false, NULL, false));
    3042                 :           0 :         }
    3043                 :             : 
    3044         [ #  # ]:           0 :         if (oldlen != buf.len)
    3045                 :           0 :                 appendStringInfoChar(&buf, '\n');
    3046                 :             : 
    3047                 :             :         /* Emit any proconfig options, one per line */
    3048                 :           0 :         tmp = SysCacheGetAttr(PROCOID, proctup, Anum_pg_proc_proconfig, &isnull);
    3049         [ #  # ]:           0 :         if (!isnull)
    3050                 :             :         {
    3051                 :           0 :                 ArrayType  *a = DatumGetArrayTypeP(tmp);
    3052                 :           0 :                 int                     i;
    3053                 :             : 
    3054         [ #  # ]:           0 :                 Assert(ARR_ELEMTYPE(a) == TEXTOID);
    3055         [ #  # ]:           0 :                 Assert(ARR_NDIM(a) == 1);
    3056         [ #  # ]:           0 :                 Assert(ARR_LBOUND(a)[0] == 1);
    3057                 :             : 
    3058         [ #  # ]:           0 :                 for (i = 1; i <= ARR_DIMS(a)[0]; i++)
    3059                 :             :                 {
    3060                 :           0 :                         Datum           d;
    3061                 :             : 
    3062                 :           0 :                         d = array_ref(a, 1, &i,
    3063                 :             :                                                   -1 /* varlenarray */ ,
    3064                 :             :                                                   -1 /* TEXT's typlen */ ,
    3065                 :             :                                                   false /* TEXT's typbyval */ ,
    3066                 :             :                                                   TYPALIGN_INT /* TEXT's typalign */ ,
    3067                 :             :                                                   &isnull);
    3068         [ #  # ]:           0 :                         if (!isnull)
    3069                 :             :                         {
    3070                 :           0 :                                 char       *configitem = TextDatumGetCString(d);
    3071                 :           0 :                                 char       *pos;
    3072                 :             : 
    3073                 :           0 :                                 pos = strchr(configitem, '=');
    3074         [ #  # ]:           0 :                                 if (pos == NULL)
    3075                 :           0 :                                         continue;
    3076                 :           0 :                                 *pos++ = '\0';
    3077                 :             : 
    3078                 :           0 :                                 appendStringInfo(&buf, " SET %s TO ",
    3079                 :           0 :                                                                  quote_identifier(configitem));
    3080                 :             : 
    3081                 :             :                                 /*
    3082                 :             :                                  * Variables that are marked GUC_LIST_QUOTE were already fully
    3083                 :             :                                  * quoted by flatten_set_variable_args() before they were put
    3084                 :             :                                  * into the proconfig array.  However, because the quoting
    3085                 :             :                                  * rules used there aren't exactly like SQL's, we have to
    3086                 :             :                                  * break the list value apart and then quote the elements as
    3087                 :             :                                  * string literals.  (The elements may be double-quoted as-is,
    3088                 :             :                                  * but we can't just feed them to the SQL parser; it would do
    3089                 :             :                                  * the wrong thing with elements that are zero-length or
    3090                 :             :                                  * longer than NAMEDATALEN.)  Also, we need a special case for
    3091                 :             :                                  * empty lists.
    3092                 :             :                                  *
    3093                 :             :                                  * Variables that are not so marked should just be emitted as
    3094                 :             :                                  * simple string literals.  If the variable is not known to
    3095                 :             :                                  * guc.c, we'll do that; this makes it unsafe to use
    3096                 :             :                                  * GUC_LIST_QUOTE for extension variables.
    3097                 :             :                                  */
    3098         [ #  # ]:           0 :                                 if (GetConfigOptionFlags(configitem, true) & GUC_LIST_QUOTE)
    3099                 :             :                                 {
    3100                 :           0 :                                         List       *namelist;
    3101                 :           0 :                                         ListCell   *lc;
    3102                 :             : 
    3103                 :             :                                         /* Parse string into list of identifiers */
    3104         [ #  # ]:           0 :                                         if (!SplitGUCList(pos, ',', &namelist))
    3105                 :             :                                         {
    3106                 :             :                                                 /* this shouldn't fail really */
    3107   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "invalid list syntax in proconfig item");
    3108                 :           0 :                                         }
    3109                 :             :                                         /* Special case: represent an empty list as NULL */
    3110         [ #  # ]:           0 :                                         if (namelist == NIL)
    3111                 :           0 :                                                 appendStringInfoString(&buf, "NULL");
    3112   [ #  #  #  #  :           0 :                                         foreach(lc, namelist)
                   #  # ]
    3113                 :             :                                         {
    3114                 :           0 :                                                 char       *curname = (char *) lfirst(lc);
    3115                 :             : 
    3116                 :           0 :                                                 simple_quote_literal(&buf, curname);
    3117         [ #  # ]:           0 :                                                 if (lnext(namelist, lc))
    3118                 :           0 :                                                         appendStringInfoString(&buf, ", ");
    3119                 :           0 :                                         }
    3120                 :           0 :                                 }
    3121                 :             :                                 else
    3122                 :           0 :                                         simple_quote_literal(&buf, pos);
    3123                 :           0 :                                 appendStringInfoChar(&buf, '\n');
    3124         [ #  # ]:           0 :                         }
    3125      [ #  #  # ]:           0 :                 }
    3126                 :           0 :         }
    3127                 :             : 
    3128                 :             :         /* And finally the function definition ... */
    3129                 :           0 :         (void) SysCacheGetAttr(PROCOID, proctup, Anum_pg_proc_prosqlbody, &isnull);
    3130   [ #  #  #  # ]:           0 :         if (proc->prolang == SQLlanguageId && !isnull)
    3131                 :             :         {
    3132                 :           0 :                 print_function_sqlbody(&buf, proctup);
    3133                 :           0 :         }
    3134                 :             :         else
    3135                 :             :         {
    3136                 :           0 :                 appendStringInfoString(&buf, "AS ");
    3137                 :             : 
    3138                 :           0 :                 tmp = SysCacheGetAttr(PROCOID, proctup, Anum_pg_proc_probin, &isnull);
    3139         [ #  # ]:           0 :                 if (!isnull)
    3140                 :             :                 {
    3141                 :           0 :                         simple_quote_literal(&buf, TextDatumGetCString(tmp));
    3142                 :           0 :                         appendStringInfoString(&buf, ", "); /* assume prosrc isn't null */
    3143                 :           0 :                 }
    3144                 :             : 
    3145                 :           0 :                 tmp = SysCacheGetAttrNotNull(PROCOID, proctup, Anum_pg_proc_prosrc);
    3146                 :           0 :                 prosrc = TextDatumGetCString(tmp);
    3147                 :             : 
    3148                 :             :                 /*
    3149                 :             :                  * We always use dollar quoting.  Figure out a suitable delimiter.
    3150                 :             :                  *
    3151                 :             :                  * Since the user is likely to be editing the function body string, we
    3152                 :             :                  * shouldn't use a short delimiter that he might easily create a
    3153                 :             :                  * conflict with.  Hence prefer "$function$"/"$procedure$", but extend
    3154                 :             :                  * if needed.
    3155                 :             :                  */
    3156                 :           0 :                 initStringInfo(&dq);
    3157                 :           0 :                 appendStringInfoChar(&dq, '$');
    3158                 :           0 :                 appendStringInfoString(&dq, (isfunction ? "function" : "procedure"));
    3159         [ #  # ]:           0 :                 while (strstr(prosrc, dq.data) != NULL)
    3160                 :           0 :                         appendStringInfoChar(&dq, 'x');
    3161                 :           0 :                 appendStringInfoChar(&dq, '$');
    3162                 :             : 
    3163                 :           0 :                 appendBinaryStringInfo(&buf, dq.data, dq.len);
    3164                 :           0 :                 appendStringInfoString(&buf, prosrc);
    3165                 :           0 :                 appendBinaryStringInfo(&buf, dq.data, dq.len);
    3166                 :             :         }
    3167                 :             : 
    3168                 :           0 :         appendStringInfoChar(&buf, '\n');
    3169                 :             : 
    3170                 :           0 :         ReleaseSysCache(proctup);
    3171                 :             : 
    3172                 :           0 :         PG_RETURN_TEXT_P(string_to_text(buf.data));
    3173                 :           0 : }
    3174                 :             : 
    3175                 :             : /*
    3176                 :             :  * pg_get_function_arguments
    3177                 :             :  *              Get a nicely-formatted list of arguments for a function.
    3178                 :             :  *              This is everything that would go between the parentheses in
    3179                 :             :  *              CREATE FUNCTION.
    3180                 :             :  */
    3181                 :             : Datum
    3182                 :           0 : pg_get_function_arguments(PG_FUNCTION_ARGS)
    3183                 :             : {
    3184                 :           0 :         Oid                     funcid = PG_GETARG_OID(0);
    3185                 :           0 :         StringInfoData buf;
    3186                 :           0 :         HeapTuple       proctup;
    3187                 :             : 
    3188                 :           0 :         proctup = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
    3189         [ #  # ]:           0 :         if (!HeapTupleIsValid(proctup))
    3190                 :           0 :                 PG_RETURN_NULL();
    3191                 :             : 
    3192                 :           0 :         initStringInfo(&buf);
    3193                 :             : 
    3194                 :           0 :         (void) print_function_arguments(&buf, proctup, false, true);
    3195                 :             : 
    3196                 :           0 :         ReleaseSysCache(proctup);
    3197                 :             : 
    3198                 :           0 :         PG_RETURN_TEXT_P(string_to_text(buf.data));
    3199                 :           0 : }
    3200                 :             : 
    3201                 :             : /*
    3202                 :             :  * pg_get_function_identity_arguments
    3203                 :             :  *              Get a formatted list of arguments for a function.
    3204                 :             :  *              This is everything that would go between the parentheses in
    3205                 :             :  *              ALTER FUNCTION, etc.  In particular, don't print defaults.
    3206                 :             :  */
    3207                 :             : Datum
    3208                 :           0 : pg_get_function_identity_arguments(PG_FUNCTION_ARGS)
    3209                 :             : {
    3210                 :           0 :         Oid                     funcid = PG_GETARG_OID(0);
    3211                 :           0 :         StringInfoData buf;
    3212                 :           0 :         HeapTuple       proctup;
    3213                 :             : 
    3214                 :           0 :         proctup = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
    3215         [ #  # ]:           0 :         if (!HeapTupleIsValid(proctup))
    3216                 :           0 :                 PG_RETURN_NULL();
    3217                 :             : 
    3218                 :           0 :         initStringInfo(&buf);
    3219                 :             : 
    3220                 :           0 :         (void) print_function_arguments(&buf, proctup, false, false);
    3221                 :             : 
    3222                 :           0 :         ReleaseSysCache(proctup);
    3223                 :             : 
    3224                 :           0 :         PG_RETURN_TEXT_P(string_to_text(buf.data));
    3225                 :           0 : }
    3226                 :             : 
    3227                 :             : /*
    3228                 :             :  * pg_get_function_result
    3229                 :             :  *              Get a nicely-formatted version of the result type of a function.
    3230                 :             :  *              This is what would appear after RETURNS in CREATE FUNCTION.
    3231                 :             :  */
    3232                 :             : Datum
    3233                 :           0 : pg_get_function_result(PG_FUNCTION_ARGS)
    3234                 :             : {
    3235                 :           0 :         Oid                     funcid = PG_GETARG_OID(0);
    3236                 :           0 :         StringInfoData buf;
    3237                 :           0 :         HeapTuple       proctup;
    3238                 :             : 
    3239                 :           0 :         proctup = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
    3240         [ #  # ]:           0 :         if (!HeapTupleIsValid(proctup))
    3241                 :           0 :                 PG_RETURN_NULL();
    3242                 :             : 
    3243         [ #  # ]:           0 :         if (((Form_pg_proc) GETSTRUCT(proctup))->prokind == PROKIND_PROCEDURE)
    3244                 :             :         {
    3245                 :           0 :                 ReleaseSysCache(proctup);
    3246                 :           0 :                 PG_RETURN_NULL();
    3247                 :           0 :         }
    3248                 :             : 
    3249                 :           0 :         initStringInfo(&buf);
    3250                 :             : 
    3251                 :           0 :         print_function_rettype(&buf, proctup);
    3252                 :             : 
    3253                 :           0 :         ReleaseSysCache(proctup);
    3254                 :             : 
    3255                 :           0 :         PG_RETURN_TEXT_P(string_to_text(buf.data));
    3256                 :           0 : }
    3257                 :             : 
    3258                 :             : /*
    3259                 :             :  * Guts of pg_get_function_result: append the function's return type
    3260                 :             :  * to the specified buffer.
    3261                 :             :  */
    3262                 :             : static void
    3263                 :           0 : print_function_rettype(StringInfo buf, HeapTuple proctup)
    3264                 :             : {
    3265                 :           0 :         Form_pg_proc proc = (Form_pg_proc) GETSTRUCT(proctup);
    3266                 :           0 :         int                     ntabargs = 0;
    3267                 :           0 :         StringInfoData rbuf;
    3268                 :             : 
    3269                 :           0 :         initStringInfo(&rbuf);
    3270                 :             : 
    3271         [ #  # ]:           0 :         if (proc->proretset)
    3272                 :             :         {
    3273                 :             :                 /* It might be a table function; try to print the arguments */
    3274                 :           0 :                 appendStringInfoString(&rbuf, "TABLE(");
    3275                 :           0 :                 ntabargs = print_function_arguments(&rbuf, proctup, true, false);
    3276         [ #  # ]:           0 :                 if (ntabargs > 0)
    3277                 :           0 :                         appendStringInfoChar(&rbuf, ')');
    3278                 :             :                 else
    3279                 :           0 :                         resetStringInfo(&rbuf);
    3280                 :           0 :         }
    3281                 :             : 
    3282         [ #  # ]:           0 :         if (ntabargs == 0)
    3283                 :             :         {
    3284                 :             :                 /* Not a table function, so do the normal thing */
    3285         [ #  # ]:           0 :                 if (proc->proretset)
    3286                 :           0 :                         appendStringInfoString(&rbuf, "SETOF ");
    3287                 :           0 :                 appendStringInfoString(&rbuf, format_type_be(proc->prorettype));
    3288                 :           0 :         }
    3289                 :             : 
    3290                 :           0 :         appendBinaryStringInfo(buf, rbuf.data, rbuf.len);
    3291                 :           0 : }
    3292                 :             : 
    3293                 :             : /*
    3294                 :             :  * Common code for pg_get_function_arguments and pg_get_function_result:
    3295                 :             :  * append the desired subset of arguments to buf.  We print only TABLE
    3296                 :             :  * arguments when print_table_args is true, and all the others when it's false.
    3297                 :             :  * We print argument defaults only if print_defaults is true.
    3298                 :             :  * Function return value is the number of arguments printed.
    3299                 :             :  */
    3300                 :             : static int
    3301                 :           0 : print_function_arguments(StringInfo buf, HeapTuple proctup,
    3302                 :             :                                                  bool print_table_args, bool print_defaults)
    3303                 :             : {
    3304                 :           0 :         Form_pg_proc proc = (Form_pg_proc) GETSTRUCT(proctup);
    3305                 :           0 :         int                     numargs;
    3306                 :           0 :         Oid                *argtypes;
    3307                 :           0 :         char      **argnames;
    3308                 :           0 :         char       *argmodes;
    3309                 :           0 :         int                     insertorderbyat = -1;
    3310                 :           0 :         int                     argsprinted;
    3311                 :           0 :         int                     inputargno;
    3312                 :           0 :         int                     nlackdefaults;
    3313                 :           0 :         List       *argdefaults = NIL;
    3314                 :           0 :         ListCell   *nextargdefault = NULL;
    3315                 :           0 :         int                     i;
    3316                 :             : 
    3317                 :           0 :         numargs = get_func_arg_info(proctup,
    3318                 :             :                                                                 &argtypes, &argnames, &argmodes);
    3319                 :             : 
    3320                 :           0 :         nlackdefaults = numargs;
    3321   [ #  #  #  # ]:           0 :         if (print_defaults && proc->pronargdefaults > 0)
    3322                 :             :         {
    3323                 :           0 :                 Datum           proargdefaults;
    3324                 :           0 :                 bool            isnull;
    3325                 :             : 
    3326                 :           0 :                 proargdefaults = SysCacheGetAttr(PROCOID, proctup,
    3327                 :             :                                                                                  Anum_pg_proc_proargdefaults,
    3328                 :             :                                                                                  &isnull);
    3329         [ #  # ]:           0 :                 if (!isnull)
    3330                 :             :                 {
    3331                 :           0 :                         char       *str;
    3332                 :             : 
    3333                 :           0 :                         str = TextDatumGetCString(proargdefaults);
    3334                 :           0 :                         argdefaults = castNode(List, stringToNode(str));
    3335                 :           0 :                         pfree(str);
    3336                 :           0 :                         nextargdefault = list_head(argdefaults);
    3337                 :             :                         /* nlackdefaults counts only *input* arguments lacking defaults */
    3338                 :           0 :                         nlackdefaults = proc->pronargs - list_length(argdefaults);
    3339                 :           0 :                 }
    3340                 :           0 :         }
    3341                 :             : 
    3342                 :             :         /* Check for special treatment of ordered-set aggregates */
    3343         [ #  # ]:           0 :         if (proc->prokind == PROKIND_AGGREGATE)
    3344                 :             :         {
    3345                 :           0 :                 HeapTuple       aggtup;
    3346                 :           0 :                 Form_pg_aggregate agg;
    3347                 :             : 
    3348                 :           0 :                 aggtup = SearchSysCache1(AGGFNOID, ObjectIdGetDatum(proc->oid));
    3349         [ #  # ]:           0 :                 if (!HeapTupleIsValid(aggtup))
    3350   [ #  #  #  # ]:           0 :                         elog(ERROR, "cache lookup failed for aggregate %u",
    3351                 :             :                                  proc->oid);
    3352                 :           0 :                 agg = (Form_pg_aggregate) GETSTRUCT(aggtup);
    3353         [ #  # ]:           0 :                 if (AGGKIND_IS_ORDERED_SET(agg->aggkind))
    3354                 :           0 :                         insertorderbyat = agg->aggnumdirectargs;
    3355                 :           0 :                 ReleaseSysCache(aggtup);
    3356                 :           0 :         }
    3357                 :             : 
    3358                 :           0 :         argsprinted = 0;
    3359                 :           0 :         inputargno = 0;
    3360         [ #  # ]:           0 :         for (i = 0; i < numargs; i++)
    3361                 :             :         {
    3362                 :           0 :                 Oid                     argtype = argtypes[i];
    3363         [ #  # ]:           0 :                 char       *argname = argnames ? argnames[i] : NULL;
    3364         [ #  # ]:           0 :                 char            argmode = argmodes ? argmodes[i] : PROARGMODE_IN;
    3365                 :           0 :                 const char *modename;
    3366                 :           0 :                 bool            isinput;
    3367                 :             : 
    3368   [ #  #  #  #  :           0 :                 switch (argmode)
                   #  # ]
    3369                 :             :                 {
    3370                 :             :                         case PROARGMODE_IN:
    3371                 :             : 
    3372                 :             :                                 /*
    3373                 :             :                                  * For procedures, explicitly mark all argument modes, so as
    3374                 :             :                                  * to avoid ambiguity with the SQL syntax for DROP PROCEDURE.
    3375                 :             :                                  */
    3376         [ #  # ]:           0 :                                 if (proc->prokind == PROKIND_PROCEDURE)
    3377                 :           0 :                                         modename = "IN ";
    3378                 :             :                                 else
    3379                 :           0 :                                         modename = "";
    3380                 :           0 :                                 isinput = true;
    3381                 :           0 :                                 break;
    3382                 :             :                         case PROARGMODE_INOUT:
    3383                 :           0 :                                 modename = "INOUT ";
    3384                 :           0 :                                 isinput = true;
    3385                 :           0 :                                 break;
    3386                 :             :                         case PROARGMODE_OUT:
    3387                 :           0 :                                 modename = "OUT ";
    3388                 :           0 :                                 isinput = false;
    3389                 :           0 :                                 break;
    3390                 :             :                         case PROARGMODE_VARIADIC:
    3391                 :           0 :                                 modename = "VARIADIC ";
    3392                 :           0 :                                 isinput = true;
    3393                 :           0 :                                 break;
    3394                 :             :                         case PROARGMODE_TABLE:
    3395                 :           0 :                                 modename = "";
    3396                 :           0 :                                 isinput = false;
    3397                 :           0 :                                 break;
    3398                 :             :                         default:
    3399   [ #  #  #  # ]:           0 :                                 elog(ERROR, "invalid parameter mode '%c'", argmode);
    3400                 :           0 :                                 modename = NULL;        /* keep compiler quiet */
    3401                 :           0 :                                 isinput = false;
    3402                 :           0 :                                 break;
    3403                 :             :                 }
    3404         [ #  # ]:           0 :                 if (isinput)
    3405                 :           0 :                         inputargno++;           /* this is a 1-based counter */
    3406                 :             : 
    3407         [ #  # ]:           0 :                 if (print_table_args != (argmode == PROARGMODE_TABLE))
    3408                 :           0 :                         continue;
    3409                 :             : 
    3410         [ #  # ]:           0 :                 if (argsprinted == insertorderbyat)
    3411                 :             :                 {
    3412         [ #  # ]:           0 :                         if (argsprinted)
    3413                 :           0 :                                 appendStringInfoChar(buf, ' ');
    3414                 :           0 :                         appendStringInfoString(buf, "ORDER BY ");
    3415                 :           0 :                 }
    3416         [ #  # ]:           0 :                 else if (argsprinted)
    3417                 :           0 :                         appendStringInfoString(buf, ", ");
    3418                 :             : 
    3419                 :           0 :                 appendStringInfoString(buf, modename);
    3420   [ #  #  #  # ]:           0 :                 if (argname && argname[0])
    3421                 :           0 :                         appendStringInfo(buf, "%s ", quote_identifier(argname));
    3422                 :           0 :                 appendStringInfoString(buf, format_type_be(argtype));
    3423   [ #  #  #  #  :           0 :                 if (print_defaults && isinput && inputargno > nlackdefaults)
                   #  # ]
    3424                 :             :                 {
    3425                 :           0 :                         Node       *expr;
    3426                 :             : 
    3427         [ #  # ]:           0 :                         Assert(nextargdefault != NULL);
    3428                 :           0 :                         expr = (Node *) lfirst(nextargdefault);
    3429                 :           0 :                         nextargdefault = lnext(argdefaults, nextargdefault);
    3430                 :             : 
    3431                 :           0 :                         appendStringInfo(buf, " DEFAULT %s",
    3432                 :           0 :                                                          deparse_expression(expr, NIL, false, false));
    3433                 :           0 :                 }
    3434                 :           0 :                 argsprinted++;
    3435                 :             : 
    3436                 :             :                 /* nasty hack: print the last arg twice for variadic ordered-set agg */
    3437   [ #  #  #  # ]:           0 :                 if (argsprinted == insertorderbyat && i == numargs - 1)
    3438                 :             :                 {
    3439                 :           0 :                         i--;
    3440                 :             :                         /* aggs shouldn't have defaults anyway, but just to be sure ... */
    3441                 :           0 :                         print_defaults = false;
    3442                 :           0 :                 }
    3443      [ #  #  # ]:           0 :         }
    3444                 :             : 
    3445                 :           0 :         return argsprinted;
    3446                 :           0 : }
    3447                 :             : 
    3448                 :             : static bool
    3449                 :           0 : is_input_argument(int nth, const char *argmodes)
    3450                 :             : {
    3451                 :           0 :         return (!argmodes
    3452         [ #  # ]:           0 :                         || argmodes[nth] == PROARGMODE_IN
    3453         [ #  # ]:           0 :                         || argmodes[nth] == PROARGMODE_INOUT
    3454         [ #  # ]:           0 :                         || argmodes[nth] == PROARGMODE_VARIADIC);
    3455                 :             : }
    3456                 :             : 
    3457                 :             : /*
    3458                 :             :  * Append used transformed types to specified buffer
    3459                 :             :  */
    3460                 :             : static void
    3461                 :           0 : print_function_trftypes(StringInfo buf, HeapTuple proctup)
    3462                 :             : {
    3463                 :           0 :         Oid                *trftypes;
    3464                 :           0 :         int                     ntypes;
    3465                 :             : 
    3466                 :           0 :         ntypes = get_func_trftypes(proctup, &trftypes);
    3467         [ #  # ]:           0 :         if (ntypes > 0)
    3468                 :             :         {
    3469                 :           0 :                 int                     i;
    3470                 :             : 
    3471                 :           0 :                 appendStringInfoString(buf, " TRANSFORM ");
    3472         [ #  # ]:           0 :                 for (i = 0; i < ntypes; i++)
    3473                 :             :                 {
    3474         [ #  # ]:           0 :                         if (i != 0)
    3475                 :           0 :                                 appendStringInfoString(buf, ", ");
    3476                 :           0 :                         appendStringInfo(buf, "FOR TYPE %s", format_type_be(trftypes[i]));
    3477                 :           0 :                 }
    3478                 :           0 :                 appendStringInfoChar(buf, '\n');
    3479                 :           0 :         }
    3480                 :           0 : }
    3481                 :             : 
    3482                 :             : /*
    3483                 :             :  * Get textual representation of a function argument's default value.  The
    3484                 :             :  * second argument of this function is the argument number among all arguments
    3485                 :             :  * (i.e. proallargtypes, *not* proargtypes), starting with 1, because that's
    3486                 :             :  * how information_schema.sql uses it.
    3487                 :             :  */
    3488                 :             : Datum
    3489                 :           0 : pg_get_function_arg_default(PG_FUNCTION_ARGS)
    3490                 :             : {
    3491                 :           0 :         Oid                     funcid = PG_GETARG_OID(0);
    3492                 :           0 :         int32           nth_arg = PG_GETARG_INT32(1);
    3493                 :           0 :         HeapTuple       proctup;
    3494                 :           0 :         Form_pg_proc proc;
    3495                 :           0 :         int                     numargs;
    3496                 :           0 :         Oid                *argtypes;
    3497                 :           0 :         char      **argnames;
    3498                 :           0 :         char       *argmodes;
    3499                 :           0 :         int                     i;
    3500                 :           0 :         List       *argdefaults;
    3501                 :           0 :         Node       *node;
    3502                 :           0 :         char       *str;
    3503                 :           0 :         int                     nth_inputarg;
    3504                 :           0 :         Datum           proargdefaults;
    3505                 :           0 :         bool            isnull;
    3506                 :           0 :         int                     nth_default;
    3507                 :             : 
    3508                 :           0 :         proctup = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
    3509         [ #  # ]:           0 :         if (!HeapTupleIsValid(proctup))
    3510                 :           0 :                 PG_RETURN_NULL();
    3511                 :             : 
    3512                 :           0 :         numargs = get_func_arg_info(proctup, &argtypes, &argnames, &argmodes);
    3513   [ #  #  #  #  :           0 :         if (nth_arg < 1 || nth_arg > numargs || !is_input_argument(nth_arg - 1, argmodes))
                   #  # ]
    3514                 :             :         {
    3515                 :           0 :                 ReleaseSysCache(proctup);
    3516                 :           0 :                 PG_RETURN_NULL();
    3517                 :           0 :         }
    3518                 :             : 
    3519                 :           0 :         nth_inputarg = 0;
    3520         [ #  # ]:           0 :         for (i = 0; i < nth_arg; i++)
    3521         [ #  # ]:           0 :                 if (is_input_argument(i, argmodes))
    3522                 :           0 :                         nth_inputarg++;
    3523                 :             : 
    3524                 :           0 :         proargdefaults = SysCacheGetAttr(PROCOID, proctup,
    3525                 :             :                                                                          Anum_pg_proc_proargdefaults,
    3526                 :             :                                                                          &isnull);
    3527         [ #  # ]:           0 :         if (isnull)
    3528                 :             :         {
    3529                 :           0 :                 ReleaseSysCache(proctup);
    3530                 :           0 :                 PG_RETURN_NULL();
    3531                 :           0 :         }
    3532                 :             : 
    3533                 :           0 :         str = TextDatumGetCString(proargdefaults);
    3534                 :           0 :         argdefaults = castNode(List, stringToNode(str));
    3535                 :           0 :         pfree(str);
    3536                 :             : 
    3537                 :           0 :         proc = (Form_pg_proc) GETSTRUCT(proctup);
    3538                 :             : 
    3539                 :             :         /*
    3540                 :             :          * Calculate index into proargdefaults: proargdefaults corresponds to the
    3541                 :             :          * last N input arguments, where N = pronargdefaults.
    3542                 :             :          */
    3543                 :           0 :         nth_default = nth_inputarg - 1 - (proc->pronargs - proc->pronargdefaults);
    3544                 :             : 
    3545   [ #  #  #  # ]:           0 :         if (nth_default < 0 || nth_default >= list_length(argdefaults))
    3546                 :             :         {
    3547                 :           0 :                 ReleaseSysCache(proctup);
    3548                 :           0 :                 PG_RETURN_NULL();
    3549                 :           0 :         }
    3550                 :           0 :         node = list_nth(argdefaults, nth_default);
    3551                 :           0 :         str = deparse_expression(node, NIL, false, false);
    3552                 :             : 
    3553                 :           0 :         ReleaseSysCache(proctup);
    3554                 :             : 
    3555                 :           0 :         PG_RETURN_TEXT_P(string_to_text(str));
    3556                 :           0 : }
    3557                 :             : 
    3558                 :             : static void
    3559                 :           0 : print_function_sqlbody(StringInfo buf, HeapTuple proctup)
    3560                 :             : {
    3561                 :           0 :         int                     numargs;
    3562                 :           0 :         Oid                *argtypes;
    3563                 :           0 :         char      **argnames;
    3564                 :           0 :         char       *argmodes;
    3565                 :           0 :         deparse_namespace dpns = {0};
    3566                 :           0 :         Datum           tmp;
    3567                 :           0 :         Node       *n;
    3568                 :             : 
    3569                 :           0 :         dpns.funcname = pstrdup(NameStr(((Form_pg_proc) GETSTRUCT(proctup))->proname));
    3570                 :           0 :         numargs = get_func_arg_info(proctup,
    3571                 :             :                                                                 &argtypes, &argnames, &argmodes);
    3572                 :           0 :         dpns.numargs = numargs;
    3573                 :           0 :         dpns.argnames = argnames;
    3574                 :             : 
    3575                 :           0 :         tmp = SysCacheGetAttrNotNull(PROCOID, proctup, Anum_pg_proc_prosqlbody);
    3576                 :           0 :         n = stringToNode(TextDatumGetCString(tmp));
    3577                 :             : 
    3578         [ #  # ]:           0 :         if (IsA(n, List))
    3579                 :             :         {
    3580                 :           0 :                 List       *stmts;
    3581                 :           0 :                 ListCell   *lc;
    3582                 :             : 
    3583                 :           0 :                 stmts = linitial(castNode(List, n));
    3584                 :             : 
    3585                 :           0 :                 appendStringInfoString(buf, "BEGIN ATOMIC\n");
    3586                 :             : 
    3587   [ #  #  #  #  :           0 :                 foreach(lc, stmts)
                   #  # ]
    3588                 :             :                 {
    3589                 :           0 :                         Query      *query = lfirst_node(Query, lc);
    3590                 :             : 
    3591                 :             :                         /* It seems advisable to get at least AccessShareLock on rels */
    3592                 :           0 :                         AcquireRewriteLocks(query, false, false);
    3593                 :           0 :                         get_query_def(query, buf, list_make1(&dpns), NULL, false,
    3594                 :             :                                                   PRETTYFLAG_INDENT, WRAP_COLUMN_DEFAULT, 1);
    3595                 :           0 :                         appendStringInfoChar(buf, ';');
    3596                 :           0 :                         appendStringInfoChar(buf, '\n');
    3597                 :           0 :                 }
    3598                 :             : 
    3599                 :           0 :                 appendStringInfoString(buf, "END");
    3600                 :           0 :         }
    3601                 :             :         else
    3602                 :             :         {
    3603                 :           0 :                 Query      *query = castNode(Query, n);
    3604                 :             : 
    3605                 :             :                 /* It seems advisable to get at least AccessShareLock on rels */
    3606                 :           0 :                 AcquireRewriteLocks(query, false, false);
    3607                 :           0 :                 get_query_def(query, buf, list_make1(&dpns), NULL, false,
    3608                 :             :                                           0, WRAP_COLUMN_DEFAULT, 0);
    3609                 :           0 :         }
    3610                 :           0 : }
    3611                 :             : 
    3612                 :             : Datum
    3613                 :           0 : pg_get_function_sqlbody(PG_FUNCTION_ARGS)
    3614                 :             : {
    3615                 :           0 :         Oid                     funcid = PG_GETARG_OID(0);
    3616                 :           0 :         StringInfoData buf;
    3617                 :           0 :         HeapTuple       proctup;
    3618                 :           0 :         bool            isnull;
    3619                 :             : 
    3620                 :           0 :         initStringInfo(&buf);
    3621                 :             : 
    3622                 :             :         /* Look up the function */
    3623                 :           0 :         proctup = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
    3624         [ #  # ]:           0 :         if (!HeapTupleIsValid(proctup))
    3625                 :           0 :                 PG_RETURN_NULL();
    3626                 :             : 
    3627                 :           0 :         (void) SysCacheGetAttr(PROCOID, proctup, Anum_pg_proc_prosqlbody, &isnull);
    3628         [ #  # ]:           0 :         if (isnull)
    3629                 :             :         {
    3630                 :           0 :                 ReleaseSysCache(proctup);
    3631                 :           0 :                 PG_RETURN_NULL();
    3632                 :           0 :         }
    3633                 :             : 
    3634                 :           0 :         print_function_sqlbody(&buf, proctup);
    3635                 :             : 
    3636                 :           0 :         ReleaseSysCache(proctup);
    3637                 :             : 
    3638                 :           0 :         PG_RETURN_TEXT_P(cstring_to_text_with_len(buf.data, buf.len));
    3639                 :           0 : }
    3640                 :             : 
    3641                 :             : 
    3642                 :             : /*
    3643                 :             :  * deparse_expression                   - General utility for deparsing expressions
    3644                 :             :  *
    3645                 :             :  * calls deparse_expression_pretty with all prettyPrinting disabled
    3646                 :             :  */
    3647                 :             : char *
    3648                 :           0 : deparse_expression(Node *expr, List *dpcontext,
    3649                 :             :                                    bool forceprefix, bool showimplicit)
    3650                 :             : {
    3651                 :           0 :         return deparse_expression_pretty(expr, dpcontext, forceprefix,
    3652                 :           0 :                                                                          showimplicit, 0, 0);
    3653                 :             : }
    3654                 :             : 
    3655                 :             : /* ----------
    3656                 :             :  * deparse_expression_pretty    - General utility for deparsing expressions
    3657                 :             :  *
    3658                 :             :  * expr is the node tree to be deparsed.  It must be a transformed expression
    3659                 :             :  * tree (ie, not the raw output of gram.y).
    3660                 :             :  *
    3661                 :             :  * dpcontext is a list of deparse_namespace nodes representing the context
    3662                 :             :  * for interpreting Vars in the node tree.  It can be NIL if no Vars are
    3663                 :             :  * expected.
    3664                 :             :  *
    3665                 :             :  * forceprefix is true to force all Vars to be prefixed with their table names.
    3666                 :             :  *
    3667                 :             :  * showimplicit is true to force all implicit casts to be shown explicitly.
    3668                 :             :  *
    3669                 :             :  * Tries to pretty up the output according to prettyFlags and startIndent.
    3670                 :             :  *
    3671                 :             :  * The result is a palloc'd string.
    3672                 :             :  * ----------
    3673                 :             :  */
    3674                 :             : static char *
    3675                 :           0 : deparse_expression_pretty(Node *expr, List *dpcontext,
    3676                 :             :                                                   bool forceprefix, bool showimplicit,
    3677                 :             :                                                   int prettyFlags, int startIndent)
    3678                 :             : {
    3679                 :           0 :         StringInfoData buf;
    3680                 :           0 :         deparse_context context;
    3681                 :             : 
    3682                 :           0 :         initStringInfo(&buf);
    3683                 :           0 :         context.buf = &buf;
    3684                 :           0 :         context.namespaces = dpcontext;
    3685                 :           0 :         context.resultDesc = NULL;
    3686                 :           0 :         context.targetList = NIL;
    3687                 :           0 :         context.windowClause = NIL;
    3688                 :           0 :         context.varprefix = forceprefix;
    3689                 :           0 :         context.prettyFlags = prettyFlags;
    3690                 :           0 :         context.wrapColumn = WRAP_COLUMN_DEFAULT;
    3691                 :           0 :         context.indentLevel = startIndent;
    3692                 :           0 :         context.colNamesVisible = true;
    3693                 :           0 :         context.inGroupBy = false;
    3694                 :           0 :         context.varInOrderBy = false;
    3695                 :           0 :         context.appendparents = NULL;
    3696                 :             : 
    3697                 :           0 :         get_rule_expr(expr, &context, showimplicit);
    3698                 :             : 
    3699                 :           0 :         return buf.data;
    3700                 :           0 : }
    3701                 :             : 
    3702                 :             : /* ----------
    3703                 :             :  * deparse_context_for                  - Build deparse context for a single relation
    3704                 :             :  *
    3705                 :             :  * Given the reference name (alias) and OID of a relation, build deparsing
    3706                 :             :  * context for an expression referencing only that relation (as varno 1,
    3707                 :             :  * varlevelsup 0).  This is sufficient for many uses of deparse_expression.
    3708                 :             :  * ----------
    3709                 :             :  */
    3710                 :             : List *
    3711                 :           0 : deparse_context_for(const char *aliasname, Oid relid)
    3712                 :             : {
    3713                 :           0 :         deparse_namespace *dpns;
    3714                 :           0 :         RangeTblEntry *rte;
    3715                 :             : 
    3716                 :           0 :         dpns = palloc0_object(deparse_namespace);
    3717                 :             : 
    3718                 :             :         /* Build a minimal RTE for the rel */
    3719                 :           0 :         rte = makeNode(RangeTblEntry);
    3720                 :           0 :         rte->rtekind = RTE_RELATION;
    3721                 :           0 :         rte->relid = relid;
    3722                 :           0 :         rte->relkind = RELKIND_RELATION;     /* no need for exactness here */
    3723                 :           0 :         rte->rellockmode = AccessShareLock;
    3724                 :           0 :         rte->alias = makeAlias(aliasname, NIL);
    3725                 :           0 :         rte->eref = rte->alias;
    3726                 :           0 :         rte->lateral = false;
    3727                 :           0 :         rte->inh = false;
    3728                 :           0 :         rte->inFromCl = true;
    3729                 :             : 
    3730                 :             :         /* Build one-element rtable */
    3731                 :           0 :         dpns->rtable = list_make1(rte);
    3732                 :           0 :         dpns->subplans = NIL;
    3733                 :           0 :         dpns->ctes = NIL;
    3734                 :           0 :         dpns->appendrels = NULL;
    3735                 :           0 :         set_rtable_names(dpns, NIL, NULL);
    3736                 :           0 :         set_simple_column_names(dpns);
    3737                 :             : 
    3738                 :             :         /* Return a one-deep namespace stack */
    3739                 :           0 :         return list_make1(dpns);
    3740                 :           0 : }
    3741                 :             : 
    3742                 :             : /*
    3743                 :             :  * deparse_context_for_plan_tree - Build deparse context for a Plan tree
    3744                 :             :  *
    3745                 :             :  * When deparsing an expression in a Plan tree, we use the plan's rangetable
    3746                 :             :  * to resolve names of simple Vars.  The initialization of column names for
    3747                 :             :  * this is rather expensive if the rangetable is large, and it'll be the same
    3748                 :             :  * for every expression in the Plan tree; so we do it just once and re-use
    3749                 :             :  * the result of this function for each expression.  (Note that the result
    3750                 :             :  * is not usable until set_deparse_context_plan() is applied to it.)
    3751                 :             :  *
    3752                 :             :  * In addition to the PlannedStmt, pass the per-RTE alias names
    3753                 :             :  * assigned by a previous call to select_rtable_names_for_explain.
    3754                 :             :  */
    3755                 :             : List *
    3756                 :           0 : deparse_context_for_plan_tree(PlannedStmt *pstmt, List *rtable_names)
    3757                 :             : {
    3758                 :           0 :         deparse_namespace *dpns;
    3759                 :             : 
    3760                 :           0 :         dpns = palloc0_object(deparse_namespace);
    3761                 :             : 
    3762                 :             :         /* Initialize fields that stay the same across the whole plan tree */
    3763                 :           0 :         dpns->rtable = pstmt->rtable;
    3764                 :           0 :         dpns->rtable_names = rtable_names;
    3765                 :           0 :         dpns->subplans = pstmt->subplans;
    3766                 :           0 :         dpns->ctes = NIL;
    3767         [ #  # ]:           0 :         if (pstmt->appendRelations)
    3768                 :             :         {
    3769                 :             :                 /* Set up the array, indexed by child relid */
    3770                 :           0 :                 int                     ntables = list_length(dpns->rtable);
    3771                 :           0 :                 ListCell   *lc;
    3772                 :             : 
    3773                 :           0 :                 dpns->appendrels = (AppendRelInfo **)
    3774                 :           0 :                         palloc0((ntables + 1) * sizeof(AppendRelInfo *));
    3775   [ #  #  #  #  :           0 :                 foreach(lc, pstmt->appendRelations)
                   #  # ]
    3776                 :             :                 {
    3777                 :           0 :                         AppendRelInfo *appinfo = lfirst_node(AppendRelInfo, lc);
    3778                 :           0 :                         Index           crelid = appinfo->child_relid;
    3779                 :             : 
    3780         [ #  # ]:           0 :                         Assert(crelid > 0 && crelid <= ntables);
    3781         [ #  # ]:           0 :                         Assert(dpns->appendrels[crelid] == NULL);
    3782                 :           0 :                         dpns->appendrels[crelid] = appinfo;
    3783                 :           0 :                 }
    3784                 :           0 :         }
    3785                 :             :         else
    3786                 :           0 :                 dpns->appendrels = NULL;     /* don't need it */
    3787                 :             : 
    3788                 :             :         /*
    3789                 :             :          * Set up column name aliases, ignoring any join RTEs; they don't matter
    3790                 :             :          * because plan trees don't contain any join alias Vars.
    3791                 :             :          */
    3792                 :           0 :         set_simple_column_names(dpns);
    3793                 :             : 
    3794                 :             :         /* Return a one-deep namespace stack */
    3795                 :           0 :         return list_make1(dpns);
    3796                 :           0 : }
    3797                 :             : 
    3798                 :             : /*
    3799                 :             :  * set_deparse_context_plan - Specify Plan node containing expression
    3800                 :             :  *
    3801                 :             :  * When deparsing an expression in a Plan tree, we might have to resolve
    3802                 :             :  * OUTER_VAR, INNER_VAR, or INDEX_VAR references.  To do this, the caller must
    3803                 :             :  * provide the parent Plan node.  Then OUTER_VAR and INNER_VAR references
    3804                 :             :  * can be resolved by drilling down into the left and right child plans.
    3805                 :             :  * Similarly, INDEX_VAR references can be resolved by reference to the
    3806                 :             :  * indextlist given in a parent IndexOnlyScan node, or to the scan tlist in
    3807                 :             :  * ForeignScan and CustomScan nodes.  (Note that we don't currently support
    3808                 :             :  * deparsing of indexquals in regular IndexScan or BitmapIndexScan nodes;
    3809                 :             :  * for those, we can only deparse the indexqualorig fields, which won't
    3810                 :             :  * contain INDEX_VAR Vars.)
    3811                 :             :  *
    3812                 :             :  * The ancestors list is a list of the Plan's parent Plan and SubPlan nodes,
    3813                 :             :  * the most-closely-nested first.  This is needed to resolve PARAM_EXEC
    3814                 :             :  * Params.  Note we assume that all the Plan nodes share the same rtable.
    3815                 :             :  *
    3816                 :             :  * For a ModifyTable plan, we might also need to resolve references to OLD/NEW
    3817                 :             :  * variables in the RETURNING list, so we copy the alias names of the OLD and
    3818                 :             :  * NEW rows from the ModifyTable plan node.
    3819                 :             :  *
    3820                 :             :  * Once this function has been called, deparse_expression() can be called on
    3821                 :             :  * subsidiary expression(s) of the specified Plan node.  To deparse
    3822                 :             :  * expressions of a different Plan node in the same Plan tree, re-call this
    3823                 :             :  * function to identify the new parent Plan node.
    3824                 :             :  *
    3825                 :             :  * The result is the same List passed in; this is a notational convenience.
    3826                 :             :  */
    3827                 :             : List *
    3828                 :           0 : set_deparse_context_plan(List *dpcontext, Plan *plan, List *ancestors)
    3829                 :             : {
    3830                 :           0 :         deparse_namespace *dpns;
    3831                 :             : 
    3832                 :             :         /* Should always have one-entry namespace list for Plan deparsing */
    3833         [ #  # ]:           0 :         Assert(list_length(dpcontext) == 1);
    3834                 :           0 :         dpns = (deparse_namespace *) linitial(dpcontext);
    3835                 :             : 
    3836                 :             :         /* Set our attention on the specific plan node passed in */
    3837                 :           0 :         dpns->ancestors = ancestors;
    3838                 :           0 :         set_deparse_plan(dpns, plan);
    3839                 :             : 
    3840                 :             :         /* For ModifyTable, set aliases for OLD and NEW in RETURNING */
    3841         [ #  # ]:           0 :         if (IsA(plan, ModifyTable))
    3842                 :             :         {
    3843                 :           0 :                 dpns->ret_old_alias = ((ModifyTable *) plan)->returningOldAlias;
    3844                 :           0 :                 dpns->ret_new_alias = ((ModifyTable *) plan)->returningNewAlias;
    3845                 :           0 :         }
    3846                 :             : 
    3847                 :           0 :         return dpcontext;
    3848                 :           0 : }
    3849                 :             : 
    3850                 :             : /*
    3851                 :             :  * select_rtable_names_for_explain      - Select RTE aliases for EXPLAIN
    3852                 :             :  *
    3853                 :             :  * Determine the relation aliases we'll use during an EXPLAIN operation.
    3854                 :             :  * This is just a frontend to set_rtable_names.  We have to expose the aliases
    3855                 :             :  * to EXPLAIN because EXPLAIN needs to know the right alias names to print.
    3856                 :             :  */
    3857                 :             : List *
    3858                 :           0 : select_rtable_names_for_explain(List *rtable, Bitmapset *rels_used)
    3859                 :             : {
    3860                 :           0 :         deparse_namespace dpns;
    3861                 :             : 
    3862                 :           0 :         memset(&dpns, 0, sizeof(dpns));
    3863                 :           0 :         dpns.rtable = rtable;
    3864                 :           0 :         dpns.subplans = NIL;
    3865                 :           0 :         dpns.ctes = NIL;
    3866                 :           0 :         dpns.appendrels = NULL;
    3867                 :           0 :         set_rtable_names(&dpns, NIL, rels_used);
    3868                 :             :         /* We needn't bother computing column aliases yet */
    3869                 :             : 
    3870                 :           0 :         return dpns.rtable_names;
    3871                 :           0 : }
    3872                 :             : 
    3873                 :             : /*
    3874                 :             :  * set_rtable_names: select RTE aliases to be used in printing a query
    3875                 :             :  *
    3876                 :             :  * We fill in dpns->rtable_names with a list of names that is one-for-one with
    3877                 :             :  * the already-filled dpns->rtable list.  Each RTE name is unique among those
    3878                 :             :  * in the new namespace plus any ancestor namespaces listed in
    3879                 :             :  * parent_namespaces.
    3880                 :             :  *
    3881                 :             :  * If rels_used isn't NULL, only RTE indexes listed in it are given aliases.
    3882                 :             :  *
    3883                 :             :  * Note that this function is only concerned with relation names, not column
    3884                 :             :  * names.
    3885                 :             :  */
    3886                 :             : static void
    3887                 :           0 : set_rtable_names(deparse_namespace *dpns, List *parent_namespaces,
    3888                 :             :                                  Bitmapset *rels_used)
    3889                 :             : {
    3890                 :           0 :         HASHCTL         hash_ctl;
    3891                 :           0 :         HTAB       *names_hash;
    3892                 :           0 :         NameHashEntry *hentry;
    3893                 :           0 :         bool            found;
    3894                 :           0 :         int                     rtindex;
    3895                 :           0 :         ListCell   *lc;
    3896                 :             : 
    3897                 :           0 :         dpns->rtable_names = NIL;
    3898                 :             :         /* nothing more to do if empty rtable */
    3899         [ #  # ]:           0 :         if (dpns->rtable == NIL)
    3900                 :           0 :                 return;
    3901                 :             : 
    3902                 :             :         /*
    3903                 :             :          * We use a hash table to hold known names, so that this process is O(N)
    3904                 :             :          * not O(N^2) for N names.
    3905                 :             :          */
    3906                 :           0 :         hash_ctl.keysize = NAMEDATALEN;
    3907                 :           0 :         hash_ctl.entrysize = sizeof(NameHashEntry);
    3908                 :           0 :         hash_ctl.hcxt = CurrentMemoryContext;
    3909                 :           0 :         names_hash = hash_create("set_rtable_names names",
    3910                 :           0 :                                                          list_length(dpns->rtable),
    3911                 :             :                                                          &hash_ctl,
    3912                 :             :                                                          HASH_ELEM | HASH_STRINGS | HASH_CONTEXT);
    3913                 :             : 
    3914                 :             :         /* Preload the hash table with names appearing in parent_namespaces */
    3915   [ #  #  #  #  :           0 :         foreach(lc, parent_namespaces)
                   #  # ]
    3916                 :             :         {
    3917                 :           0 :                 deparse_namespace *olddpns = (deparse_namespace *) lfirst(lc);
    3918                 :           0 :                 ListCell   *lc2;
    3919                 :             : 
    3920   [ #  #  #  #  :           0 :                 foreach(lc2, olddpns->rtable_names)
                   #  # ]
    3921                 :             :                 {
    3922                 :           0 :                         char       *oldname = (char *) lfirst(lc2);
    3923                 :             : 
    3924         [ #  # ]:           0 :                         if (oldname == NULL)
    3925                 :           0 :                                 continue;
    3926                 :           0 :                         hentry = (NameHashEntry *) hash_search(names_hash,
    3927                 :           0 :                                                                                                    oldname,
    3928                 :             :                                                                                                    HASH_ENTER,
    3929                 :             :                                                                                                    &found);
    3930                 :             :                         /* we do not complain about duplicate names in parent namespaces */
    3931                 :           0 :                         hentry->counter = 0;
    3932         [ #  # ]:           0 :                 }
    3933                 :           0 :         }
    3934                 :             : 
    3935                 :             :         /* Now we can scan the rtable */
    3936                 :           0 :         rtindex = 1;
    3937   [ #  #  #  #  :           0 :         foreach(lc, dpns->rtable)
                   #  # ]
    3938                 :             :         {
    3939                 :           0 :                 RangeTblEntry *rte = (RangeTblEntry *) lfirst(lc);
    3940                 :           0 :                 char       *refname;
    3941                 :             : 
    3942                 :             :                 /* Just in case this takes an unreasonable amount of time ... */
    3943         [ #  # ]:           0 :                 CHECK_FOR_INTERRUPTS();
    3944                 :             : 
    3945   [ #  #  #  # ]:           0 :                 if (rels_used && !bms_is_member(rtindex, rels_used))
    3946                 :             :                 {
    3947                 :             :                         /* Ignore unreferenced RTE */
    3948                 :           0 :                         refname = NULL;
    3949                 :           0 :                 }
    3950         [ #  # ]:           0 :                 else if (rte->alias)
    3951                 :             :                 {
    3952                 :             :                         /* If RTE has a user-defined alias, prefer that */
    3953                 :           0 :                         refname = rte->alias->aliasname;
    3954                 :           0 :                 }
    3955         [ #  # ]:           0 :                 else if (rte->rtekind == RTE_RELATION)
    3956                 :             :                 {
    3957                 :             :                         /* Use the current actual name of the relation */
    3958                 :           0 :                         refname = get_rel_name(rte->relid);
    3959                 :           0 :                 }
    3960         [ #  # ]:           0 :                 else if (rte->rtekind == RTE_JOIN)
    3961                 :             :                 {
    3962                 :             :                         /* Unnamed join has no refname */
    3963                 :           0 :                         refname = NULL;
    3964                 :           0 :                 }
    3965                 :             :                 else
    3966                 :             :                 {
    3967                 :             :                         /* Otherwise use whatever the parser assigned */
    3968                 :           0 :                         refname = rte->eref->aliasname;
    3969                 :             :                 }
    3970                 :             : 
    3971                 :             :                 /*
    3972                 :             :                  * If the selected name isn't unique, append digits to make it so, and
    3973                 :             :                  * make a new hash entry for it once we've got a unique name.  For a
    3974                 :             :                  * very long input name, we might have to truncate to stay within
    3975                 :             :                  * NAMEDATALEN.
    3976                 :             :                  */
    3977         [ #  # ]:           0 :                 if (refname)
    3978                 :             :                 {
    3979                 :           0 :                         hentry = (NameHashEntry *) hash_search(names_hash,
    3980                 :           0 :                                                                                                    refname,
    3981                 :             :                                                                                                    HASH_ENTER,
    3982                 :             :                                                                                                    &found);
    3983         [ #  # ]:           0 :                         if (found)
    3984                 :             :                         {
    3985                 :             :                                 /* Name already in use, must choose a new one */
    3986                 :           0 :                                 int                     refnamelen = strlen(refname);
    3987                 :           0 :                                 char       *modname = (char *) palloc(refnamelen + 16);
    3988                 :           0 :                                 NameHashEntry *hentry2;
    3989                 :             : 
    3990                 :           0 :                                 do
    3991                 :             :                                 {
    3992                 :           0 :                                         hentry->counter++;
    3993                 :           0 :                                         for (;;)
    3994                 :             :                                         {
    3995                 :           0 :                                                 memcpy(modname, refname, refnamelen);
    3996                 :           0 :                                                 sprintf(modname + refnamelen, "_%d", hentry->counter);
    3997         [ #  # ]:           0 :                                                 if (strlen(modname) < NAMEDATALEN)
    3998                 :           0 :                                                         break;
    3999                 :             :                                                 /* drop chars from refname to keep all the digits */
    4000                 :           0 :                                                 refnamelen = pg_mbcliplen(refname, refnamelen,
    4001                 :           0 :                                                                                                   refnamelen - 1);
    4002                 :             :                                         }
    4003                 :           0 :                                         hentry2 = (NameHashEntry *) hash_search(names_hash,
    4004                 :           0 :                                                                                                                         modname,
    4005                 :             :                                                                                                                         HASH_ENTER,
    4006                 :             :                                                                                                                         &found);
    4007         [ #  # ]:           0 :                                 } while (found);
    4008                 :           0 :                                 hentry2->counter = 0;        /* init new hash entry */
    4009                 :           0 :                                 refname = modname;
    4010                 :           0 :                         }
    4011                 :             :                         else
    4012                 :             :                         {
    4013                 :             :                                 /* Name not previously used, need only initialize hentry */
    4014                 :           0 :                                 hentry->counter = 0;
    4015                 :             :                         }
    4016                 :           0 :                 }
    4017                 :             : 
    4018                 :           0 :                 dpns->rtable_names = lappend(dpns->rtable_names, refname);
    4019                 :           0 :                 rtindex++;
    4020                 :           0 :         }
    4021                 :             : 
    4022                 :           0 :         hash_destroy(names_hash);
    4023                 :           0 : }
    4024                 :             : 
    4025                 :             : /*
    4026                 :             :  * set_deparse_for_query: set up deparse_namespace for deparsing a Query tree
    4027                 :             :  *
    4028                 :             :  * For convenience, this is defined to initialize the deparse_namespace struct
    4029                 :             :  * from scratch.
    4030                 :             :  */
    4031                 :             : static void
    4032                 :           0 : set_deparse_for_query(deparse_namespace *dpns, Query *query,
    4033                 :             :                                           List *parent_namespaces)
    4034                 :             : {
    4035                 :           0 :         ListCell   *lc;
    4036                 :           0 :         ListCell   *lc2;
    4037                 :             : 
    4038                 :             :         /* Initialize *dpns and fill rtable/ctes links */
    4039                 :           0 :         memset(dpns, 0, sizeof(deparse_namespace));
    4040                 :           0 :         dpns->rtable = query->rtable;
    4041                 :           0 :         dpns->subplans = NIL;
    4042                 :           0 :         dpns->ctes = query->cteList;
    4043                 :           0 :         dpns->appendrels = NULL;
    4044                 :           0 :         dpns->ret_old_alias = query->returningOldAlias;
    4045                 :           0 :         dpns->ret_new_alias = query->returningNewAlias;
    4046                 :             : 
    4047                 :             :         /* Assign a unique relation alias to each RTE */
    4048                 :           0 :         set_rtable_names(dpns, parent_namespaces, NULL);
    4049                 :             : 
    4050                 :             :         /* Initialize dpns->rtable_columns to contain zeroed structs */
    4051                 :           0 :         dpns->rtable_columns = NIL;
    4052         [ #  # ]:           0 :         while (list_length(dpns->rtable_columns) < list_length(dpns->rtable))
    4053                 :           0 :                 dpns->rtable_columns = lappend(dpns->rtable_columns,
    4054                 :           0 :                                                                            palloc0(sizeof(deparse_columns)));
    4055                 :             : 
    4056                 :             :         /* If it's a utility query, it won't have a jointree */
    4057         [ #  # ]:           0 :         if (query->jointree)
    4058                 :             :         {
    4059                 :             :                 /* Detect whether global uniqueness of USING names is needed */
    4060                 :           0 :                 dpns->unique_using =
    4061                 :           0 :                         has_dangerous_join_using(dpns, (Node *) query->jointree);
    4062                 :             : 
    4063                 :             :                 /*
    4064                 :             :                  * Select names for columns merged by USING, via a recursive pass over
    4065                 :             :                  * the query jointree.
    4066                 :             :                  */
    4067                 :           0 :                 set_using_names(dpns, (Node *) query->jointree, NIL);
    4068                 :           0 :         }
    4069                 :             : 
    4070                 :             :         /*
    4071                 :             :          * Now assign remaining column aliases for each RTE.  We do this in a
    4072                 :             :          * linear scan of the rtable, so as to process RTEs whether or not they
    4073                 :             :          * are in the jointree (we mustn't miss NEW.*, INSERT target relations,
    4074                 :             :          * etc).  JOIN RTEs must be processed after their children, but this is
    4075                 :             :          * okay because they appear later in the rtable list than their children
    4076                 :             :          * (cf Asserts in identify_join_columns()).
    4077                 :             :          */
    4078   [ #  #  #  #  :           0 :         forboth(lc, dpns->rtable, lc2, dpns->rtable_columns)
          #  #  #  #  #  
                #  #  # ]
    4079                 :             :         {
    4080                 :           0 :                 RangeTblEntry *rte = (RangeTblEntry *) lfirst(lc);
    4081                 :           0 :                 deparse_columns *colinfo = (deparse_columns *) lfirst(lc2);
    4082                 :             : 
    4083         [ #  # ]:           0 :                 if (rte->rtekind == RTE_JOIN)
    4084                 :           0 :                         set_join_column_names(dpns, rte, colinfo);
    4085                 :             :                 else
    4086                 :           0 :                         set_relation_column_names(dpns, rte, colinfo);
    4087                 :           0 :         }
    4088                 :           0 : }
    4089                 :             : 
    4090                 :             : /*
    4091                 :             :  * set_simple_column_names: fill in column aliases for non-query situations
    4092                 :             :  *
    4093                 :             :  * This handles EXPLAIN and cases where we only have relation RTEs.  Without
    4094                 :             :  * a join tree, we can't do anything smart about join RTEs, but we don't
    4095                 :             :  * need to, because EXPLAIN should never see join alias Vars anyway.
    4096                 :             :  * If we find a join RTE we'll just skip it, leaving its deparse_columns
    4097                 :             :  * struct all-zero.  If somehow we try to deparse a join alias Var, we'll
    4098                 :             :  * error out cleanly because the struct's num_cols will be zero.
    4099                 :             :  */
    4100                 :             : static void
    4101                 :           0 : set_simple_column_names(deparse_namespace *dpns)
    4102                 :             : {
    4103                 :           0 :         ListCell   *lc;
    4104                 :           0 :         ListCell   *lc2;
    4105                 :             : 
    4106                 :             :         /* Initialize dpns->rtable_columns to contain zeroed structs */
    4107                 :           0 :         dpns->rtable_columns = NIL;
    4108         [ #  # ]:           0 :         while (list_length(dpns->rtable_columns) < list_length(dpns->rtable))
    4109                 :           0 :                 dpns->rtable_columns = lappend(dpns->rtable_columns,
    4110                 :           0 :                                                                            palloc0(sizeof(deparse_columns)));
    4111                 :             : 
    4112                 :             :         /* Assign unique column aliases within each non-join RTE */
    4113   [ #  #  #  #  :           0 :         forboth(lc, dpns->rtable, lc2, dpns->rtable_columns)
          #  #  #  #  #  
                #  #  # ]
    4114                 :             :         {
    4115                 :           0 :                 RangeTblEntry *rte = (RangeTblEntry *) lfirst(lc);
    4116                 :           0 :                 deparse_columns *colinfo = (deparse_columns *) lfirst(lc2);
    4117                 :             : 
    4118         [ #  # ]:           0 :                 if (rte->rtekind != RTE_JOIN)
    4119                 :           0 :                         set_relation_column_names(dpns, rte, colinfo);
    4120                 :           0 :         }
    4121                 :           0 : }
    4122                 :             : 
    4123                 :             : /*
    4124                 :             :  * has_dangerous_join_using: search jointree for unnamed JOIN USING
    4125                 :             :  *
    4126                 :             :  * Merged columns of a JOIN USING may act differently from either of the input
    4127                 :             :  * columns, either because they are merged with COALESCE (in a FULL JOIN) or
    4128                 :             :  * because an implicit coercion of the underlying input column is required.
    4129                 :             :  * In such a case the column must be referenced as a column of the JOIN not as
    4130                 :             :  * a column of either input.  And this is problematic if the join is unnamed
    4131                 :             :  * (alias-less): we cannot qualify the column's name with an RTE name, since
    4132                 :             :  * there is none.  (Forcibly assigning an alias to the join is not a solution,
    4133                 :             :  * since that will prevent legal references to tables below the join.)
    4134                 :             :  * To ensure that every column in the query is unambiguously referenceable,
    4135                 :             :  * we must assign such merged columns names that are globally unique across
    4136                 :             :  * the whole query, aliasing other columns out of the way as necessary.
    4137                 :             :  *
    4138                 :             :  * Because the ensuing re-aliasing is fairly damaging to the readability of
    4139                 :             :  * the query, we don't do this unless we have to.  So, we must pre-scan
    4140                 :             :  * the join tree to see if we have to, before starting set_using_names().
    4141                 :             :  */
    4142                 :             : static bool
    4143                 :           0 : has_dangerous_join_using(deparse_namespace *dpns, Node *jtnode)
    4144                 :             : {
    4145         [ #  # ]:           0 :         if (IsA(jtnode, RangeTblRef))
    4146                 :             :         {
    4147                 :             :                 /* nothing to do here */
    4148                 :           0 :         }
    4149         [ #  # ]:           0 :         else if (IsA(jtnode, FromExpr))
    4150                 :             :         {
    4151                 :           0 :                 FromExpr   *f = (FromExpr *) jtnode;
    4152                 :           0 :                 ListCell   *lc;
    4153                 :             : 
    4154   [ #  #  #  #  :           0 :                 foreach(lc, f->fromlist)
             #  #  #  # ]
    4155                 :             :                 {
    4156         [ #  # ]:           0 :                         if (has_dangerous_join_using(dpns, (Node *) lfirst(lc)))
    4157                 :           0 :                                 return true;
    4158                 :           0 :                 }
    4159         [ #  # ]:           0 :         }
    4160         [ #  # ]:           0 :         else if (IsA(jtnode, JoinExpr))
    4161                 :             :         {
    4162                 :           0 :                 JoinExpr   *j = (JoinExpr *) jtnode;
    4163                 :             : 
    4164                 :             :                 /* Is it an unnamed JOIN with USING? */
    4165   [ #  #  #  # ]:           0 :                 if (j->alias == NULL && j->usingClause)
    4166                 :             :                 {
    4167                 :             :                         /*
    4168                 :             :                          * Yes, so check each join alias var to see if any of them are not
    4169                 :             :                          * simple references to underlying columns.  If so, we have a
    4170                 :             :                          * dangerous situation and must pick unique aliases.
    4171                 :             :                          */
    4172                 :           0 :                         RangeTblEntry *jrte = rt_fetch(j->rtindex, dpns->rtable);
    4173                 :             : 
    4174                 :             :                         /* We need only examine the merged columns */
    4175   [ #  #  #  # ]:           0 :                         for (int i = 0; i < jrte->joinmergedcols; i++)
    4176                 :             :                         {
    4177                 :           0 :                                 Node       *aliasvar = list_nth(jrte->joinaliasvars, i);
    4178                 :             : 
    4179         [ #  # ]:           0 :                                 if (!IsA(aliasvar, Var))
    4180                 :           0 :                                         return true;
    4181         [ #  # ]:           0 :                         }
    4182         [ #  # ]:           0 :                 }
    4183                 :             : 
    4184                 :             :                 /* Nope, but inspect children */
    4185         [ #  # ]:           0 :                 if (has_dangerous_join_using(dpns, j->larg))
    4186                 :           0 :                         return true;
    4187         [ #  # ]:           0 :                 if (has_dangerous_join_using(dpns, j->rarg))
    4188                 :           0 :                         return true;
    4189         [ #  # ]:           0 :         }
    4190                 :             :         else
    4191   [ #  #  #  # ]:           0 :                 elog(ERROR, "unrecognized node type: %d",
    4192                 :             :                          (int) nodeTag(jtnode));
    4193                 :           0 :         return false;
    4194                 :           0 : }
    4195                 :             : 
    4196                 :             : /*
    4197                 :             :  * set_using_names: select column aliases to be used for merged USING columns
    4198                 :             :  *
    4199                 :             :  * We do this during a recursive descent of the query jointree.
    4200                 :             :  * dpns->unique_using must already be set to determine the global strategy.
    4201                 :             :  *
    4202                 :             :  * Column alias info is saved in the dpns->rtable_columns list, which is
    4203                 :             :  * assumed to be filled with pre-zeroed deparse_columns structs.
    4204                 :             :  *
    4205                 :             :  * parentUsing is a list of all USING aliases assigned in parent joins of
    4206                 :             :  * the current jointree node.  (The passed-in list must not be modified.)
    4207                 :             :  *
    4208                 :             :  * Note that we do not use per-deparse_columns hash tables in this function.
    4209                 :             :  * The number of names that need to be assigned should be small enough that
    4210                 :             :  * we don't need to trouble with that.
    4211                 :             :  */
    4212                 :             : static void
    4213                 :           0 : set_using_names(deparse_namespace *dpns, Node *jtnode, List *parentUsing)
    4214                 :             : {
    4215         [ #  # ]:           0 :         if (IsA(jtnode, RangeTblRef))
    4216                 :             :         {
    4217                 :             :                 /* nothing to do now */
    4218                 :           0 :         }
    4219         [ #  # ]:           0 :         else if (IsA(jtnode, FromExpr))
    4220                 :             :         {
    4221                 :           0 :                 FromExpr   *f = (FromExpr *) jtnode;
    4222                 :           0 :                 ListCell   *lc;
    4223                 :             : 
    4224   [ #  #  #  #  :           0 :                 foreach(lc, f->fromlist)
                   #  # ]
    4225                 :           0 :                         set_using_names(dpns, (Node *) lfirst(lc), parentUsing);
    4226                 :           0 :         }
    4227         [ #  # ]:           0 :         else if (IsA(jtnode, JoinExpr))
    4228                 :             :         {
    4229                 :           0 :                 JoinExpr   *j = (JoinExpr *) jtnode;
    4230                 :           0 :                 RangeTblEntry *rte = rt_fetch(j->rtindex, dpns->rtable);
    4231                 :           0 :                 deparse_columns *colinfo = deparse_columns_fetch(j->rtindex, dpns);
    4232                 :           0 :                 int                *leftattnos;
    4233                 :           0 :                 int                *rightattnos;
    4234                 :           0 :                 deparse_columns *leftcolinfo;
    4235                 :           0 :                 deparse_columns *rightcolinfo;
    4236                 :           0 :                 int                     i;
    4237                 :           0 :                 ListCell   *lc;
    4238                 :             : 
    4239                 :             :                 /* Get info about the shape of the join */
    4240                 :           0 :                 identify_join_columns(j, rte, colinfo);
    4241                 :           0 :                 leftattnos = colinfo->leftattnos;
    4242                 :           0 :                 rightattnos = colinfo->rightattnos;
    4243                 :             : 
    4244                 :             :                 /* Look up the not-yet-filled-in child deparse_columns structs */
    4245                 :           0 :                 leftcolinfo = deparse_columns_fetch(colinfo->leftrti, dpns);
    4246                 :           0 :                 rightcolinfo = deparse_columns_fetch(colinfo->rightrti, dpns);
    4247                 :             : 
    4248                 :             :                 /*
    4249                 :             :                  * If this join is unnamed, then we cannot substitute new aliases at
    4250                 :             :                  * this level, so any name requirements pushed down to here must be
    4251                 :             :                  * pushed down again to the children.
    4252                 :             :                  */
    4253         [ #  # ]:           0 :                 if (rte->alias == NULL)
    4254                 :             :                 {
    4255         [ #  # ]:           0 :                         for (i = 0; i < colinfo->num_cols; i++)
    4256                 :             :                         {
    4257                 :           0 :                                 char       *colname = colinfo->colnames[i];
    4258                 :             : 
    4259         [ #  # ]:           0 :                                 if (colname == NULL)
    4260                 :           0 :                                         continue;
    4261                 :             : 
    4262                 :             :                                 /* Push down to left column, unless it's a system column */
    4263         [ #  # ]:           0 :                                 if (leftattnos[i] > 0)
    4264                 :             :                                 {
    4265                 :           0 :                                         expand_colnames_array_to(leftcolinfo, leftattnos[i]);
    4266                 :           0 :                                         leftcolinfo->colnames[leftattnos[i] - 1] = colname;
    4267                 :           0 :                                 }
    4268                 :             : 
    4269                 :             :                                 /* Same on the righthand side */
    4270         [ #  # ]:           0 :                                 if (rightattnos[i] > 0)
    4271                 :             :                                 {
    4272                 :           0 :                                         expand_colnames_array_to(rightcolinfo, rightattnos[i]);
    4273                 :           0 :                                         rightcolinfo->colnames[rightattnos[i] - 1] = colname;
    4274                 :           0 :                                 }
    4275      [ #  #  # ]:           0 :                         }
    4276                 :           0 :                 }
    4277                 :             : 
    4278                 :             :                 /*
    4279                 :             :                  * If there's a USING clause, select the USING column names and push
    4280                 :             :                  * those names down to the children.  We have two strategies:
    4281                 :             :                  *
    4282                 :             :                  * If dpns->unique_using is true, we force all USING names to be
    4283                 :             :                  * unique across the whole query level.  In principle we'd only need
    4284                 :             :                  * the names of dangerous USING columns to be globally unique, but to
    4285                 :             :                  * safely assign all USING names in a single pass, we have to enforce
    4286                 :             :                  * the same uniqueness rule for all of them.  However, if a USING
    4287                 :             :                  * column's name has been pushed down from the parent, we should use
    4288                 :             :                  * it as-is rather than making a uniqueness adjustment.  This is
    4289                 :             :                  * necessary when we're at an unnamed join, and it creates no risk of
    4290                 :             :                  * ambiguity.  Also, if there's a user-written output alias for a
    4291                 :             :                  * merged column, we prefer to use that rather than the input name;
    4292                 :             :                  * this simplifies the logic and seems likely to lead to less aliasing
    4293                 :             :                  * overall.
    4294                 :             :                  *
    4295                 :             :                  * If dpns->unique_using is false, we only need USING names to be
    4296                 :             :                  * unique within their own join RTE.  We still need to honor
    4297                 :             :                  * pushed-down names, though.
    4298                 :             :                  *
    4299                 :             :                  * Though significantly different in results, these two strategies are
    4300                 :             :                  * implemented by the same code, with only the difference of whether
    4301                 :             :                  * to put assigned names into dpns->using_names.
    4302                 :             :                  */
    4303         [ #  # ]:           0 :                 if (j->usingClause)
    4304                 :             :                 {
    4305                 :             :                         /* Copy the input parentUsing list so we don't modify it */
    4306                 :           0 :                         parentUsing = list_copy(parentUsing);
    4307                 :             : 
    4308                 :             :                         /* USING names must correspond to the first join output columns */
    4309                 :           0 :                         expand_colnames_array_to(colinfo, list_length(j->usingClause));
    4310                 :           0 :                         i = 0;
    4311   [ #  #  #  #  :           0 :                         foreach(lc, j->usingClause)
                   #  # ]
    4312                 :             :                         {
    4313                 :           0 :                                 char       *colname = strVal(lfirst(lc));
    4314                 :             : 
    4315                 :             :                                 /* Assert it's a merged column */
    4316         [ #  # ]:           0 :                                 Assert(leftattnos[i] != 0 && rightattnos[i] != 0);
    4317                 :             : 
    4318                 :             :                                 /* Adopt passed-down name if any, else select unique name */
    4319         [ #  # ]:           0 :                                 if (colinfo->colnames[i] != NULL)
    4320                 :           0 :                                         colname = colinfo->colnames[i];
    4321                 :             :                                 else
    4322                 :             :                                 {
    4323                 :             :                                         /* Prefer user-written output alias if any */
    4324   [ #  #  #  # ]:           0 :                                         if (rte->alias && i < list_length(rte->alias->colnames))
    4325                 :           0 :                                                 colname = strVal(list_nth(rte->alias->colnames, i));
    4326                 :             :                                         /* Make it appropriately unique */
    4327                 :           0 :                                         colname = make_colname_unique(colname, dpns, colinfo);
    4328         [ #  # ]:           0 :                                         if (dpns->unique_using)
    4329                 :           0 :                                                 dpns->using_names = lappend(dpns->using_names,
    4330                 :           0 :                                                                                                         colname);
    4331                 :             :                                         /* Save it as output column name, too */
    4332                 :           0 :                                         colinfo->colnames[i] = colname;
    4333                 :             :                                 }
    4334                 :             : 
    4335                 :             :                                 /* Remember selected names for use later */
    4336                 :           0 :                                 colinfo->usingNames = lappend(colinfo->usingNames, colname);
    4337                 :           0 :                                 parentUsing = lappend(parentUsing, colname);
    4338                 :             : 
    4339                 :             :                                 /* Push down to left column, unless it's a system column */
    4340         [ #  # ]:           0 :                                 if (leftattnos[i] > 0)
    4341                 :             :                                 {
    4342                 :           0 :                                         expand_colnames_array_to(leftcolinfo, leftattnos[i]);
    4343                 :           0 :                                         leftcolinfo->colnames[leftattnos[i] - 1] = colname;
    4344                 :           0 :                                 }
    4345                 :             : 
    4346                 :             :                                 /* Same on the righthand side */
    4347         [ #  # ]:           0 :                                 if (rightattnos[i] > 0)
    4348                 :             :                                 {
    4349                 :           0 :                                         expand_colnames_array_to(rightcolinfo, rightattnos[i]);
    4350                 :           0 :                                         rightcolinfo->colnames[rightattnos[i] - 1] = colname;
    4351                 :           0 :                                 }
    4352                 :             : 
    4353                 :           0 :                                 i++;
    4354                 :           0 :                         }
    4355                 :           0 :                 }
    4356                 :             : 
    4357                 :             :                 /* Mark child deparse_columns structs with correct parentUsing info */
    4358                 :           0 :                 leftcolinfo->parentUsing = parentUsing;
    4359                 :           0 :                 rightcolinfo->parentUsing = parentUsing;
    4360                 :             : 
    4361                 :             :                 /* Now recursively assign USING column names in children */
    4362                 :           0 :                 set_using_names(dpns, j->larg, parentUsing);
    4363                 :           0 :                 set_using_names(dpns, j->rarg, parentUsing);
    4364                 :           0 :         }
    4365                 :             :         else
    4366   [ #  #  #  # ]:           0 :                 elog(ERROR, "unrecognized node type: %d",
    4367                 :             :                          (int) nodeTag(jtnode));
    4368                 :           0 : }
    4369                 :             : 
    4370                 :             : /*
    4371                 :             :  * set_relation_column_names: select column aliases for a non-join RTE
    4372                 :             :  *
    4373                 :             :  * Column alias info is saved in *colinfo, which is assumed to be pre-zeroed.
    4374                 :             :  * If any colnames entries are already filled in, those override local
    4375                 :             :  * choices.
    4376                 :             :  */
    4377                 :             : static void
    4378                 :           0 : set_relation_column_names(deparse_namespace *dpns, RangeTblEntry *rte,
    4379                 :             :                                                   deparse_columns *colinfo)
    4380                 :             : {
    4381                 :           0 :         int                     ncolumns;
    4382                 :           0 :         char      **real_colnames;
    4383                 :           0 :         bool            changed_any;
    4384                 :           0 :         int                     noldcolumns;
    4385                 :           0 :         int                     i;
    4386                 :           0 :         int                     j;
    4387                 :             : 
    4388                 :             :         /*
    4389                 :             :          * Construct an array of the current "real" column names of the RTE.
    4390                 :             :          * real_colnames[] will be indexed by physical column number, with NULL
    4391                 :             :          * entries for dropped columns.
    4392                 :             :          */
    4393         [ #  # ]:           0 :         if (rte->rtekind == RTE_RELATION)
    4394                 :             :         {
    4395                 :             :                 /* Relation --- look to the system catalogs for up-to-date info */
    4396                 :           0 :                 Relation        rel;
    4397                 :           0 :                 TupleDesc       tupdesc;
    4398                 :             : 
    4399                 :           0 :                 rel = relation_open(rte->relid, AccessShareLock);
    4400                 :           0 :                 tupdesc = RelationGetDescr(rel);
    4401                 :             : 
    4402                 :           0 :                 ncolumns = tupdesc->natts;
    4403                 :           0 :                 real_colnames = (char **) palloc(ncolumns * sizeof(char *));
    4404                 :             : 
    4405         [ #  # ]:           0 :                 for (i = 0; i < ncolumns; i++)
    4406                 :             :                 {
    4407                 :           0 :                         Form_pg_attribute attr = TupleDescAttr(tupdesc, i);
    4408                 :             : 
    4409         [ #  # ]:           0 :                         if (attr->attisdropped)
    4410                 :           0 :                                 real_colnames[i] = NULL;
    4411                 :             :                         else
    4412                 :           0 :                                 real_colnames[i] = pstrdup(NameStr(attr->attname));
    4413                 :           0 :                 }
    4414                 :           0 :                 relation_close(rel, AccessShareLock);
    4415                 :           0 :         }
    4416                 :             :         else
    4417                 :             :         {
    4418                 :             :                 /* Otherwise get the column names from eref or expandRTE() */
    4419                 :           0 :                 List       *colnames;
    4420                 :           0 :                 ListCell   *lc;
    4421                 :             : 
    4422                 :             :                 /*
    4423                 :             :                  * Functions returning composites have the annoying property that some
    4424                 :             :                  * of the composite type's columns might have been dropped since the
    4425                 :             :                  * query was parsed.  If possible, use expandRTE() to handle that
    4426                 :             :                  * case, since it has the tedious logic needed to find out about
    4427                 :             :                  * dropped columns.  However, if we're explaining a plan, then we
    4428                 :             :                  * don't have rte->functions because the planner thinks that won't be
    4429                 :             :                  * needed later, and that breaks expandRTE().  So in that case we have
    4430                 :             :                  * to rely on rte->eref, which may lead us to report a dropped
    4431                 :             :                  * column's old name; that seems close enough for EXPLAIN's purposes.
    4432                 :             :                  *
    4433                 :             :                  * For non-RELATION, non-FUNCTION RTEs, we can just look at rte->eref,
    4434                 :             :                  * which should be sufficiently up-to-date: no other RTE types can
    4435                 :             :                  * have columns get dropped from under them after parsing.
    4436                 :             :                  */
    4437   [ #  #  #  # ]:           0 :                 if (rte->rtekind == RTE_FUNCTION && rte->functions != NIL)
    4438                 :             :                 {
    4439                 :             :                         /* Since we're not creating Vars, rtindex etc. don't matter */
    4440                 :           0 :                         expandRTE(rte, 1, 0, VAR_RETURNING_DEFAULT, -1,
    4441                 :             :                                           true /* include dropped */ , &colnames, NULL);
    4442                 :           0 :                 }
    4443                 :             :                 else
    4444                 :           0 :                         colnames = rte->eref->colnames;
    4445                 :             : 
    4446                 :           0 :                 ncolumns = list_length(colnames);
    4447                 :           0 :                 real_colnames = (char **) palloc(ncolumns * sizeof(char *));
    4448                 :             : 
    4449                 :           0 :                 i = 0;
    4450   [ #  #  #  #  :           0 :                 foreach(lc, colnames)
                   #  # ]
    4451                 :             :                 {
    4452                 :             :                         /*
    4453                 :             :                          * If the column name we find here is an empty string, then it's a
    4454                 :             :                          * dropped column, so change to NULL.
    4455                 :             :                          */
    4456                 :           0 :                         char       *cname = strVal(lfirst(lc));
    4457                 :             : 
    4458         [ #  # ]:           0 :                         if (cname[0] == '\0')
    4459                 :           0 :                                 cname = NULL;
    4460                 :           0 :                         real_colnames[i] = cname;
    4461                 :           0 :                         i++;
    4462                 :           0 :                 }
    4463                 :           0 :         }
    4464                 :             : 
    4465                 :             :         /*
    4466                 :             :          * Ensure colinfo->colnames has a slot for each column.  (It could be long
    4467                 :             :          * enough already, if we pushed down a name for the last column.)  Note:
    4468                 :             :          * it's possible that there are now more columns than there were when the
    4469                 :             :          * query was parsed, ie colnames could be longer than rte->eref->colnames.
    4470                 :             :          * We must assign unique aliases to the new columns too, else there could
    4471                 :             :          * be unresolved conflicts when the view/rule is reloaded.
    4472                 :             :          */
    4473                 :           0 :         expand_colnames_array_to(colinfo, ncolumns);
    4474         [ #  # ]:           0 :         Assert(colinfo->num_cols == ncolumns);
    4475                 :             : 
    4476                 :             :         /*
    4477                 :             :          * Make sufficiently large new_colnames and is_new_col arrays, too.
    4478                 :             :          *
    4479                 :             :          * Note: because we leave colinfo->num_new_cols zero until after the loop,
    4480                 :             :          * colname_is_unique will not consult that array, which is fine because it
    4481                 :             :          * would only be duplicate effort.
    4482                 :             :          */
    4483                 :           0 :         colinfo->new_colnames = (char **) palloc(ncolumns * sizeof(char *));
    4484                 :           0 :         colinfo->is_new_col = (bool *) palloc(ncolumns * sizeof(bool));
    4485                 :             : 
    4486                 :             :         /* If the RTE is wide enough, use a hash table to avoid O(N^2) costs */
    4487                 :           0 :         build_colinfo_names_hash(colinfo);
    4488                 :             : 
    4489                 :             :         /*
    4490                 :             :          * Scan the columns, select a unique alias for each one, and store it in
    4491                 :             :          * colinfo->colnames and colinfo->new_colnames.  The former array has NULL
    4492                 :             :          * entries for dropped columns, the latter omits them.  Also mark
    4493                 :             :          * new_colnames entries as to whether they are new since parse time; this
    4494                 :             :          * is the case for entries beyond the length of rte->eref->colnames.
    4495                 :             :          */
    4496                 :           0 :         noldcolumns = list_length(rte->eref->colnames);
    4497                 :           0 :         changed_any = false;
    4498                 :           0 :         j = 0;
    4499         [ #  # ]:           0 :         for (i = 0; i < ncolumns; i++)
    4500                 :             :         {
    4501                 :           0 :                 char       *real_colname = real_colnames[i];
    4502                 :           0 :                 char       *colname = colinfo->colnames[i];
    4503                 :             : 
    4504                 :             :                 /* Skip dropped columns */
    4505         [ #  # ]:           0 :                 if (real_colname == NULL)
    4506                 :             :                 {
    4507         [ #  # ]:           0 :                         Assert(colname == NULL);        /* colnames[i] is already NULL */
    4508                 :           0 :                         continue;
    4509                 :             :                 }
    4510                 :             : 
    4511                 :             :                 /* If alias already assigned, that's what to use */
    4512         [ #  # ]:           0 :                 if (colname == NULL)
    4513                 :             :                 {
    4514                 :             :                         /* If user wrote an alias, prefer that over real column name */
    4515   [ #  #  #  # ]:           0 :                         if (rte->alias && i < list_length(rte->alias->colnames))
    4516                 :           0 :                                 colname = strVal(list_nth(rte->alias->colnames, i));
    4517                 :             :                         else
    4518                 :           0 :                                 colname = real_colname;
    4519                 :             : 
    4520                 :             :                         /* Unique-ify and insert into colinfo */
    4521                 :           0 :                         colname = make_colname_unique(colname, dpns, colinfo);
    4522                 :             : 
    4523                 :           0 :                         colinfo->colnames[i] = colname;
    4524                 :           0 :                         add_to_names_hash(colinfo, colname);
    4525                 :           0 :                 }
    4526                 :             : 
    4527                 :             :                 /* Put names of non-dropped columns in new_colnames[] too */
    4528                 :           0 :                 colinfo->new_colnames[j] = colname;
    4529                 :             :                 /* And mark them as new or not */
    4530                 :           0 :                 colinfo->is_new_col[j] = (i >= noldcolumns);
    4531                 :           0 :                 j++;
    4532                 :             : 
    4533                 :             :                 /* Remember if any assigned aliases differ from "real" name */
    4534   [ #  #  #  # ]:           0 :                 if (!changed_any && strcmp(colname, real_colname) != 0)
    4535                 :           0 :                         changed_any = true;
    4536      [ #  #  # ]:           0 :         }
    4537                 :             : 
    4538                 :             :         /* We're now done needing the colinfo's names_hash */
    4539                 :           0 :         destroy_colinfo_names_hash(colinfo);
    4540                 :             : 
    4541                 :             :         /*
    4542                 :             :          * Set correct length for new_colnames[] array.  (Note: if columns have
    4543                 :             :          * been added, colinfo->num_cols includes them, which is not really quite
    4544                 :             :          * right but is harmless, since any new columns must be at the end where
    4545                 :             :          * they won't affect varattnos of pre-existing columns.)
    4546                 :             :          */
    4547                 :           0 :         colinfo->num_new_cols = j;
    4548                 :             : 
    4549                 :             :         /*
    4550                 :             :          * For a relation RTE, we need only print the alias column names if any
    4551                 :             :          * are different from the underlying "real" names.  For a function RTE,
    4552                 :             :          * always emit a complete column alias list; this is to protect against
    4553                 :             :          * possible instability of the default column names (eg, from altering
    4554                 :             :          * parameter names).  For tablefunc RTEs, we never print aliases, because
    4555                 :             :          * the column names are part of the clause itself.  For other RTE types,
    4556                 :             :          * print if we changed anything OR if there were user-written column
    4557                 :             :          * aliases (since the latter would be part of the underlying "reality").
    4558                 :             :          */
    4559         [ #  # ]:           0 :         if (rte->rtekind == RTE_RELATION)
    4560                 :           0 :                 colinfo->printaliases = changed_any;
    4561         [ #  # ]:           0 :         else if (rte->rtekind == RTE_FUNCTION)
    4562                 :           0 :                 colinfo->printaliases = true;
    4563         [ #  # ]:           0 :         else if (rte->rtekind == RTE_TABLEFUNC)
    4564                 :           0 :                 colinfo->printaliases = false;
    4565   [ #  #  #  # ]:           0 :         else if (rte->alias && rte->alias->colnames != NIL)
    4566                 :           0 :                 colinfo->printaliases = true;
    4567                 :             :         else
    4568                 :           0 :                 colinfo->printaliases = changed_any;
    4569                 :           0 : }
    4570                 :             : 
    4571                 :             : /*
    4572                 :             :  * set_join_column_names: select column aliases for a join RTE
    4573                 :             :  *
    4574                 :             :  * Column alias info is saved in *colinfo, which is assumed to be pre-zeroed.
    4575                 :             :  * If any colnames entries are already filled in, those override local
    4576                 :             :  * choices.  Also, names for USING columns were already chosen by
    4577                 :             :  * set_using_names().  We further expect that column alias selection has been
    4578                 :             :  * completed for both input RTEs.
    4579                 :             :  */
    4580                 :             : static void
    4581                 :           0 : set_join_column_names(deparse_namespace *dpns, RangeTblEntry *rte,
    4582                 :             :                                           deparse_columns *colinfo)
    4583                 :             : {
    4584                 :           0 :         deparse_columns *leftcolinfo;
    4585                 :           0 :         deparse_columns *rightcolinfo;
    4586                 :           0 :         bool            changed_any;
    4587                 :           0 :         int                     noldcolumns;
    4588                 :           0 :         int                     nnewcolumns;
    4589                 :           0 :         Bitmapset  *leftmerged = NULL;
    4590                 :           0 :         Bitmapset  *rightmerged = NULL;
    4591                 :           0 :         int                     i;
    4592                 :           0 :         int                     j;
    4593                 :           0 :         int                     ic;
    4594                 :           0 :         int                     jc;
    4595                 :             : 
    4596                 :             :         /* Look up the previously-filled-in child deparse_columns structs */
    4597                 :           0 :         leftcolinfo = deparse_columns_fetch(colinfo->leftrti, dpns);
    4598                 :           0 :         rightcolinfo = deparse_columns_fetch(colinfo->rightrti, dpns);
    4599                 :             : 
    4600                 :             :         /*
    4601                 :             :          * Ensure colinfo->colnames has a slot for each column.  (It could be long
    4602                 :             :          * enough already, if we pushed down a name for the last column.)  Note:
    4603                 :             :          * it's possible that one or both inputs now have more columns than there
    4604                 :             :          * were when the query was parsed, but we'll deal with that below.  We
    4605                 :             :          * only need entries in colnames for pre-existing columns.
    4606                 :             :          */
    4607                 :           0 :         noldcolumns = list_length(rte->eref->colnames);
    4608                 :           0 :         expand_colnames_array_to(colinfo, noldcolumns);
    4609         [ #  # ]:           0 :         Assert(colinfo->num_cols == noldcolumns);
    4610                 :             : 
    4611                 :             :         /* If the RTE is wide enough, use a hash table to avoid O(N^2) costs */
    4612                 :           0 :         build_colinfo_names_hash(colinfo);
    4613                 :             : 
    4614                 :             :         /*
    4615                 :             :          * Scan the join output columns, select an alias for each one, and store
    4616                 :             :          * it in colinfo->colnames.  If there are USING columns, set_using_names()
    4617                 :             :          * already selected their names, so we can start the loop at the first
    4618                 :             :          * non-merged column.
    4619                 :             :          */
    4620                 :           0 :         changed_any = false;
    4621         [ #  # ]:           0 :         for (i = list_length(colinfo->usingNames); i < noldcolumns; i++)
    4622                 :             :         {
    4623                 :           0 :                 char       *colname = colinfo->colnames[i];
    4624                 :           0 :                 char       *real_colname;
    4625                 :             : 
    4626                 :             :                 /* Join column must refer to at least one input column */
    4627   [ #  #  #  # ]:           0 :                 Assert(colinfo->leftattnos[i] != 0 || colinfo->rightattnos[i] != 0);
    4628                 :             : 
    4629                 :             :                 /* Get the child column name */
    4630         [ #  # ]:           0 :                 if (colinfo->leftattnos[i] > 0)
    4631                 :           0 :                         real_colname = leftcolinfo->colnames[colinfo->leftattnos[i] - 1];
    4632         [ #  # ]:           0 :                 else if (colinfo->rightattnos[i] > 0)
    4633                 :           0 :                         real_colname = rightcolinfo->colnames[colinfo->rightattnos[i] - 1];
    4634                 :             :                 else
    4635                 :             :                 {
    4636                 :             :                         /* We're joining system columns --- use eref name */
    4637                 :           0 :                         real_colname = strVal(list_nth(rte->eref->colnames, i));
    4638                 :             :                 }
    4639                 :             : 
    4640                 :             :                 /* If child col has been dropped, no need to assign a join colname */
    4641         [ #  # ]:           0 :                 if (real_colname == NULL)
    4642                 :             :                 {
    4643                 :           0 :                         colinfo->colnames[i] = NULL;
    4644                 :           0 :                         continue;
    4645                 :             :                 }
    4646                 :             : 
    4647                 :             :                 /* In an unnamed join, just report child column names as-is */
    4648         [ #  # ]:           0 :                 if (rte->alias == NULL)
    4649                 :             :                 {
    4650                 :           0 :                         colinfo->colnames[i] = real_colname;
    4651                 :           0 :                         add_to_names_hash(colinfo, real_colname);
    4652                 :           0 :                         continue;
    4653                 :             :                 }
    4654                 :             : 
    4655                 :             :                 /* If alias already assigned, that's what to use */
    4656         [ #  # ]:           0 :                 if (colname == NULL)
    4657                 :             :                 {
    4658                 :             :                         /* If user wrote an alias, prefer that over real column name */
    4659   [ #  #  #  # ]:           0 :                         if (rte->alias && i < list_length(rte->alias->colnames))
    4660                 :           0 :                                 colname = strVal(list_nth(rte->alias->colnames, i));
    4661                 :             :                         else
    4662                 :           0 :                                 colname = real_colname;
    4663                 :             : 
    4664                 :             :                         /* Unique-ify and insert into colinfo */
    4665                 :           0 :                         colname = make_colname_unique(colname, dpns, colinfo);
    4666                 :             : 
    4667                 :           0 :                         colinfo->colnames[i] = colname;
    4668                 :           0 :                         add_to_names_hash(colinfo, colname);
    4669                 :           0 :                 }
    4670                 :             : 
    4671                 :             :                 /* Remember if any assigned aliases differ from "real" name */
    4672   [ #  #  #  # ]:           0 :                 if (!changed_any && strcmp(colname, real_colname) != 0)
    4673                 :           0 :                         changed_any = true;
    4674         [ #  # ]:           0 :         }
    4675                 :             : 
    4676                 :             :         /*
    4677                 :             :          * Calculate number of columns the join would have if it were re-parsed
    4678                 :             :          * now, and create storage for the new_colnames and is_new_col arrays.
    4679                 :             :          *
    4680                 :             :          * Note: colname_is_unique will be consulting new_colnames[] during the
    4681                 :             :          * loops below, so its not-yet-filled entries must be zeroes.
    4682                 :             :          */
    4683                 :           0 :         nnewcolumns = leftcolinfo->num_new_cols + rightcolinfo->num_new_cols -
    4684                 :           0 :                 list_length(colinfo->usingNames);
    4685                 :           0 :         colinfo->num_new_cols = nnewcolumns;
    4686                 :           0 :         colinfo->new_colnames = (char **) palloc0(nnewcolumns * sizeof(char *));
    4687                 :           0 :         colinfo->is_new_col = (bool *) palloc0(nnewcolumns * sizeof(bool));
    4688                 :             : 
    4689                 :             :         /*
    4690                 :             :          * Generating the new_colnames array is a bit tricky since any new columns
    4691                 :             :          * added since parse time must be inserted in the right places.  This code
    4692                 :             :          * must match the parser, which will order a join's columns as merged
    4693                 :             :          * columns first (in USING-clause order), then non-merged columns from the
    4694                 :             :          * left input (in attnum order), then non-merged columns from the right
    4695                 :             :          * input (ditto).  If one of the inputs is itself a join, its columns will
    4696                 :             :          * be ordered according to the same rule, which means newly-added columns
    4697                 :             :          * might not be at the end.  We can figure out what's what by consulting
    4698                 :             :          * the leftattnos and rightattnos arrays plus the input is_new_col arrays.
    4699                 :             :          *
    4700                 :             :          * In these loops, i indexes leftattnos/rightattnos (so it's join varattno
    4701                 :             :          * less one), j indexes new_colnames/is_new_col, and ic/jc have similar
    4702                 :             :          * meanings for the current child RTE.
    4703                 :             :          */
    4704                 :             : 
    4705                 :             :         /* Handle merged columns; they are first and can't be new */
    4706                 :           0 :         i = j = 0;
    4707   [ #  #  #  # ]:           0 :         while (i < noldcolumns &&
    4708         [ #  # ]:           0 :                    colinfo->leftattnos[i] != 0 &&
    4709                 :           0 :                    colinfo->rightattnos[i] != 0)
    4710                 :             :         {
    4711                 :             :                 /* column name is already determined and known unique */
    4712                 :           0 :                 colinfo->new_colnames[j] = colinfo->colnames[i];
    4713                 :           0 :                 colinfo->is_new_col[j] = false;
    4714                 :             : 
    4715                 :             :                 /* build bitmapsets of child attnums of merged columns */
    4716         [ #  # ]:           0 :                 if (colinfo->leftattnos[i] > 0)
    4717                 :           0 :                         leftmerged = bms_add_member(leftmerged, colinfo->leftattnos[i]);
    4718         [ #  # ]:           0 :                 if (colinfo->rightattnos[i] > 0)
    4719                 :           0 :                         rightmerged = bms_add_member(rightmerged, colinfo->rightattnos[i]);
    4720                 :             : 
    4721                 :           0 :                 i++, j++;
    4722                 :             :         }
    4723                 :             : 
    4724                 :             :         /* Handle non-merged left-child columns */
    4725                 :           0 :         ic = 0;
    4726         [ #  # ]:           0 :         for (jc = 0; jc < leftcolinfo->num_new_cols; jc++)
    4727                 :             :         {
    4728                 :           0 :                 char       *child_colname = leftcolinfo->new_colnames[jc];
    4729                 :             : 
    4730         [ #  # ]:           0 :                 if (!leftcolinfo->is_new_col[jc])
    4731                 :             :                 {
    4732                 :             :                         /* Advance ic to next non-dropped old column of left child */
    4733   [ #  #  #  # ]:           0 :                         while (ic < leftcolinfo->num_cols &&
    4734                 :           0 :                                    leftcolinfo->colnames[ic] == NULL)
    4735                 :           0 :                                 ic++;
    4736         [ #  # ]:           0 :                         Assert(ic < leftcolinfo->num_cols);
    4737                 :           0 :                         ic++;
    4738                 :             :                         /* If it is a merged column, we already processed it */
    4739         [ #  # ]:           0 :                         if (bms_is_member(ic, leftmerged))
    4740                 :           0 :                                 continue;
    4741                 :             :                         /* Else, advance i to the corresponding existing join column */
    4742   [ #  #  #  # ]:           0 :                         while (i < colinfo->num_cols &&
    4743                 :           0 :                                    colinfo->colnames[i] == NULL)
    4744                 :           0 :                                 i++;
    4745         [ #  # ]:           0 :                         Assert(i < colinfo->num_cols);
    4746         [ #  # ]:           0 :                         Assert(ic == colinfo->leftattnos[i]);
    4747                 :             :                         /* Use the already-assigned name of this column */
    4748                 :           0 :                         colinfo->new_colnames[j] = colinfo->colnames[i];
    4749                 :           0 :                         i++;
    4750                 :           0 :                 }
    4751                 :             :                 else
    4752                 :             :                 {
    4753                 :             :                         /*
    4754                 :             :                          * Unique-ify the new child column name and assign, unless we're
    4755                 :             :                          * in an unnamed join, in which case just copy
    4756                 :             :                          */
    4757         [ #  # ]:           0 :                         if (rte->alias != NULL)
    4758                 :             :                         {
    4759                 :           0 :                                 colinfo->new_colnames[j] =
    4760                 :           0 :                                         make_colname_unique(child_colname, dpns, colinfo);
    4761   [ #  #  #  # ]:           0 :                                 if (!changed_any &&
    4762                 :           0 :                                         strcmp(colinfo->new_colnames[j], child_colname) != 0)
    4763                 :           0 :                                         changed_any = true;
    4764                 :           0 :                         }
    4765                 :             :                         else
    4766                 :           0 :                                 colinfo->new_colnames[j] = child_colname;
    4767                 :           0 :                         add_to_names_hash(colinfo, colinfo->new_colnames[j]);
    4768                 :             :                 }
    4769                 :             : 
    4770                 :           0 :                 colinfo->is_new_col[j] = leftcolinfo->is_new_col[jc];
    4771                 :           0 :                 j++;
    4772         [ #  # ]:           0 :         }
    4773                 :             : 
    4774                 :             :         /* Handle non-merged right-child columns in exactly the same way */
    4775                 :           0 :         ic = 0;
    4776         [ #  # ]:           0 :         for (jc = 0; jc < rightcolinfo->num_new_cols; jc++)
    4777                 :             :         {
    4778                 :           0 :                 char       *child_colname = rightcolinfo->new_colnames[jc];
    4779                 :             : 
    4780         [ #  # ]:           0 :                 if (!rightcolinfo->is_new_col[jc])
    4781                 :             :                 {
    4782                 :             :                         /* Advance ic to next non-dropped old column of right child */
    4783   [ #  #  #  # ]:           0 :                         while (ic < rightcolinfo->num_cols &&
    4784                 :           0 :                                    rightcolinfo->colnames[ic] == NULL)
    4785                 :           0 :                                 ic++;
    4786         [ #  # ]:           0 :                         Assert(ic < rightcolinfo->num_cols);
    4787                 :           0 :                         ic++;
    4788                 :             :                         /* If it is a merged column, we already processed it */
    4789         [ #  # ]:           0 :                         if (bms_is_member(ic, rightmerged))
    4790                 :           0 :                                 continue;
    4791                 :             :                         /* Else, advance i to the corresponding existing join column */
    4792   [ #  #  #  # ]:           0 :                         while (i < colinfo->num_cols &&
    4793                 :           0 :                                    colinfo->colnames[i] == NULL)
    4794                 :           0 :                                 i++;
    4795         [ #  # ]:           0 :                         Assert(i < colinfo->num_cols);
    4796         [ #  # ]:           0 :                         Assert(ic == colinfo->rightattnos[i]);
    4797                 :             :                         /* Use the already-assigned name of this column */
    4798                 :           0 :                         colinfo->new_colnames[j] = colinfo->colnames[i];
    4799                 :           0 :                         i++;
    4800                 :           0 :                 }
    4801                 :             :                 else
    4802                 :             :                 {
    4803                 :             :                         /*
    4804                 :             :                          * Unique-ify the new child column name and assign, unless we're
    4805                 :             :                          * in an unnamed join, in which case just copy
    4806                 :             :                          */
    4807         [ #  # ]:           0 :                         if (rte->alias != NULL)
    4808                 :             :                         {
    4809                 :           0 :                                 colinfo->new_colnames[j] =
    4810                 :           0 :                                         make_colname_unique(child_colname, dpns, colinfo);
    4811   [ #  #  #  # ]:           0 :                                 if (!changed_any &&
    4812                 :           0 :                                         strcmp(colinfo->new_colnames[j], child_colname) != 0)
    4813                 :           0 :                                         changed_any = true;
    4814                 :           0 :                         }
    4815                 :             :                         else
    4816                 :           0 :                                 colinfo->new_colnames[j] = child_colname;
    4817                 :           0 :                         add_to_names_hash(colinfo, colinfo->new_colnames[j]);
    4818                 :             :                 }
    4819                 :             : 
    4820                 :           0 :                 colinfo->is_new_col[j] = rightcolinfo->is_new_col[jc];
    4821                 :           0 :                 j++;
    4822         [ #  # ]:           0 :         }
    4823                 :             : 
    4824                 :             :         /* Assert we processed the right number of columns */
    4825                 :             : #ifdef USE_ASSERT_CHECKING
    4826   [ #  #  #  # ]:           0 :         while (i < colinfo->num_cols && colinfo->colnames[i] == NULL)
    4827                 :           0 :                 i++;
    4828         [ #  # ]:           0 :         Assert(i == colinfo->num_cols);
    4829         [ #  # ]:           0 :         Assert(j == nnewcolumns);
    4830                 :             : #endif
    4831                 :             : 
    4832                 :             :         /* We're now done needing the colinfo's names_hash */
    4833                 :           0 :         destroy_colinfo_names_hash(colinfo);
    4834                 :             : 
    4835                 :             :         /*
    4836                 :             :          * For a named join, print column aliases if we changed any from the child
    4837                 :             :          * names.  Unnamed joins cannot print aliases.
    4838                 :             :          */
    4839         [ #  # ]:           0 :         if (rte->alias != NULL)
    4840                 :           0 :                 colinfo->printaliases = changed_any;
    4841                 :             :         else
    4842                 :           0 :                 colinfo->printaliases = false;
    4843                 :           0 : }
    4844                 :             : 
    4845                 :             : /*
    4846                 :             :  * colname_is_unique: is colname distinct from already-chosen column names?
    4847                 :             :  *
    4848                 :             :  * dpns is query-wide info, colinfo is for the column's RTE
    4849                 :             :  */
    4850                 :             : static bool
    4851                 :           0 : colname_is_unique(const char *colname, deparse_namespace *dpns,
    4852                 :             :                                   deparse_columns *colinfo)
    4853                 :             : {
    4854                 :           0 :         int                     i;
    4855                 :           0 :         ListCell   *lc;
    4856                 :             : 
    4857                 :             :         /*
    4858                 :             :          * If we have a hash table, consult that instead of linearly scanning the
    4859                 :             :          * colinfo's strings.
    4860                 :             :          */
    4861         [ #  # ]:           0 :         if (colinfo->names_hash)
    4862                 :             :         {
    4863                 :           0 :                 if (hash_search(colinfo->names_hash,
    4864                 :           0 :                                                 colname,
    4865                 :             :                                                 HASH_FIND,
    4866         [ #  # ]:           0 :                                                 NULL) != NULL)
    4867                 :           0 :                         return false;
    4868                 :           0 :         }
    4869                 :             :         else
    4870                 :             :         {
    4871                 :             :                 /* Check against already-assigned column aliases within RTE */
    4872         [ #  # ]:           0 :                 for (i = 0; i < colinfo->num_cols; i++)
    4873                 :             :                 {
    4874                 :           0 :                         char       *oldname = colinfo->colnames[i];
    4875                 :             : 
    4876   [ #  #  #  # ]:           0 :                         if (oldname && strcmp(oldname, colname) == 0)
    4877                 :           0 :                                 return false;
    4878         [ #  # ]:           0 :                 }
    4879                 :             : 
    4880                 :             :                 /*
    4881                 :             :                  * If we're building a new_colnames array, check that too (this will
    4882                 :             :                  * be partially but not completely redundant with the previous checks)
    4883                 :             :                  */
    4884         [ #  # ]:           0 :                 for (i = 0; i < colinfo->num_new_cols; i++)
    4885                 :             :                 {
    4886                 :           0 :                         char       *oldname = colinfo->new_colnames[i];
    4887                 :             : 
    4888   [ #  #  #  # ]:           0 :                         if (oldname && strcmp(oldname, colname) == 0)
    4889                 :           0 :                                 return false;
    4890         [ #  # ]:           0 :                 }
    4891                 :             : 
    4892                 :             :                 /*
    4893                 :             :                  * Also check against names already assigned for parent-join USING
    4894                 :             :                  * cols
    4895                 :             :                  */
    4896   [ #  #  #  #  :           0 :                 foreach(lc, colinfo->parentUsing)
             #  #  #  # ]
    4897                 :             :                 {
    4898                 :           0 :                         char       *oldname = (char *) lfirst(lc);
    4899                 :             : 
    4900         [ #  # ]:           0 :                         if (strcmp(oldname, colname) == 0)
    4901                 :           0 :                                 return false;
    4902         [ #  # ]:           0 :                 }
    4903                 :             :         }
    4904                 :             : 
    4905                 :             :         /*
    4906                 :             :          * Also check against USING-column names that must be globally unique.
    4907                 :             :          * These are not hashed, but there should be few of them.
    4908                 :             :          */
    4909   [ #  #  #  #  :           0 :         foreach(lc, dpns->using_names)
             #  #  #  # ]
    4910                 :             :         {
    4911                 :           0 :                 char       *oldname = (char *) lfirst(lc);
    4912                 :             : 
    4913         [ #  # ]:           0 :                 if (strcmp(oldname, colname) == 0)
    4914                 :           0 :                         return false;
    4915         [ #  # ]:           0 :         }
    4916                 :             : 
    4917                 :           0 :         return true;
    4918                 :           0 : }
    4919                 :             : 
    4920                 :             : /*
    4921                 :             :  * make_colname_unique: modify colname if necessary to make it unique
    4922                 :             :  *
    4923                 :             :  * dpns is query-wide info, colinfo is for the column's RTE
    4924                 :             :  */
    4925                 :             : static char *
    4926                 :           0 : make_colname_unique(char *colname, deparse_namespace *dpns,
    4927                 :             :                                         deparse_columns *colinfo)
    4928                 :             : {
    4929                 :             :         /*
    4930                 :             :          * If the selected name isn't unique, append digits to make it so.  For a
    4931                 :             :          * very long input name, we might have to truncate to stay within
    4932                 :             :          * NAMEDATALEN.
    4933                 :             :          */
    4934         [ #  # ]:           0 :         if (!colname_is_unique(colname, dpns, colinfo))
    4935                 :             :         {
    4936                 :           0 :                 int                     colnamelen = strlen(colname);
    4937                 :           0 :                 char       *modname = (char *) palloc(colnamelen + 16);
    4938                 :           0 :                 int                     i = 0;
    4939                 :             : 
    4940                 :           0 :                 do
    4941                 :             :                 {
    4942                 :           0 :                         i++;
    4943                 :           0 :                         for (;;)
    4944                 :             :                         {
    4945                 :           0 :                                 memcpy(modname, colname, colnamelen);
    4946                 :           0 :                                 sprintf(modname + colnamelen, "_%d", i);
    4947         [ #  # ]:           0 :                                 if (strlen(modname) < NAMEDATALEN)
    4948                 :           0 :                                         break;
    4949                 :             :                                 /* drop chars from colname to keep all the digits */
    4950                 :           0 :                                 colnamelen = pg_mbcliplen(colname, colnamelen,
    4951                 :           0 :                                                                                   colnamelen - 1);
    4952                 :             :                         }
    4953         [ #  # ]:           0 :                 } while (!colname_is_unique(modname, dpns, colinfo));
    4954                 :           0 :                 colname = modname;
    4955                 :           0 :         }
    4956                 :           0 :         return colname;
    4957                 :             : }
    4958                 :             : 
    4959                 :             : /*
    4960                 :             :  * expand_colnames_array_to: make colinfo->colnames at least n items long
    4961                 :             :  *
    4962                 :             :  * Any added array entries are initialized to zero.
    4963                 :             :  */
    4964                 :             : static void
    4965                 :           0 : expand_colnames_array_to(deparse_columns *colinfo, int n)
    4966                 :             : {
    4967         [ #  # ]:           0 :         if (n > colinfo->num_cols)
    4968                 :             :         {
    4969         [ #  # ]:           0 :                 if (colinfo->colnames == NULL)
    4970                 :           0 :                         colinfo->colnames = palloc0_array(char *, n);
    4971                 :             :                 else
    4972                 :           0 :                         colinfo->colnames = repalloc0_array(colinfo->colnames, char *, colinfo->num_cols, n);
    4973                 :           0 :                 colinfo->num_cols = n;
    4974                 :           0 :         }
    4975                 :           0 : }
    4976                 :             : 
    4977                 :             : /*
    4978                 :             :  * build_colinfo_names_hash: optionally construct a hash table for colinfo
    4979                 :             :  */
    4980                 :             : static void
    4981                 :           0 : build_colinfo_names_hash(deparse_columns *colinfo)
    4982                 :             : {
    4983                 :           0 :         HASHCTL         hash_ctl;
    4984                 :           0 :         int                     i;
    4985                 :           0 :         ListCell   *lc;
    4986                 :             : 
    4987                 :             :         /*
    4988                 :             :          * Use a hash table only for RTEs with at least 32 columns.  (The cutoff
    4989                 :             :          * is somewhat arbitrary, but let's choose it so that this code does get
    4990                 :             :          * exercised in the regression tests.)
    4991                 :             :          */
    4992         [ #  # ]:           0 :         if (colinfo->num_cols < 32)
    4993                 :           0 :                 return;
    4994                 :             : 
    4995                 :             :         /*
    4996                 :             :          * Set up the hash table.  The entries are just strings with no other
    4997                 :             :          * payload.
    4998                 :             :          */
    4999                 :           0 :         hash_ctl.keysize = NAMEDATALEN;
    5000                 :           0 :         hash_ctl.entrysize = NAMEDATALEN;
    5001                 :           0 :         hash_ctl.hcxt = CurrentMemoryContext;
    5002                 :           0 :         colinfo->names_hash = hash_create("deparse_columns names",
    5003                 :           0 :                                                                           colinfo->num_cols + colinfo->num_new_cols,
    5004                 :             :                                                                           &hash_ctl,
    5005                 :             :                                                                           HASH_ELEM | HASH_STRINGS | HASH_CONTEXT);
    5006                 :             : 
    5007                 :             :         /*
    5008                 :             :          * Preload the hash table with any names already present (these would have
    5009                 :             :          * come from set_using_names).
    5010                 :             :          */
    5011         [ #  # ]:           0 :         for (i = 0; i < colinfo->num_cols; i++)
    5012                 :             :         {
    5013                 :           0 :                 char       *oldname = colinfo->colnames[i];
    5014                 :             : 
    5015         [ #  # ]:           0 :                 if (oldname)
    5016                 :           0 :                         add_to_names_hash(colinfo, oldname);
    5017                 :           0 :         }
    5018                 :             : 
    5019         [ #  # ]:           0 :         for (i = 0; i < colinfo->num_new_cols; i++)
    5020                 :             :         {
    5021                 :           0 :                 char       *oldname = colinfo->new_colnames[i];
    5022                 :             : 
    5023         [ #  # ]:           0 :                 if (oldname)
    5024                 :           0 :                         add_to_names_hash(colinfo, oldname);
    5025                 :           0 :         }
    5026                 :             : 
    5027   [ #  #  #  #  :           0 :         foreach(lc, colinfo->parentUsing)
                   #  # ]
    5028                 :             :         {
    5029                 :           0 :                 char       *oldname = (char *) lfirst(lc);
    5030                 :             : 
    5031                 :           0 :                 add_to_names_hash(colinfo, oldname);
    5032                 :           0 :         }
    5033         [ #  # ]:           0 : }
    5034                 :             : 
    5035                 :             : /*
    5036                 :             :  * add_to_names_hash: add a string to the names_hash, if we're using one
    5037                 :             :  */
    5038                 :             : static void
    5039                 :           0 : add_to_names_hash(deparse_columns *colinfo, const char *name)
    5040                 :             : {
    5041         [ #  # ]:           0 :         if (colinfo->names_hash)
    5042                 :           0 :                 (void) hash_search(colinfo->names_hash,
    5043                 :           0 :                                                    name,
    5044                 :             :                                                    HASH_ENTER,
    5045                 :             :                                                    NULL);
    5046                 :           0 : }
    5047                 :             : 
    5048                 :             : /*
    5049                 :             :  * destroy_colinfo_names_hash: destroy hash table when done with it
    5050                 :             :  */
    5051                 :             : static void
    5052                 :           0 : destroy_colinfo_names_hash(deparse_columns *colinfo)
    5053                 :             : {
    5054         [ #  # ]:           0 :         if (colinfo->names_hash)
    5055                 :             :         {
    5056                 :           0 :                 hash_destroy(colinfo->names_hash);
    5057                 :           0 :                 colinfo->names_hash = NULL;
    5058                 :           0 :         }
    5059                 :           0 : }
    5060                 :             : 
    5061                 :             : /*
    5062                 :             :  * identify_join_columns: figure out where columns of a join come from
    5063                 :             :  *
    5064                 :             :  * Fills the join-specific fields of the colinfo struct, except for
    5065                 :             :  * usingNames which is filled later.
    5066                 :             :  */
    5067                 :             : static void
    5068                 :           0 : identify_join_columns(JoinExpr *j, RangeTblEntry *jrte,
    5069                 :             :                                           deparse_columns *colinfo)
    5070                 :             : {
    5071                 :           0 :         int                     numjoincols;
    5072                 :           0 :         int                     jcolno;
    5073                 :           0 :         int                     rcolno;
    5074                 :           0 :         ListCell   *lc;
    5075                 :             : 
    5076                 :             :         /* Extract left/right child RT indexes */
    5077         [ #  # ]:           0 :         if (IsA(j->larg, RangeTblRef))
    5078                 :           0 :                 colinfo->leftrti = ((RangeTblRef *) j->larg)->rtindex;
    5079         [ #  # ]:           0 :         else if (IsA(j->larg, JoinExpr))
    5080                 :           0 :                 colinfo->leftrti = ((JoinExpr *) j->larg)->rtindex;
    5081                 :             :         else
    5082   [ #  #  #  # ]:           0 :                 elog(ERROR, "unrecognized node type in jointree: %d",
    5083                 :             :                          (int) nodeTag(j->larg));
    5084         [ #  # ]:           0 :         if (IsA(j->rarg, RangeTblRef))
    5085                 :           0 :                 colinfo->rightrti = ((RangeTblRef *) j->rarg)->rtindex;
    5086         [ #  # ]:           0 :         else if (IsA(j->rarg, JoinExpr))
    5087                 :           0 :                 colinfo->rightrti = ((JoinExpr *) j->rarg)->rtindex;
    5088                 :             :         else
    5089   [ #  #  #  # ]:           0 :                 elog(ERROR, "unrecognized node type in jointree: %d",
    5090                 :             :                          (int) nodeTag(j->rarg));
    5091                 :             : 
    5092                 :             :         /* Assert children will be processed earlier than join in second pass */
    5093         [ #  # ]:           0 :         Assert(colinfo->leftrti < j->rtindex);
    5094         [ #  # ]:           0 :         Assert(colinfo->rightrti < j->rtindex);
    5095                 :             : 
    5096                 :             :         /* Initialize result arrays with zeroes */
    5097                 :           0 :         numjoincols = list_length(jrte->joinaliasvars);
    5098         [ #  # ]:           0 :         Assert(numjoincols == list_length(jrte->eref->colnames));
    5099                 :           0 :         colinfo->leftattnos = (int *) palloc0(numjoincols * sizeof(int));
    5100                 :           0 :         colinfo->rightattnos = (int *) palloc0(numjoincols * sizeof(int));
    5101                 :             : 
    5102                 :             :         /*
    5103                 :             :          * Deconstruct RTE's joinleftcols/joinrightcols into desired format.
    5104                 :             :          * Recall that the column(s) merged due to USING are the first column(s)
    5105                 :             :          * of the join output.  We need not do anything special while scanning
    5106                 :             :          * joinleftcols, but while scanning joinrightcols we must distinguish
    5107                 :             :          * merged from unmerged columns.
    5108                 :             :          */
    5109                 :           0 :         jcolno = 0;
    5110   [ #  #  #  #  :           0 :         foreach(lc, jrte->joinleftcols)
                   #  # ]
    5111                 :             :         {
    5112                 :           0 :                 int                     leftattno = lfirst_int(lc);
    5113                 :             : 
    5114                 :           0 :                 colinfo->leftattnos[jcolno++] = leftattno;
    5115                 :           0 :         }
    5116                 :           0 :         rcolno = 0;
    5117   [ #  #  #  #  :           0 :         foreach(lc, jrte->joinrightcols)
                   #  # ]
    5118                 :             :         {
    5119                 :           0 :                 int                     rightattno = lfirst_int(lc);
    5120                 :             : 
    5121         [ #  # ]:           0 :                 if (rcolno < jrte->joinmergedcols)        /* merged column? */
    5122                 :           0 :                         colinfo->rightattnos[rcolno] = rightattno;
    5123                 :             :                 else
    5124                 :           0 :                         colinfo->rightattnos[jcolno++] = rightattno;
    5125                 :           0 :                 rcolno++;
    5126                 :           0 :         }
    5127         [ #  # ]:           0 :         Assert(jcolno == numjoincols);
    5128                 :           0 : }
    5129                 :             : 
    5130                 :             : /*
    5131                 :             :  * get_rtable_name: convenience function to get a previously assigned RTE alias
    5132                 :             :  *
    5133                 :             :  * The RTE must belong to the topmost namespace level in "context".
    5134                 :             :  */
    5135                 :             : static char *
    5136                 :           0 : get_rtable_name(int rtindex, deparse_context *context)
    5137                 :             : {
    5138                 :           0 :         deparse_namespace *dpns = (deparse_namespace *) linitial(context->namespaces);
    5139                 :             : 
    5140         [ #  # ]:           0 :         Assert(rtindex > 0 && rtindex <= list_length(dpns->rtable_names));
    5141                 :           0 :         return (char *) list_nth(dpns->rtable_names, rtindex - 1);
    5142                 :           0 : }
    5143                 :             : 
    5144                 :             : /*
    5145                 :             :  * set_deparse_plan: set up deparse_namespace to parse subexpressions
    5146                 :             :  * of a given Plan node
    5147                 :             :  *
    5148                 :             :  * This sets the plan, outer_plan, inner_plan, outer_tlist, inner_tlist,
    5149                 :             :  * and index_tlist fields.  Caller must already have adjusted the ancestors
    5150                 :             :  * list if necessary.  Note that the rtable, subplans, and ctes fields do
    5151                 :             :  * not need to change when shifting attention to different plan nodes in a
    5152                 :             :  * single plan tree.
    5153                 :             :  */
    5154                 :             : static void
    5155                 :           0 : set_deparse_plan(deparse_namespace *dpns, Plan *plan)
    5156                 :             : {
    5157                 :           0 :         dpns->plan = plan;
    5158                 :             : 
    5159                 :             :         /*
    5160                 :             :          * We special-case Append and MergeAppend to pretend that the first child
    5161                 :             :          * plan is the OUTER referent; we have to interpret OUTER Vars in their
    5162                 :             :          * tlists according to one of the children, and the first one is the most
    5163                 :             :          * natural choice.
    5164                 :             :          */
    5165         [ #  # ]:           0 :         if (IsA(plan, Append))
    5166                 :           0 :                 dpns->outer_plan = linitial(((Append *) plan)->appendplans);
    5167         [ #  # ]:           0 :         else if (IsA(plan, MergeAppend))
    5168                 :           0 :                 dpns->outer_plan = linitial(((MergeAppend *) plan)->mergeplans);
    5169                 :             :         else
    5170                 :           0 :                 dpns->outer_plan = outerPlan(plan);
    5171                 :             : 
    5172         [ #  # ]:           0 :         if (dpns->outer_plan)
    5173                 :           0 :                 dpns->outer_tlist = dpns->outer_plan->targetlist;
    5174                 :             :         else
    5175                 :           0 :                 dpns->outer_tlist = NIL;
    5176                 :             : 
    5177                 :             :         /*
    5178                 :             :          * For a SubqueryScan, pretend the subplan is INNER referent.  (We don't
    5179                 :             :          * use OUTER because that could someday conflict with the normal meaning.)
    5180                 :             :          * Likewise, for a CteScan, pretend the subquery's plan is INNER referent.
    5181                 :             :          * For a WorkTableScan, locate the parent RecursiveUnion plan node and use
    5182                 :             :          * that as INNER referent.
    5183                 :             :          *
    5184                 :             :          * For MERGE, pretend the ModifyTable's source plan (its outer plan) is
    5185                 :             :          * INNER referent.  This is the join from the target relation to the data
    5186                 :             :          * source, and all INNER_VAR Vars in other parts of the query refer to its
    5187                 :             :          * targetlist.
    5188                 :             :          *
    5189                 :             :          * For ON CONFLICT .. UPDATE we just need the inner tlist to point to the
    5190                 :             :          * excluded expression's tlist. (Similar to the SubqueryScan we don't want
    5191                 :             :          * to reuse OUTER, it's used for RETURNING in some modify table cases,
    5192                 :             :          * although not INSERT .. CONFLICT).
    5193                 :             :          */
    5194         [ #  # ]:           0 :         if (IsA(plan, SubqueryScan))
    5195                 :           0 :                 dpns->inner_plan = ((SubqueryScan *) plan)->subplan;
    5196         [ #  # ]:           0 :         else if (IsA(plan, CteScan))
    5197                 :           0 :                 dpns->inner_plan = list_nth(dpns->subplans,
    5198                 :           0 :                                                                         ((CteScan *) plan)->ctePlanId - 1);
    5199         [ #  # ]:           0 :         else if (IsA(plan, WorkTableScan))
    5200                 :           0 :                 dpns->inner_plan = find_recursive_union(dpns,
    5201                 :           0 :                                                                                                 (WorkTableScan *) plan);
    5202         [ #  # ]:           0 :         else if (IsA(plan, ModifyTable))
    5203                 :             :         {
    5204         [ #  # ]:           0 :                 if (((ModifyTable *) plan)->operation == CMD_MERGE)
    5205                 :           0 :                         dpns->inner_plan = outerPlan(plan);
    5206                 :             :                 else
    5207                 :           0 :                         dpns->inner_plan = plan;
    5208                 :           0 :         }
    5209                 :             :         else
    5210                 :           0 :                 dpns->inner_plan = innerPlan(plan);
    5211                 :             : 
    5212   [ #  #  #  # ]:           0 :         if (IsA(plan, ModifyTable) && ((ModifyTable *) plan)->operation == CMD_INSERT)
    5213                 :           0 :                 dpns->inner_tlist = ((ModifyTable *) plan)->exclRelTlist;
    5214         [ #  # ]:           0 :         else if (dpns->inner_plan)
    5215                 :           0 :                 dpns->inner_tlist = dpns->inner_plan->targetlist;
    5216                 :             :         else
    5217                 :           0 :                 dpns->inner_tlist = NIL;
    5218                 :             : 
    5219                 :             :         /* Set up referent for INDEX_VAR Vars, if needed */
    5220         [ #  # ]:           0 :         if (IsA(plan, IndexOnlyScan))
    5221                 :           0 :                 dpns->index_tlist = ((IndexOnlyScan *) plan)->indextlist;
    5222         [ #  # ]:           0 :         else if (IsA(plan, ForeignScan))
    5223                 :           0 :                 dpns->index_tlist = ((ForeignScan *) plan)->fdw_scan_tlist;
    5224         [ #  # ]:           0 :         else if (IsA(plan, CustomScan))
    5225                 :           0 :                 dpns->index_tlist = ((CustomScan *) plan)->custom_scan_tlist;
    5226                 :             :         else
    5227                 :           0 :                 dpns->index_tlist = NIL;
    5228                 :           0 : }
    5229                 :             : 
    5230                 :             : /*
    5231                 :             :  * Locate the ancestor plan node that is the RecursiveUnion generating
    5232                 :             :  * the WorkTableScan's work table.  We can match on wtParam, since that
    5233                 :             :  * should be unique within the plan tree.
    5234                 :             :  */
    5235                 :             : static Plan *
    5236                 :           0 : find_recursive_union(deparse_namespace *dpns, WorkTableScan *wtscan)
    5237                 :             : {
    5238                 :           0 :         ListCell   *lc;
    5239                 :             : 
    5240   [ #  #  #  #  :           0 :         foreach(lc, dpns->ancestors)
             #  #  #  # ]
    5241                 :             :         {
    5242                 :           0 :                 Plan       *ancestor = (Plan *) lfirst(lc);
    5243                 :             : 
    5244   [ #  #  #  # ]:           0 :                 if (IsA(ancestor, RecursiveUnion) &&
    5245                 :           0 :                         ((RecursiveUnion *) ancestor)->wtParam == wtscan->wtParam)
    5246                 :           0 :                         return ancestor;
    5247         [ #  # ]:           0 :         }
    5248   [ #  #  #  # ]:           0 :         elog(ERROR, "could not find RecursiveUnion for WorkTableScan with wtParam %d",
    5249                 :             :                  wtscan->wtParam);
    5250                 :           0 :         return NULL;
    5251                 :           0 : }
    5252                 :             : 
    5253                 :             : /*
    5254                 :             :  * push_child_plan: temporarily transfer deparsing attention to a child plan
    5255                 :             :  *
    5256                 :             :  * When expanding an OUTER_VAR or INNER_VAR reference, we must adjust the
    5257                 :             :  * deparse context in case the referenced expression itself uses
    5258                 :             :  * OUTER_VAR/INNER_VAR.  We modify the top stack entry in-place to avoid
    5259                 :             :  * affecting levelsup issues (although in a Plan tree there really shouldn't
    5260                 :             :  * be any).
    5261                 :             :  *
    5262                 :             :  * Caller must provide a local deparse_namespace variable to save the
    5263                 :             :  * previous state for pop_child_plan.
    5264                 :             :  */
    5265                 :             : static void
    5266                 :           0 : push_child_plan(deparse_namespace *dpns, Plan *plan,
    5267                 :             :                                 deparse_namespace *save_dpns)
    5268                 :             : {
    5269                 :             :         /* Save state for restoration later */
    5270                 :           0 :         *save_dpns = *dpns;
    5271                 :             : 
    5272                 :             :         /* Link current plan node into ancestors list */
    5273                 :           0 :         dpns->ancestors = lcons(dpns->plan, dpns->ancestors);
    5274                 :             : 
    5275                 :             :         /* Set attention on selected child */
    5276                 :           0 :         set_deparse_plan(dpns, plan);
    5277                 :           0 : }
    5278                 :             : 
    5279                 :             : /*
    5280                 :             :  * pop_child_plan: undo the effects of push_child_plan
    5281                 :             :  */
    5282                 :             : static void
    5283                 :           0 : pop_child_plan(deparse_namespace *dpns, deparse_namespace *save_dpns)
    5284                 :             : {
    5285                 :           0 :         List       *ancestors;
    5286                 :             : 
    5287                 :             :         /* Get rid of ancestors list cell added by push_child_plan */
    5288                 :           0 :         ancestors = list_delete_first(dpns->ancestors);
    5289                 :             : 
    5290                 :             :         /* Restore fields changed by push_child_plan */
    5291                 :           0 :         *dpns = *save_dpns;
    5292                 :             : 
    5293                 :             :         /* Make sure dpns->ancestors is right (may be unnecessary) */
    5294                 :           0 :         dpns->ancestors = ancestors;
    5295                 :           0 : }
    5296                 :             : 
    5297                 :             : /*
    5298                 :             :  * push_ancestor_plan: temporarily transfer deparsing attention to an
    5299                 :             :  * ancestor plan
    5300                 :             :  *
    5301                 :             :  * When expanding a Param reference, we must adjust the deparse context
    5302                 :             :  * to match the plan node that contains the expression being printed;
    5303                 :             :  * otherwise we'd fail if that expression itself contains a Param or
    5304                 :             :  * OUTER_VAR/INNER_VAR/INDEX_VAR variable.
    5305                 :             :  *
    5306                 :             :  * The target ancestor is conveniently identified by the ListCell holding it
    5307                 :             :  * in dpns->ancestors.
    5308                 :             :  *
    5309                 :             :  * Caller must provide a local deparse_namespace variable to save the
    5310                 :             :  * previous state for pop_ancestor_plan.
    5311                 :             :  */
    5312                 :             : static void
    5313                 :           0 : push_ancestor_plan(deparse_namespace *dpns, ListCell *ancestor_cell,
    5314                 :             :                                    deparse_namespace *save_dpns)
    5315                 :             : {
    5316                 :           0 :         Plan       *plan = (Plan *) lfirst(ancestor_cell);
    5317                 :             : 
    5318                 :             :         /* Save state for restoration later */
    5319                 :           0 :         *save_dpns = *dpns;
    5320                 :             : 
    5321                 :             :         /* Build a new ancestor list with just this node's ancestors */
    5322                 :           0 :         dpns->ancestors =
    5323                 :           0 :                 list_copy_tail(dpns->ancestors,
    5324                 :           0 :                                            list_cell_number(dpns->ancestors, ancestor_cell) + 1);
    5325                 :             : 
    5326                 :             :         /* Set attention on selected ancestor */
    5327                 :           0 :         set_deparse_plan(dpns, plan);
    5328                 :           0 : }
    5329                 :             : 
    5330                 :             : /*
    5331                 :             :  * pop_ancestor_plan: undo the effects of push_ancestor_plan
    5332                 :             :  */
    5333                 :             : static void
    5334                 :           0 : pop_ancestor_plan(deparse_namespace *dpns, deparse_namespace *save_dpns)
    5335                 :             : {
    5336                 :             :         /* Free the ancestor list made in push_ancestor_plan */
    5337                 :           0 :         list_free(dpns->ancestors);
    5338                 :             : 
    5339                 :             :         /* Restore fields changed by push_ancestor_plan */
    5340                 :           0 :         *dpns = *save_dpns;
    5341                 :           0 : }
    5342                 :             : 
    5343                 :             : 
    5344                 :             : /* ----------
    5345                 :             :  * make_ruledef                 - reconstruct the CREATE RULE command
    5346                 :             :  *                                for a given pg_rewrite tuple
    5347                 :             :  * ----------
    5348                 :             :  */
    5349                 :             : static void
    5350                 :           0 : make_ruledef(StringInfo buf, HeapTuple ruletup, TupleDesc rulettc,
    5351                 :             :                          int prettyFlags)
    5352                 :             : {
    5353                 :           0 :         char       *rulename;
    5354                 :           0 :         char            ev_type;
    5355                 :           0 :         Oid                     ev_class;
    5356                 :           0 :         bool            is_instead;
    5357                 :           0 :         char       *ev_qual;
    5358                 :           0 :         char       *ev_action;
    5359                 :           0 :         List       *actions;
    5360                 :           0 :         Relation        ev_relation;
    5361                 :           0 :         TupleDesc       viewResultDesc = NULL;
    5362                 :           0 :         int                     fno;
    5363                 :           0 :         Datum           dat;
    5364                 :           0 :         bool            isnull;
    5365                 :             : 
    5366                 :             :         /*
    5367                 :             :          * Get the attribute values from the rules tuple
    5368                 :             :          */
    5369                 :           0 :         fno = SPI_fnumber(rulettc, "rulename");
    5370                 :           0 :         dat = SPI_getbinval(ruletup, rulettc, fno, &isnull);
    5371         [ #  # ]:           0 :         Assert(!isnull);
    5372                 :           0 :         rulename = NameStr(*(DatumGetName(dat)));
    5373                 :             : 
    5374                 :           0 :         fno = SPI_fnumber(rulettc, "ev_type");
    5375                 :           0 :         dat = SPI_getbinval(ruletup, rulettc, fno, &isnull);
    5376         [ #  # ]:           0 :         Assert(!isnull);
    5377                 :           0 :         ev_type = DatumGetChar(dat);
    5378                 :             : 
    5379                 :           0 :         fno = SPI_fnumber(rulettc, "ev_class");
    5380                 :           0 :         dat = SPI_getbinval(ruletup, rulettc, fno, &isnull);
    5381         [ #  # ]:           0 :         Assert(!isnull);
    5382                 :           0 :         ev_class = DatumGetObjectId(dat);
    5383                 :             : 
    5384                 :           0 :         fno = SPI_fnumber(rulettc, "is_instead");
    5385                 :           0 :         dat = SPI_getbinval(ruletup, rulettc, fno, &isnull);
    5386         [ #  # ]:           0 :         Assert(!isnull);
    5387                 :           0 :         is_instead = DatumGetBool(dat);
    5388                 :             : 
    5389                 :           0 :         fno = SPI_fnumber(rulettc, "ev_qual");
    5390                 :           0 :         ev_qual = SPI_getvalue(ruletup, rulettc, fno);
    5391         [ #  # ]:           0 :         Assert(ev_qual != NULL);
    5392                 :             : 
    5393                 :           0 :         fno = SPI_fnumber(rulettc, "ev_action");
    5394                 :           0 :         ev_action = SPI_getvalue(ruletup, rulettc, fno);
    5395         [ #  # ]:           0 :         Assert(ev_action != NULL);
    5396                 :           0 :         actions = (List *) stringToNode(ev_action);
    5397         [ #  # ]:           0 :         if (actions == NIL)
    5398   [ #  #  #  # ]:           0 :                 elog(ERROR, "invalid empty ev_action list");
    5399                 :             : 
    5400                 :           0 :         ev_relation = table_open(ev_class, AccessShareLock);
    5401                 :             : 
    5402                 :             :         /*
    5403                 :             :          * Build the rules definition text
    5404                 :             :          */
    5405                 :           0 :         appendStringInfo(buf, "CREATE RULE %s AS",
    5406                 :           0 :                                          quote_identifier(rulename));
    5407                 :             : 
    5408         [ #  # ]:           0 :         if (prettyFlags & PRETTYFLAG_INDENT)
    5409                 :           0 :                 appendStringInfoString(buf, "\n    ON ");
    5410                 :             :         else
    5411                 :           0 :                 appendStringInfoString(buf, " ON ");
    5412                 :             : 
    5413                 :             :         /* The event the rule is fired for */
    5414   [ #  #  #  #  :           0 :         switch (ev_type)
                      # ]
    5415                 :             :         {
    5416                 :             :                 case '1':
    5417                 :           0 :                         appendStringInfoString(buf, "SELECT");
    5418                 :           0 :                         viewResultDesc = RelationGetDescr(ev_relation);
    5419                 :           0 :                         break;
    5420                 :             : 
    5421                 :             :                 case '2':
    5422                 :           0 :                         appendStringInfoString(buf, "UPDATE");
    5423                 :           0 :                         break;
    5424                 :             : 
    5425                 :             :                 case '3':
    5426                 :           0 :                         appendStringInfoString(buf, "INSERT");
    5427                 :           0 :                         break;
    5428                 :             : 
    5429                 :             :                 case '4':
    5430                 :           0 :                         appendStringInfoString(buf, "DELETE");
    5431                 :           0 :                         break;
    5432                 :             : 
    5433                 :             :                 default:
    5434   [ #  #  #  # ]:           0 :                         ereport(ERROR,
    5435                 :             :                                         (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
    5436                 :             :                                          errmsg("rule \"%s\" has unsupported event type %d",
    5437                 :             :                                                         rulename, ev_type)));
    5438                 :           0 :                         break;
    5439                 :             :         }
    5440                 :             : 
    5441                 :             :         /* The relation the rule is fired on */
    5442                 :           0 :         appendStringInfo(buf, " TO %s",
    5443         [ #  # ]:           0 :                                          (prettyFlags & PRETTYFLAG_SCHEMA) ?
    5444                 :           0 :                                          generate_relation_name(ev_class, NIL) :
    5445                 :           0 :                                          generate_qualified_relation_name(ev_class));
    5446                 :             : 
    5447                 :             :         /* If the rule has an event qualification, add it */
    5448         [ #  # ]:           0 :         if (strcmp(ev_qual, "<>") != 0)
    5449                 :             :         {
    5450                 :           0 :                 Node       *qual;
    5451                 :           0 :                 Query      *query;
    5452                 :           0 :                 deparse_context context;
    5453                 :           0 :                 deparse_namespace dpns;
    5454                 :             : 
    5455         [ #  # ]:           0 :                 if (prettyFlags & PRETTYFLAG_INDENT)
    5456                 :           0 :                         appendStringInfoString(buf, "\n  ");
    5457                 :           0 :                 appendStringInfoString(buf, " WHERE ");
    5458                 :             : 
    5459                 :           0 :                 qual = stringToNode(ev_qual);
    5460                 :             : 
    5461                 :             :                 /*
    5462                 :             :                  * We need to make a context for recognizing any Vars in the qual
    5463                 :             :                  * (which can only be references to OLD and NEW).  Use the rtable of
    5464                 :             :                  * the first query in the action list for this purpose.
    5465                 :             :                  */
    5466                 :           0 :                 query = (Query *) linitial(actions);
    5467                 :             : 
    5468                 :             :                 /*
    5469                 :             :                  * If the action is INSERT...SELECT, OLD/NEW have been pushed down
    5470                 :             :                  * into the SELECT, and that's what we need to look at. (Ugly kluge
    5471                 :             :                  * ... try to fix this when we redesign querytrees.)
    5472                 :             :                  */
    5473                 :           0 :                 query = getInsertSelectQuery(query, NULL);
    5474                 :             : 
    5475                 :             :                 /* Must acquire locks right away; see notes in get_query_def() */
    5476                 :           0 :                 AcquireRewriteLocks(query, false, false);
    5477                 :             : 
    5478                 :           0 :                 context.buf = buf;
    5479                 :           0 :                 context.namespaces = list_make1(&dpns);
    5480                 :           0 :                 context.resultDesc = NULL;
    5481                 :           0 :                 context.targetList = NIL;
    5482                 :           0 :                 context.windowClause = NIL;
    5483                 :           0 :                 context.varprefix = (list_length(query->rtable) != 1);
    5484                 :           0 :                 context.prettyFlags = prettyFlags;
    5485                 :           0 :                 context.wrapColumn = WRAP_COLUMN_DEFAULT;
    5486                 :           0 :                 context.indentLevel = PRETTYINDENT_STD;
    5487                 :           0 :                 context.colNamesVisible = true;
    5488                 :           0 :                 context.inGroupBy = false;
    5489                 :           0 :                 context.varInOrderBy = false;
    5490                 :           0 :                 context.appendparents = NULL;
    5491                 :             : 
    5492                 :           0 :                 set_deparse_for_query(&dpns, query, NIL);
    5493                 :             : 
    5494                 :           0 :                 get_rule_expr(qual, &context, false);
    5495                 :           0 :         }
    5496                 :             : 
    5497                 :           0 :         appendStringInfoString(buf, " DO ");
    5498                 :             : 
    5499                 :             :         /* The INSTEAD keyword (if so) */
    5500         [ #  # ]:           0 :         if (is_instead)
    5501                 :           0 :                 appendStringInfoString(buf, "INSTEAD ");
    5502                 :             : 
    5503                 :             :         /* Finally the rules actions */
    5504         [ #  # ]:           0 :         if (list_length(actions) > 1)
    5505                 :             :         {
    5506                 :           0 :                 ListCell   *action;
    5507                 :           0 :                 Query      *query;
    5508                 :             : 
    5509                 :           0 :                 appendStringInfoChar(buf, '(');
    5510   [ #  #  #  #  :           0 :                 foreach(action, actions)
                   #  # ]
    5511                 :             :                 {
    5512                 :           0 :                         query = (Query *) lfirst(action);
    5513                 :           0 :                         get_query_def(query, buf, NIL, viewResultDesc, true,
    5514                 :           0 :                                                   prettyFlags, WRAP_COLUMN_DEFAULT, 0);
    5515         [ #  # ]:           0 :                         if (prettyFlags)
    5516                 :           0 :                                 appendStringInfoString(buf, ";\n");
    5517                 :             :                         else
    5518                 :           0 :                                 appendStringInfoString(buf, "; ");
    5519                 :           0 :                 }
    5520                 :           0 :                 appendStringInfoString(buf, ");");
    5521                 :           0 :         }
    5522                 :             :         else
    5523                 :             :         {
    5524                 :           0 :                 Query      *query;
    5525                 :             : 
    5526                 :           0 :                 query = (Query *) linitial(actions);
    5527                 :           0 :                 get_query_def(query, buf, NIL, viewResultDesc, true,
    5528                 :           0 :                                           prettyFlags, WRAP_COLUMN_DEFAULT, 0);
    5529                 :           0 :                 appendStringInfoChar(buf, ';');
    5530                 :           0 :         }
    5531                 :             : 
    5532                 :           0 :         table_close(ev_relation, AccessShareLock);
    5533                 :           0 : }
    5534                 :             : 
    5535                 :             : 
    5536                 :             : /* ----------
    5537                 :             :  * make_viewdef                 - reconstruct the SELECT part of a
    5538                 :             :  *                                view rewrite rule
    5539                 :             :  * ----------
    5540                 :             :  */
    5541                 :             : static void
    5542                 :           0 : make_viewdef(StringInfo buf, HeapTuple ruletup, TupleDesc rulettc,
    5543                 :             :                          int prettyFlags, int wrapColumn)
    5544                 :             : {
    5545                 :           0 :         Query      *query;
    5546                 :           0 :         char            ev_type;
    5547                 :           0 :         Oid                     ev_class;
    5548                 :           0 :         bool            is_instead;
    5549                 :           0 :         char       *ev_qual;
    5550                 :           0 :         char       *ev_action;
    5551                 :           0 :         List       *actions;
    5552                 :           0 :         Relation        ev_relation;
    5553                 :           0 :         int                     fno;
    5554                 :           0 :         Datum           dat;
    5555                 :           0 :         bool            isnull;
    5556                 :             : 
    5557                 :             :         /*
    5558                 :             :          * Get the attribute values from the rules tuple
    5559                 :             :          */
    5560                 :           0 :         fno = SPI_fnumber(rulettc, "ev_type");
    5561                 :           0 :         dat = SPI_getbinval(ruletup, rulettc, fno, &isnull);
    5562         [ #  # ]:           0 :         Assert(!isnull);
    5563                 :           0 :         ev_type = DatumGetChar(dat);
    5564                 :             : 
    5565                 :           0 :         fno = SPI_fnumber(rulettc, "ev_class");
    5566                 :           0 :         dat = SPI_getbinval(ruletup, rulettc, fno, &isnull);
    5567         [ #  # ]:           0 :         Assert(!isnull);
    5568                 :           0 :         ev_class = DatumGetObjectId(dat);
    5569                 :             : 
    5570                 :           0 :         fno = SPI_fnumber(rulettc, "is_instead");
    5571                 :           0 :         dat = SPI_getbinval(ruletup, rulettc, fno, &isnull);
    5572         [ #  # ]:           0 :         Assert(!isnull);
    5573                 :           0 :         is_instead = DatumGetBool(dat);
    5574                 :             : 
    5575                 :           0 :         fno = SPI_fnumber(rulettc, "ev_qual");
    5576                 :           0 :         ev_qual = SPI_getvalue(ruletup, rulettc, fno);
    5577         [ #  # ]:           0 :         Assert(ev_qual != NULL);
    5578                 :             : 
    5579                 :           0 :         fno = SPI_fnumber(rulettc, "ev_action");
    5580                 :           0 :         ev_action = SPI_getvalue(ruletup, rulettc, fno);
    5581         [ #  # ]:           0 :         Assert(ev_action != NULL);
    5582                 :           0 :         actions = (List *) stringToNode(ev_action);
    5583                 :             : 
    5584         [ #  # ]:           0 :         if (list_length(actions) != 1)
    5585                 :             :         {
    5586                 :             :                 /* keep output buffer empty and leave */
    5587                 :           0 :                 return;
    5588                 :             :         }
    5589                 :             : 
    5590                 :           0 :         query = (Query *) linitial(actions);
    5591                 :             : 
    5592   [ #  #  #  # ]:           0 :         if (ev_type != '1' || !is_instead ||
    5593   [ #  #  #  # ]:           0 :                 strcmp(ev_qual, "<>") != 0 || query->commandType != CMD_SELECT)
    5594                 :             :         {
    5595                 :             :                 /* keep output buffer empty and leave */
    5596                 :           0 :                 return;
    5597                 :             :         }
    5598                 :             : 
    5599                 :           0 :         ev_relation = table_open(ev_class, AccessShareLock);
    5600                 :             : 
    5601                 :           0 :         get_query_def(query, buf, NIL, RelationGetDescr(ev_relation), true,
    5602                 :           0 :                                   prettyFlags, wrapColumn, 0);
    5603                 :           0 :         appendStringInfoChar(buf, ';');
    5604                 :             : 
    5605                 :           0 :         table_close(ev_relation, AccessShareLock);
    5606         [ #  # ]:           0 : }
    5607                 :             : 
    5608                 :             : 
    5609                 :             : /* ----------
    5610                 :             :  * get_query_def                        - Parse back one query parsetree
    5611                 :             :  *
    5612                 :             :  * query: parsetree to be displayed
    5613                 :             :  * buf: output text is appended to buf
    5614                 :             :  * parentnamespace: list (initially empty) of outer-level deparse_namespace's
    5615                 :             :  * resultDesc: if not NULL, the output tuple descriptor for the view
    5616                 :             :  *              represented by a SELECT query.  We use the column names from it
    5617                 :             :  *              to label SELECT output columns, in preference to names in the query
    5618                 :             :  * colNamesVisible: true if the surrounding context cares about the output
    5619                 :             :  *              column names at all (as, for example, an EXISTS() context does not);
    5620                 :             :  *              when false, we can suppress dummy column labels such as "?column?"
    5621                 :             :  * prettyFlags: bitmask of PRETTYFLAG_XXX options
    5622                 :             :  * wrapColumn: maximum line length, or -1 to disable wrapping
    5623                 :             :  * startIndent: initial indentation amount
    5624                 :             :  * ----------
    5625                 :             :  */
    5626                 :             : static void
    5627                 :           0 : get_query_def(Query *query, StringInfo buf, List *parentnamespace,
    5628                 :             :                           TupleDesc resultDesc, bool colNamesVisible,
    5629                 :             :                           int prettyFlags, int wrapColumn, int startIndent)
    5630                 :             : {
    5631                 :           0 :         deparse_context context;
    5632                 :           0 :         deparse_namespace dpns;
    5633                 :           0 :         int                     rtable_size;
    5634                 :             : 
    5635                 :             :         /* Guard against excessively long or deeply-nested queries */
    5636         [ #  # ]:           0 :         CHECK_FOR_INTERRUPTS();
    5637                 :           0 :         check_stack_depth();
    5638                 :             : 
    5639         [ #  # ]:           0 :         rtable_size = query->hasGroupRTE ?
    5640                 :           0 :                 list_length(query->rtable) - 1 :
    5641                 :           0 :                 list_length(query->rtable);
    5642                 :             : 
    5643                 :             :         /*
    5644                 :             :          * Replace any Vars in the query's targetlist and havingQual that
    5645                 :             :          * reference GROUP outputs with the underlying grouping expressions.
    5646                 :             :          */
    5647         [ #  # ]:           0 :         if (query->hasGroupRTE)
    5648                 :             :         {
    5649                 :           0 :                 query->targetList = (List *)
    5650                 :           0 :                         flatten_group_exprs(NULL, query, (Node *) query->targetList);
    5651                 :           0 :                 query->havingQual =
    5652                 :           0 :                         flatten_group_exprs(NULL, query, query->havingQual);
    5653                 :           0 :         }
    5654                 :             : 
    5655                 :             :         /*
    5656                 :             :          * Before we begin to examine the query, acquire locks on referenced
    5657                 :             :          * relations, and fix up deleted columns in JOIN RTEs.  This ensures
    5658                 :             :          * consistent results.  Note we assume it's OK to scribble on the passed
    5659                 :             :          * querytree!
    5660                 :             :          *
    5661                 :             :          * We are only deparsing the query (we are not about to execute it), so we
    5662                 :             :          * only need AccessShareLock on the relations it mentions.
    5663                 :             :          */
    5664                 :           0 :         AcquireRewriteLocks(query, false, false);
    5665                 :             : 
    5666                 :           0 :         context.buf = buf;
    5667                 :           0 :         context.namespaces = lcons(&dpns, list_copy(parentnamespace));
    5668                 :           0 :         context.resultDesc = NULL;
    5669                 :           0 :         context.targetList = NIL;
    5670                 :           0 :         context.windowClause = NIL;
    5671         [ #  # ]:           0 :         context.varprefix = (parentnamespace != NIL ||
    5672                 :           0 :                                                  rtable_size != 1);
    5673                 :           0 :         context.prettyFlags = prettyFlags;
    5674                 :           0 :         context.wrapColumn = wrapColumn;
    5675                 :           0 :         context.indentLevel = startIndent;
    5676                 :           0 :         context.colNamesVisible = colNamesVisible;
    5677                 :           0 :         context.inGroupBy = false;
    5678                 :           0 :         context.varInOrderBy = false;
    5679                 :           0 :         context.appendparents = NULL;
    5680                 :             : 
    5681                 :           0 :         set_deparse_for_query(&dpns, query, parentnamespace);
    5682                 :             : 
    5683   [ #  #  #  #  :           0 :         switch (query->commandType)
             #  #  #  # ]
    5684                 :             :         {
    5685                 :             :                 case CMD_SELECT:
    5686                 :             :                         /* We set context.resultDesc only if it's a SELECT */
    5687                 :           0 :                         context.resultDesc = resultDesc;
    5688                 :           0 :                         get_select_query_def(query, &context);
    5689                 :           0 :                         break;
    5690                 :             : 
    5691                 :             :                 case CMD_UPDATE:
    5692                 :           0 :                         get_update_query_def(query, &context);
    5693                 :           0 :                         break;
    5694                 :             : 
    5695                 :             :                 case CMD_INSERT:
    5696                 :           0 :                         get_insert_query_def(query, &context);
    5697                 :           0 :                         break;
    5698                 :             : 
    5699                 :             :                 case CMD_DELETE:
    5700                 :           0 :                         get_delete_query_def(query, &context);
    5701                 :           0 :                         break;
    5702                 :             : 
    5703                 :             :                 case CMD_MERGE:
    5704                 :           0 :                         get_merge_query_def(query, &context);
    5705                 :           0 :                         break;
    5706                 :             : 
    5707                 :             :                 case CMD_NOTHING:
    5708                 :           0 :                         appendStringInfoString(buf, "NOTHING");
    5709                 :           0 :                         break;
    5710                 :             : 
    5711                 :             :                 case CMD_UTILITY:
    5712                 :           0 :                         get_utility_query_def(query, &context);
    5713                 :           0 :                         break;
    5714                 :             : 
    5715                 :             :                 default:
    5716   [ #  #  #  # ]:           0 :                         elog(ERROR, "unrecognized query command type: %d",
    5717                 :             :                                  query->commandType);
    5718                 :           0 :                         break;
    5719                 :             :         }
    5720                 :           0 : }
    5721                 :             : 
    5722                 :             : /* ----------
    5723                 :             :  * get_values_def                       - Parse back a VALUES list
    5724                 :             :  * ----------
    5725                 :             :  */
    5726                 :             : static void
    5727                 :           0 : get_values_def(List *values_lists, deparse_context *context)
    5728                 :             : {
    5729                 :           0 :         StringInfo      buf = context->buf;
    5730                 :           0 :         bool            first_list = true;
    5731                 :           0 :         ListCell   *vtl;
    5732                 :             : 
    5733                 :           0 :         appendStringInfoString(buf, "VALUES ");
    5734                 :             : 
    5735   [ #  #  #  #  :           0 :         foreach(vtl, values_lists)
                   #  # ]
    5736                 :             :         {
    5737                 :           0 :                 List       *sublist = (List *) lfirst(vtl);
    5738                 :           0 :                 bool            first_col = true;
    5739                 :           0 :                 ListCell   *lc;
    5740                 :             : 
    5741         [ #  # ]:           0 :                 if (first_list)
    5742                 :           0 :                         first_list = false;
    5743                 :             :                 else
    5744                 :           0 :                         appendStringInfoString(buf, ", ");
    5745                 :             : 
    5746                 :           0 :                 appendStringInfoChar(buf, '(');
    5747   [ #  #  #  #  :           0 :                 foreach(lc, sublist)
                   #  # ]
    5748                 :             :                 {
    5749                 :           0 :                         Node       *col = (Node *) lfirst(lc);
    5750                 :             : 
    5751         [ #  # ]:           0 :                         if (first_col)
    5752                 :           0 :                                 first_col = false;
    5753                 :             :                         else
    5754                 :           0 :                                 appendStringInfoChar(buf, ',');
    5755                 :             : 
    5756                 :             :                         /*
    5757                 :             :                          * Print the value.  Whole-row Vars need special treatment.
    5758                 :             :                          */
    5759                 :           0 :                         get_rule_expr_toplevel(col, context, false);
    5760                 :           0 :                 }
    5761                 :           0 :                 appendStringInfoChar(buf, ')');
    5762                 :           0 :         }
    5763                 :           0 : }
    5764                 :             : 
    5765                 :             : /* ----------
    5766                 :             :  * get_with_clause                      - Parse back a WITH clause
    5767                 :             :  * ----------
    5768                 :             :  */
    5769                 :             : static void
    5770                 :           0 : get_with_clause(Query *query, deparse_context *context)
    5771                 :             : {
    5772                 :           0 :         StringInfo      buf = context->buf;
    5773                 :           0 :         const char *sep;
    5774                 :           0 :         ListCell   *l;
    5775                 :             : 
    5776         [ #  # ]:           0 :         if (query->cteList == NIL)
    5777                 :           0 :                 return;
    5778                 :             : 
    5779         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
    5780                 :             :         {
    5781                 :           0 :                 context->indentLevel += PRETTYINDENT_STD;
    5782                 :           0 :                 appendStringInfoChar(buf, ' ');
    5783                 :           0 :         }
    5784                 :             : 
    5785         [ #  # ]:           0 :         if (query->hasRecursive)
    5786                 :           0 :                 sep = "WITH RECURSIVE ";
    5787                 :             :         else
    5788                 :           0 :                 sep = "WITH ";
    5789   [ #  #  #  #  :           0 :         foreach(l, query->cteList)
                   #  # ]
    5790                 :             :         {
    5791                 :           0 :                 CommonTableExpr *cte = (CommonTableExpr *) lfirst(l);
    5792                 :             : 
    5793                 :           0 :                 appendStringInfoString(buf, sep);
    5794                 :           0 :                 appendStringInfoString(buf, quote_identifier(cte->ctename));
    5795         [ #  # ]:           0 :                 if (cte->aliascolnames)
    5796                 :             :                 {
    5797                 :           0 :                         bool            first = true;
    5798                 :           0 :                         ListCell   *col;
    5799                 :             : 
    5800                 :           0 :                         appendStringInfoChar(buf, '(');
    5801   [ #  #  #  #  :           0 :                         foreach(col, cte->aliascolnames)
                   #  # ]
    5802                 :             :                         {
    5803         [ #  # ]:           0 :                                 if (first)
    5804                 :           0 :                                         first = false;
    5805                 :             :                                 else
    5806                 :           0 :                                         appendStringInfoString(buf, ", ");
    5807                 :           0 :                                 appendStringInfoString(buf,
    5808                 :           0 :                                                                            quote_identifier(strVal(lfirst(col))));
    5809                 :           0 :                         }
    5810                 :           0 :                         appendStringInfoChar(buf, ')');
    5811                 :           0 :                 }
    5812                 :           0 :                 appendStringInfoString(buf, " AS ");
    5813      [ #  #  # ]:           0 :                 switch (cte->ctematerialized)
    5814                 :             :                 {
    5815                 :             :                         case CTEMaterializeDefault:
    5816                 :             :                                 break;
    5817                 :             :                         case CTEMaterializeAlways:
    5818                 :           0 :                                 appendStringInfoString(buf, "MATERIALIZED ");
    5819                 :           0 :                                 break;
    5820                 :             :                         case CTEMaterializeNever:
    5821                 :           0 :                                 appendStringInfoString(buf, "NOT MATERIALIZED ");
    5822                 :           0 :                                 break;
    5823                 :             :                 }
    5824                 :           0 :                 appendStringInfoChar(buf, '(');
    5825         [ #  # ]:           0 :                 if (PRETTY_INDENT(context))
    5826                 :           0 :                         appendContextKeyword(context, "", 0, 0, 0);
    5827                 :           0 :                 get_query_def((Query *) cte->ctequery, buf, context->namespaces, NULL,
    5828                 :             :                                           true,
    5829                 :           0 :                                           context->prettyFlags, context->wrapColumn,
    5830                 :           0 :                                           context->indentLevel);
    5831         [ #  # ]:           0 :                 if (PRETTY_INDENT(context))
    5832                 :           0 :                         appendContextKeyword(context, "", 0, 0, 0);
    5833                 :           0 :                 appendStringInfoChar(buf, ')');
    5834                 :             : 
    5835         [ #  # ]:           0 :                 if (cte->search_clause)
    5836                 :             :                 {
    5837                 :           0 :                         bool            first = true;
    5838                 :           0 :                         ListCell   *lc;
    5839                 :             : 
    5840                 :           0 :                         appendStringInfo(buf, " SEARCH %s FIRST BY ",
    5841                 :           0 :                                                          cte->search_clause->search_breadth_first ? "BREADTH" : "DEPTH");
    5842                 :             : 
    5843   [ #  #  #  #  :           0 :                         foreach(lc, cte->search_clause->search_col_list)
                   #  # ]
    5844                 :             :                         {
    5845         [ #  # ]:           0 :                                 if (first)
    5846                 :           0 :                                         first = false;
    5847                 :             :                                 else
    5848                 :           0 :                                         appendStringInfoString(buf, ", ");
    5849                 :           0 :                                 appendStringInfoString(buf,
    5850                 :           0 :                                                                            quote_identifier(strVal(lfirst(lc))));
    5851                 :           0 :                         }
    5852                 :             : 
    5853                 :           0 :                         appendStringInfo(buf, " SET %s", quote_identifier(cte->search_clause->search_seq_column));
    5854                 :           0 :                 }
    5855                 :             : 
    5856         [ #  # ]:           0 :                 if (cte->cycle_clause)
    5857                 :             :                 {
    5858                 :           0 :                         bool            first = true;
    5859                 :           0 :                         ListCell   *lc;
    5860                 :             : 
    5861                 :           0 :                         appendStringInfoString(buf, " CYCLE ");
    5862                 :             : 
    5863   [ #  #  #  #  :           0 :                         foreach(lc, cte->cycle_clause->cycle_col_list)
                   #  # ]
    5864                 :             :                         {
    5865         [ #  # ]:           0 :                                 if (first)
    5866                 :           0 :                                         first = false;
    5867                 :             :                                 else
    5868                 :           0 :                                         appendStringInfoString(buf, ", ");
    5869                 :           0 :                                 appendStringInfoString(buf,
    5870                 :           0 :                                                                            quote_identifier(strVal(lfirst(lc))));
    5871                 :           0 :                         }
    5872                 :             : 
    5873                 :           0 :                         appendStringInfo(buf, " SET %s", quote_identifier(cte->cycle_clause->cycle_mark_column));
    5874                 :             : 
    5875                 :             :                         {
    5876                 :           0 :                                 Const      *cmv = castNode(Const, cte->cycle_clause->cycle_mark_value);
    5877                 :           0 :                                 Const      *cmd = castNode(Const, cte->cycle_clause->cycle_mark_default);
    5878                 :             : 
    5879   [ #  #  #  #  :           0 :                                 if (!(cmv->consttype == BOOLOID && !cmv->constisnull && DatumGetBool(cmv->constvalue) == true &&
             #  #  #  # ]
    5880   [ #  #  #  # ]:           0 :                                           cmd->consttype == BOOLOID && !cmd->constisnull && DatumGetBool(cmd->constvalue) == false))
    5881                 :             :                                 {
    5882                 :           0 :                                         appendStringInfoString(buf, " TO ");
    5883                 :           0 :                                         get_rule_expr(cte->cycle_clause->cycle_mark_value, context, false);
    5884                 :           0 :                                         appendStringInfoString(buf, " DEFAULT ");
    5885                 :           0 :                                         get_rule_expr(cte->cycle_clause->cycle_mark_default, context, false);
    5886                 :           0 :                                 }
    5887                 :           0 :                         }
    5888                 :             : 
    5889                 :           0 :                         appendStringInfo(buf, " USING %s", quote_identifier(cte->cycle_clause->cycle_path_column));
    5890                 :           0 :                 }
    5891                 :             : 
    5892                 :           0 :                 sep = ", ";
    5893                 :           0 :         }
    5894                 :             : 
    5895         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
    5896                 :             :         {
    5897                 :           0 :                 context->indentLevel -= PRETTYINDENT_STD;
    5898                 :           0 :                 appendContextKeyword(context, "", 0, 0, 0);
    5899                 :           0 :         }
    5900                 :             :         else
    5901                 :           0 :                 appendStringInfoChar(buf, ' ');
    5902         [ #  # ]:           0 : }
    5903                 :             : 
    5904                 :             : /* ----------
    5905                 :             :  * get_select_query_def                 - Parse back a SELECT parsetree
    5906                 :             :  * ----------
    5907                 :             :  */
    5908                 :             : static void
    5909                 :           0 : get_select_query_def(Query *query, deparse_context *context)
    5910                 :             : {
    5911                 :           0 :         StringInfo      buf = context->buf;
    5912                 :           0 :         bool            force_colno;
    5913                 :           0 :         ListCell   *l;
    5914                 :             : 
    5915                 :             :         /* Insert the WITH clause if given */
    5916                 :           0 :         get_with_clause(query, context);
    5917                 :             : 
    5918                 :             :         /* Subroutines may need to consult the SELECT targetlist and windowClause */
    5919                 :           0 :         context->targetList = query->targetList;
    5920                 :           0 :         context->windowClause = query->windowClause;
    5921                 :             : 
    5922                 :             :         /*
    5923                 :             :          * If the Query node has a setOperations tree, then it's the top level of
    5924                 :             :          * a UNION/INTERSECT/EXCEPT query; only the WITH, ORDER BY and LIMIT
    5925                 :             :          * fields are interesting in the top query itself.
    5926                 :             :          */
    5927         [ #  # ]:           0 :         if (query->setOperations)
    5928                 :             :         {
    5929                 :           0 :                 get_setop_query(query->setOperations, query, context);
    5930                 :             :                 /* ORDER BY clauses must be simple in this case */
    5931                 :           0 :                 force_colno = true;
    5932                 :           0 :         }
    5933                 :             :         else
    5934                 :             :         {
    5935                 :           0 :                 get_basic_select_query(query, context);
    5936                 :           0 :                 force_colno = false;
    5937                 :             :         }
    5938                 :             : 
    5939                 :             :         /* Add the ORDER BY clause if given */
    5940         [ #  # ]:           0 :         if (query->sortClause != NIL)
    5941                 :             :         {
    5942                 :           0 :                 appendContextKeyword(context, " ORDER BY ",
    5943                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 1);
    5944                 :           0 :                 get_rule_orderby(query->sortClause, query->targetList,
    5945                 :           0 :                                                  force_colno, context);
    5946                 :           0 :         }
    5947                 :             : 
    5948                 :             :         /*
    5949                 :             :          * Add the LIMIT/OFFSET clauses if given. If non-default options, use the
    5950                 :             :          * standard spelling of LIMIT.
    5951                 :             :          */
    5952         [ #  # ]:           0 :         if (query->limitOffset != NULL)
    5953                 :             :         {
    5954                 :           0 :                 appendContextKeyword(context, " OFFSET ",
    5955                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 0);
    5956                 :           0 :                 get_rule_expr(query->limitOffset, context, false);
    5957                 :           0 :         }
    5958         [ #  # ]:           0 :         if (query->limitCount != NULL)
    5959                 :             :         {
    5960         [ #  # ]:           0 :                 if (query->limitOption == LIMIT_OPTION_WITH_TIES)
    5961                 :             :                 {
    5962                 :             :                         /*
    5963                 :             :                          * The limitCount arg is a c_expr, so it needs parens. Simple
    5964                 :             :                          * literals and function expressions would not need parens, but
    5965                 :             :                          * unfortunately it's hard to tell if the expression will be
    5966                 :             :                          * printed as a simple literal like 123 or as a typecast
    5967                 :             :                          * expression, like '-123'::int4. The grammar accepts the former
    5968                 :             :                          * without quoting, but not the latter.
    5969                 :             :                          */
    5970                 :           0 :                         appendContextKeyword(context, " FETCH FIRST ",
    5971                 :             :                                                                  -PRETTYINDENT_STD, PRETTYINDENT_STD, 0);
    5972                 :           0 :                         appendStringInfoChar(buf, '(');
    5973                 :           0 :                         get_rule_expr(query->limitCount, context, false);
    5974                 :           0 :                         appendStringInfoChar(buf, ')');
    5975                 :           0 :                         appendStringInfoString(buf, " ROWS WITH TIES");
    5976                 :           0 :                 }
    5977                 :             :                 else
    5978                 :             :                 {
    5979                 :           0 :                         appendContextKeyword(context, " LIMIT ",
    5980                 :             :                                                                  -PRETTYINDENT_STD, PRETTYINDENT_STD, 0);
    5981   [ #  #  #  # ]:           0 :                         if (IsA(query->limitCount, Const) &&
    5982                 :           0 :                                 ((Const *) query->limitCount)->constisnull)
    5983                 :           0 :                                 appendStringInfoString(buf, "ALL");
    5984                 :             :                         else
    5985                 :           0 :                                 get_rule_expr(query->limitCount, context, false);
    5986                 :             :                 }
    5987                 :           0 :         }
    5988                 :             : 
    5989                 :             :         /* Add FOR [KEY] UPDATE/SHARE clauses if present */
    5990         [ #  # ]:           0 :         if (query->hasForUpdate)
    5991                 :             :         {
    5992   [ #  #  #  #  :           0 :                 foreach(l, query->rowMarks)
                   #  # ]
    5993                 :             :                 {
    5994                 :           0 :                         RowMarkClause *rc = (RowMarkClause *) lfirst(l);
    5995                 :             : 
    5996                 :             :                         /* don't print implicit clauses */
    5997         [ #  # ]:           0 :                         if (rc->pushedDown)
    5998                 :           0 :                                 continue;
    5999                 :             : 
    6000   [ #  #  #  #  :           0 :                         switch (rc->strength)
                   #  # ]
    6001                 :             :                         {
    6002                 :             :                                 case LCS_NONE:
    6003                 :             :                                         /* we intentionally throw an error for LCS_NONE */
    6004   [ #  #  #  # ]:           0 :                                         elog(ERROR, "unrecognized LockClauseStrength %d",
    6005                 :             :                                                  (int) rc->strength);
    6006                 :           0 :                                         break;
    6007                 :             :                                 case LCS_FORKEYSHARE:
    6008                 :           0 :                                         appendContextKeyword(context, " FOR KEY SHARE",
    6009                 :             :                                                                                  -PRETTYINDENT_STD, PRETTYINDENT_STD, 0);
    6010                 :           0 :                                         break;
    6011                 :             :                                 case LCS_FORSHARE:
    6012                 :           0 :                                         appendContextKeyword(context, " FOR SHARE",
    6013                 :             :                                                                                  -PRETTYINDENT_STD, PRETTYINDENT_STD, 0);
    6014                 :           0 :                                         break;
    6015                 :             :                                 case LCS_FORNOKEYUPDATE:
    6016                 :           0 :                                         appendContextKeyword(context, " FOR NO KEY UPDATE",
    6017                 :             :                                                                                  -PRETTYINDENT_STD, PRETTYINDENT_STD, 0);
    6018                 :           0 :                                         break;
    6019                 :             :                                 case LCS_FORUPDATE:
    6020                 :           0 :                                         appendContextKeyword(context, " FOR UPDATE",
    6021                 :             :                                                                                  -PRETTYINDENT_STD, PRETTYINDENT_STD, 0);
    6022                 :           0 :                                         break;
    6023                 :             :                         }
    6024                 :             : 
    6025                 :           0 :                         appendStringInfo(buf, " OF %s",
    6026                 :           0 :                                                          quote_identifier(get_rtable_name(rc->rti,
    6027                 :           0 :                                                                                                                           context)));
    6028         [ #  # ]:           0 :                         if (rc->waitPolicy == LockWaitError)
    6029                 :           0 :                                 appendStringInfoString(buf, " NOWAIT");
    6030         [ #  # ]:           0 :                         else if (rc->waitPolicy == LockWaitSkip)
    6031                 :           0 :                                 appendStringInfoString(buf, " SKIP LOCKED");
    6032      [ #  #  # ]:           0 :                 }
    6033                 :           0 :         }
    6034                 :           0 : }
    6035                 :             : 
    6036                 :             : /*
    6037                 :             :  * Detect whether query looks like SELECT ... FROM VALUES(),
    6038                 :             :  * with no need to rename the output columns of the VALUES RTE.
    6039                 :             :  * If so, return the VALUES RTE.  Otherwise return NULL.
    6040                 :             :  */
    6041                 :             : static RangeTblEntry *
    6042                 :           0 : get_simple_values_rte(Query *query, TupleDesc resultDesc)
    6043                 :             : {
    6044                 :           0 :         RangeTblEntry *result = NULL;
    6045                 :           0 :         ListCell   *lc;
    6046                 :             : 
    6047                 :             :         /*
    6048                 :             :          * We want to detect a match even if the Query also contains OLD or NEW
    6049                 :             :          * rule RTEs.  So the idea is to scan the rtable and see if there is only
    6050                 :             :          * one inFromCl RTE that is a VALUES RTE.
    6051                 :             :          */
    6052   [ #  #  #  #  :           0 :         foreach(lc, query->rtable)
             #  #  #  # ]
    6053                 :             :         {
    6054                 :           0 :                 RangeTblEntry *rte = (RangeTblEntry *) lfirst(lc);
    6055                 :             : 
    6056   [ #  #  #  # ]:           0 :                 if (rte->rtekind == RTE_VALUES && rte->inFromCl)
    6057                 :             :                 {
    6058         [ #  # ]:           0 :                         if (result)
    6059                 :           0 :                                 return NULL;    /* multiple VALUES (probably not possible) */
    6060                 :           0 :                         result = rte;
    6061                 :           0 :                 }
    6062   [ #  #  #  # ]:           0 :                 else if (rte->rtekind == RTE_RELATION && !rte->inFromCl)
    6063                 :           0 :                         continue;                       /* ignore rule entries */
    6064                 :             :                 else
    6065                 :           0 :                         return NULL;            /* something else -> not simple VALUES */
    6066      [ #  #  # ]:           0 :         }
    6067                 :             : 
    6068                 :             :         /*
    6069                 :             :          * We don't need to check the targetlist in any great detail, because
    6070                 :             :          * parser/analyze.c will never generate a "bare" VALUES RTE --- they only
    6071                 :             :          * appear inside auto-generated sub-queries with very restricted
    6072                 :             :          * structure.  However, DefineView might have modified the tlist by
    6073                 :             :          * injecting new column aliases, or we might have some other column
    6074                 :             :          * aliases forced by a resultDesc.  We can only simplify if the RTE's
    6075                 :             :          * column names match the names that get_target_list() would select.
    6076                 :             :          */
    6077         [ #  # ]:           0 :         if (result)
    6078                 :             :         {
    6079                 :           0 :                 ListCell   *lcn;
    6080                 :           0 :                 int                     colno;
    6081                 :             : 
    6082         [ #  # ]:           0 :                 if (list_length(query->targetList) != list_length(result->eref->colnames))
    6083                 :           0 :                         return NULL;            /* this probably cannot happen */
    6084                 :           0 :                 colno = 0;
    6085   [ #  #  #  #  :           0 :                 forboth(lc, query->targetList, lcn, result->eref->colnames)
          #  #  #  #  #  
             #  #  #  #  
                      # ]
    6086                 :             :                 {
    6087                 :           0 :                         TargetEntry *tle = (TargetEntry *) lfirst(lc);
    6088                 :           0 :                         char       *cname = strVal(lfirst(lcn));
    6089                 :           0 :                         char       *colname;
    6090                 :             : 
    6091         [ #  # ]:           0 :                         if (tle->resjunk)
    6092                 :           0 :                                 return NULL;    /* this probably cannot happen */
    6093                 :             : 
    6094                 :             :                         /* compute name that get_target_list would use for column */
    6095                 :           0 :                         colno++;
    6096   [ #  #  #  # ]:           0 :                         if (resultDesc && colno <= resultDesc->natts)
    6097                 :           0 :                                 colname = NameStr(TupleDescAttr(resultDesc, colno - 1)->attname);
    6098                 :             :                         else
    6099                 :           0 :                                 colname = tle->resname;
    6100                 :             : 
    6101                 :             :                         /* does it match the VALUES RTE? */
    6102   [ #  #  #  # ]:           0 :                         if (colname == NULL || strcmp(colname, cname) != 0)
    6103                 :           0 :                                 return NULL;    /* column name has been changed */
    6104         [ #  # ]:           0 :                 }
    6105         [ #  # ]:           0 :         }
    6106                 :             : 
    6107                 :           0 :         return result;
    6108                 :           0 : }
    6109                 :             : 
    6110                 :             : static void
    6111                 :           0 : get_basic_select_query(Query *query, deparse_context *context)
    6112                 :             : {
    6113                 :           0 :         StringInfo      buf = context->buf;
    6114                 :           0 :         RangeTblEntry *values_rte;
    6115                 :           0 :         char       *sep;
    6116                 :           0 :         ListCell   *l;
    6117                 :             : 
    6118         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
    6119                 :             :         {
    6120                 :           0 :                 context->indentLevel += PRETTYINDENT_STD;
    6121                 :           0 :                 appendStringInfoChar(buf, ' ');
    6122                 :           0 :         }
    6123                 :             : 
    6124                 :             :         /*
    6125                 :             :          * If the query looks like SELECT * FROM (VALUES ...), then print just the
    6126                 :             :          * VALUES part.  This reverses what transformValuesClause() did at parse
    6127                 :             :          * time.
    6128                 :             :          */
    6129                 :           0 :         values_rte = get_simple_values_rte(query, context->resultDesc);
    6130         [ #  # ]:           0 :         if (values_rte)
    6131                 :             :         {
    6132                 :           0 :                 get_values_def(values_rte->values_lists, context);
    6133                 :           0 :                 return;
    6134                 :             :         }
    6135                 :             : 
    6136                 :             :         /*
    6137                 :             :          * Build up the query string - first we say SELECT
    6138                 :             :          */
    6139         [ #  # ]:           0 :         if (query->isReturn)
    6140                 :           0 :                 appendStringInfoString(buf, "RETURN");
    6141                 :             :         else
    6142                 :           0 :                 appendStringInfoString(buf, "SELECT");
    6143                 :             : 
    6144                 :             :         /* Add the DISTINCT clause if given */
    6145         [ #  # ]:           0 :         if (query->distinctClause != NIL)
    6146                 :             :         {
    6147         [ #  # ]:           0 :                 if (query->hasDistinctOn)
    6148                 :             :                 {
    6149                 :           0 :                         appendStringInfoString(buf, " DISTINCT ON (");
    6150                 :           0 :                         sep = "";
    6151   [ #  #  #  #  :           0 :                         foreach(l, query->distinctClause)
                   #  # ]
    6152                 :             :                         {
    6153                 :           0 :                                 SortGroupClause *srt = (SortGroupClause *) lfirst(l);
    6154                 :             : 
    6155                 :           0 :                                 appendStringInfoString(buf, sep);
    6156                 :           0 :                                 get_rule_sortgroupclause(srt->tleSortGroupRef, query->targetList,
    6157                 :           0 :                                                                                  false, context);
    6158                 :           0 :                                 sep = ", ";
    6159                 :           0 :                         }
    6160                 :           0 :                         appendStringInfoChar(buf, ')');
    6161                 :           0 :                 }
    6162                 :             :                 else
    6163                 :           0 :                         appendStringInfoString(buf, " DISTINCT");
    6164                 :           0 :         }
    6165                 :             : 
    6166                 :             :         /* Then we tell what to select (the targetlist) */
    6167                 :           0 :         get_target_list(query->targetList, context);
    6168                 :             : 
    6169                 :             :         /* Add the FROM clause if needed */
    6170                 :           0 :         get_from_clause(query, " FROM ", context);
    6171                 :             : 
    6172                 :             :         /* Add the WHERE clause if given */
    6173         [ #  # ]:           0 :         if (query->jointree->quals != NULL)
    6174                 :             :         {
    6175                 :           0 :                 appendContextKeyword(context, " WHERE ",
    6176                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 1);
    6177                 :           0 :                 get_rule_expr(query->jointree->quals, context, false);
    6178                 :           0 :         }
    6179                 :             : 
    6180                 :             :         /* Add the GROUP BY clause if given */
    6181   [ #  #  #  # ]:           0 :         if (query->groupClause != NULL || query->groupingSets != NULL)
    6182                 :             :         {
    6183                 :           0 :                 bool            save_ingroupby;
    6184                 :             : 
    6185                 :           0 :                 appendContextKeyword(context, " GROUP BY ",
    6186                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 1);
    6187         [ #  # ]:           0 :                 if (query->groupDistinct)
    6188                 :           0 :                         appendStringInfoString(buf, "DISTINCT ");
    6189                 :             : 
    6190                 :           0 :                 save_ingroupby = context->inGroupBy;
    6191                 :           0 :                 context->inGroupBy = true;
    6192                 :             : 
    6193         [ #  # ]:           0 :                 if (query->groupByAll)
    6194                 :           0 :                         appendStringInfoString(buf, "ALL");
    6195         [ #  # ]:           0 :                 else if (query->groupingSets == NIL)
    6196                 :             :                 {
    6197                 :           0 :                         sep = "";
    6198   [ #  #  #  #  :           0 :                         foreach(l, query->groupClause)
                   #  # ]
    6199                 :             :                         {
    6200                 :           0 :                                 SortGroupClause *grp = (SortGroupClause *) lfirst(l);
    6201                 :             : 
    6202                 :           0 :                                 appendStringInfoString(buf, sep);
    6203                 :           0 :                                 get_rule_sortgroupclause(grp->tleSortGroupRef, query->targetList,
    6204                 :           0 :                                                                                  false, context);
    6205                 :           0 :                                 sep = ", ";
    6206                 :           0 :                         }
    6207                 :           0 :                 }
    6208                 :             :                 else
    6209                 :             :                 {
    6210                 :           0 :                         sep = "";
    6211   [ #  #  #  #  :           0 :                         foreach(l, query->groupingSets)
                   #  # ]
    6212                 :             :                         {
    6213                 :           0 :                                 GroupingSet *grp = lfirst(l);
    6214                 :             : 
    6215                 :           0 :                                 appendStringInfoString(buf, sep);
    6216                 :           0 :                                 get_rule_groupingset(grp, query->targetList, true, context);
    6217                 :           0 :                                 sep = ", ";
    6218                 :           0 :                         }
    6219                 :             :                 }
    6220                 :             : 
    6221                 :           0 :                 context->inGroupBy = save_ingroupby;
    6222                 :           0 :         }
    6223                 :             : 
    6224                 :             :         /* Add the HAVING clause if given */
    6225         [ #  # ]:           0 :         if (query->havingQual != NULL)
    6226                 :             :         {
    6227                 :           0 :                 appendContextKeyword(context, " HAVING ",
    6228                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 0);
    6229                 :           0 :                 get_rule_expr(query->havingQual, context, false);
    6230                 :           0 :         }
    6231                 :             : 
    6232                 :             :         /* Add the WINDOW clause if needed */
    6233         [ #  # ]:           0 :         if (query->windowClause != NIL)
    6234                 :           0 :                 get_rule_windowclause(query, context);
    6235         [ #  # ]:           0 : }
    6236                 :             : 
    6237                 :             : /* ----------
    6238                 :             :  * get_target_list                      - Parse back a SELECT target list
    6239                 :             :  *
    6240                 :             :  * This is also used for RETURNING lists in INSERT/UPDATE/DELETE/MERGE.
    6241                 :             :  * ----------
    6242                 :             :  */
    6243                 :             : static void
    6244                 :           0 : get_target_list(List *targetList, deparse_context *context)
    6245                 :             : {
    6246                 :           0 :         StringInfo      buf = context->buf;
    6247                 :           0 :         StringInfoData targetbuf;
    6248                 :           0 :         bool            last_was_multiline = false;
    6249                 :           0 :         char       *sep;
    6250                 :           0 :         int                     colno;
    6251                 :           0 :         ListCell   *l;
    6252                 :             : 
    6253                 :             :         /* we use targetbuf to hold each TLE's text temporarily */
    6254                 :           0 :         initStringInfo(&targetbuf);
    6255                 :             : 
    6256                 :           0 :         sep = " ";
    6257                 :           0 :         colno = 0;
    6258   [ #  #  #  #  :           0 :         foreach(l, targetList)
                   #  # ]
    6259                 :             :         {
    6260                 :           0 :                 TargetEntry *tle = (TargetEntry *) lfirst(l);
    6261                 :           0 :                 char       *colname;
    6262                 :           0 :                 char       *attname;
    6263                 :             : 
    6264         [ #  # ]:           0 :                 if (tle->resjunk)
    6265                 :           0 :                         continue;                       /* ignore junk entries */
    6266                 :             : 
    6267                 :           0 :                 appendStringInfoString(buf, sep);
    6268                 :           0 :                 sep = ", ";
    6269                 :           0 :                 colno++;
    6270                 :             : 
    6271                 :             :                 /*
    6272                 :             :                  * Put the new field text into targetbuf so we can decide after we've
    6273                 :             :                  * got it whether or not it needs to go on a new line.
    6274                 :             :                  */
    6275                 :           0 :                 resetStringInfo(&targetbuf);
    6276                 :           0 :                 context->buf = &targetbuf;
    6277                 :             : 
    6278                 :             :                 /*
    6279                 :             :                  * We special-case Var nodes rather than using get_rule_expr. This is
    6280                 :             :                  * needed because get_rule_expr will display a whole-row Var as
    6281                 :             :                  * "foo.*", which is the preferred notation in most contexts, but at
    6282                 :             :                  * the top level of a SELECT list it's not right (the parser will
    6283                 :             :                  * expand that notation into multiple columns, yielding behavior
    6284                 :             :                  * different from a whole-row Var).  We need to call get_variable
    6285                 :             :                  * directly so that we can tell it to do the right thing, and so that
    6286                 :             :                  * we can get the attribute name which is the default AS label.
    6287                 :             :                  */
    6288   [ #  #  #  # ]:           0 :                 if (tle->expr && (IsA(tle->expr, Var)))
    6289                 :             :                 {
    6290                 :           0 :                         attname = get_variable((Var *) tle->expr, 0, true, context);
    6291                 :           0 :                 }
    6292                 :             :                 else
    6293                 :             :                 {
    6294                 :           0 :                         get_rule_expr((Node *) tle->expr, context, true);
    6295                 :             : 
    6296                 :             :                         /*
    6297                 :             :                          * When colNamesVisible is true, we should always show the
    6298                 :             :                          * assigned column name explicitly.  Otherwise, show it only if
    6299                 :             :                          * it's not FigureColname's fallback.
    6300                 :             :                          */
    6301                 :           0 :                         attname = context->colNamesVisible ? NULL : "?column?";
    6302                 :             :                 }
    6303                 :             : 
    6304                 :             :                 /*
    6305                 :             :                  * Figure out what the result column should be called.  In the context
    6306                 :             :                  * of a view, use the view's tuple descriptor (so as to pick up the
    6307                 :             :                  * effects of any column RENAME that's been done on the view).
    6308                 :             :                  * Otherwise, just use what we can find in the TLE.
    6309                 :             :                  */
    6310   [ #  #  #  # ]:           0 :                 if (context->resultDesc && colno <= context->resultDesc->natts)
    6311                 :           0 :                         colname = NameStr(TupleDescAttr(context->resultDesc,
    6312                 :             :                                                                                         colno - 1)->attname);
    6313                 :             :                 else
    6314                 :           0 :                         colname = tle->resname;
    6315                 :             : 
    6316                 :             :                 /* Show AS unless the column's name is correct as-is */
    6317         [ #  # ]:           0 :                 if (colname)                    /* resname could be NULL */
    6318                 :             :                 {
    6319   [ #  #  #  # ]:           0 :                         if (attname == NULL || strcmp(attname, colname) != 0)
    6320                 :           0 :                                 appendStringInfo(&targetbuf, " AS %s", quote_identifier(colname));
    6321                 :           0 :                 }
    6322                 :             : 
    6323                 :             :                 /* Restore context's output buffer */
    6324                 :           0 :                 context->buf = buf;
    6325                 :             : 
    6326                 :             :                 /* Consider line-wrapping if enabled */
    6327   [ #  #  #  # ]:           0 :                 if (PRETTY_INDENT(context) && context->wrapColumn >= 0)
    6328                 :             :                 {
    6329                 :           0 :                         int                     leading_nl_pos;
    6330                 :             : 
    6331                 :             :                         /* Does the new field start with a new line? */
    6332   [ #  #  #  # ]:           0 :                         if (targetbuf.len > 0 && targetbuf.data[0] == '\n')
    6333                 :           0 :                                 leading_nl_pos = 0;
    6334                 :             :                         else
    6335                 :           0 :                                 leading_nl_pos = -1;
    6336                 :             : 
    6337                 :             :                         /* If so, we shouldn't add anything */
    6338         [ #  # ]:           0 :                         if (leading_nl_pos >= 0)
    6339                 :             :                         {
    6340                 :             :                                 /* instead, remove any trailing spaces currently in buf */
    6341                 :           0 :                                 removeStringInfoSpaces(buf);
    6342                 :           0 :                         }
    6343                 :             :                         else
    6344                 :             :                         {
    6345                 :           0 :                                 char       *trailing_nl;
    6346                 :             : 
    6347                 :             :                                 /* Locate the start of the current line in the output buffer */
    6348                 :           0 :                                 trailing_nl = strrchr(buf->data, '\n');
    6349         [ #  # ]:           0 :                                 if (trailing_nl == NULL)
    6350                 :           0 :                                         trailing_nl = buf->data;
    6351                 :             :                                 else
    6352                 :           0 :                                         trailing_nl++;
    6353                 :             : 
    6354                 :             :                                 /*
    6355                 :             :                                  * Add a newline, plus some indentation, if the new field is
    6356                 :             :                                  * not the first and either the new field would cause an
    6357                 :             :                                  * overflow or the last field used more than one line.
    6358                 :             :                                  */
    6359   [ #  #  #  # ]:           0 :                                 if (colno > 1 &&
    6360         [ #  # ]:           0 :                                         ((strlen(trailing_nl) + targetbuf.len > context->wrapColumn) ||
    6361                 :           0 :                                          last_was_multiline))
    6362                 :           0 :                                         appendContextKeyword(context, "", -PRETTYINDENT_STD,
    6363                 :             :                                                                                  PRETTYINDENT_STD, PRETTYINDENT_VAR);
    6364                 :           0 :                         }
    6365                 :             : 
    6366                 :             :                         /* Remember this field's multiline status for next iteration */
    6367                 :           0 :                         last_was_multiline =
    6368                 :           0 :                                 (strchr(targetbuf.data + leading_nl_pos + 1, '\n') != NULL);
    6369                 :           0 :                 }
    6370                 :             : 
    6371                 :             :                 /* Add the new field */
    6372                 :           0 :                 appendBinaryStringInfo(buf, targetbuf.data, targetbuf.len);
    6373      [ #  #  # ]:           0 :         }
    6374                 :             : 
    6375                 :             :         /* clean up */
    6376                 :           0 :         pfree(targetbuf.data);
    6377                 :           0 : }
    6378                 :             : 
    6379                 :             : static void
    6380                 :           0 : get_returning_clause(Query *query, deparse_context *context)
    6381                 :             : {
    6382                 :           0 :         StringInfo      buf = context->buf;
    6383                 :             : 
    6384         [ #  # ]:           0 :         if (query->returningList)
    6385                 :             :         {
    6386                 :           0 :                 bool            have_with = false;
    6387                 :             : 
    6388                 :           0 :                 appendContextKeyword(context, " RETURNING",
    6389                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 1);
    6390                 :             : 
    6391                 :             :                 /* Add WITH (OLD/NEW) options, if they're not the defaults */
    6392   [ #  #  #  # ]:           0 :                 if (query->returningOldAlias && strcmp(query->returningOldAlias, "old") != 0)
    6393                 :             :                 {
    6394                 :           0 :                         appendStringInfo(buf, " WITH (OLD AS %s",
    6395                 :           0 :                                                          quote_identifier(query->returningOldAlias));
    6396                 :           0 :                         have_with = true;
    6397                 :           0 :                 }
    6398   [ #  #  #  # ]:           0 :                 if (query->returningNewAlias && strcmp(query->returningNewAlias, "new") != 0)
    6399                 :             :                 {
    6400         [ #  # ]:           0 :                         if (have_with)
    6401                 :           0 :                                 appendStringInfo(buf, ", NEW AS %s",
    6402                 :           0 :                                                                  quote_identifier(query->returningNewAlias));
    6403                 :             :                         else
    6404                 :             :                         {
    6405                 :           0 :                                 appendStringInfo(buf, " WITH (NEW AS %s",
    6406                 :           0 :                                                                  quote_identifier(query->returningNewAlias));
    6407                 :           0 :                                 have_with = true;
    6408                 :             :                         }
    6409                 :           0 :                 }
    6410         [ #  # ]:           0 :                 if (have_with)
    6411                 :           0 :                         appendStringInfoChar(buf, ')');
    6412                 :             : 
    6413                 :             :                 /* Add the returning expressions themselves */
    6414                 :           0 :                 get_target_list(query->returningList, context);
    6415                 :           0 :         }
    6416                 :           0 : }
    6417                 :             : 
    6418                 :             : static void
    6419                 :           0 : get_setop_query(Node *setOp, Query *query, deparse_context *context)
    6420                 :             : {
    6421                 :           0 :         StringInfo      buf = context->buf;
    6422                 :           0 :         bool            need_paren;
    6423                 :             : 
    6424                 :             :         /* Guard against excessively long or deeply-nested queries */
    6425         [ #  # ]:           0 :         CHECK_FOR_INTERRUPTS();
    6426                 :           0 :         check_stack_depth();
    6427                 :             : 
    6428         [ #  # ]:           0 :         if (IsA(setOp, RangeTblRef))
    6429                 :             :         {
    6430                 :           0 :                 RangeTblRef *rtr = (RangeTblRef *) setOp;
    6431                 :           0 :                 RangeTblEntry *rte = rt_fetch(rtr->rtindex, query->rtable);
    6432                 :           0 :                 Query      *subquery = rte->subquery;
    6433                 :             : 
    6434         [ #  # ]:           0 :                 Assert(subquery != NULL);
    6435                 :             : 
    6436                 :             :                 /*
    6437                 :             :                  * We need parens if WITH, ORDER BY, FOR UPDATE, or LIMIT; see gram.y.
    6438                 :             :                  * Also add parens if the leaf query contains its own set operations.
    6439                 :             :                  * (That shouldn't happen unless one of the other clauses is also
    6440                 :             :                  * present, see transformSetOperationTree; but let's be safe.)
    6441                 :             :                  */
    6442         [ #  # ]:           0 :                 need_paren = (subquery->cteList ||
    6443         [ #  # ]:           0 :                                           subquery->sortClause ||
    6444         [ #  # ]:           0 :                                           subquery->rowMarks ||
    6445         [ #  # ]:           0 :                                           subquery->limitOffset ||
    6446         [ #  # ]:           0 :                                           subquery->limitCount ||
    6447                 :           0 :                                           subquery->setOperations);
    6448         [ #  # ]:           0 :                 if (need_paren)
    6449                 :           0 :                         appendStringInfoChar(buf, '(');
    6450                 :           0 :                 get_query_def(subquery, buf, context->namespaces,
    6451                 :           0 :                                           context->resultDesc, context->colNamesVisible,
    6452                 :           0 :                                           context->prettyFlags, context->wrapColumn,
    6453                 :           0 :                                           context->indentLevel);
    6454         [ #  # ]:           0 :                 if (need_paren)
    6455                 :           0 :                         appendStringInfoChar(buf, ')');
    6456                 :           0 :         }
    6457         [ #  # ]:           0 :         else if (IsA(setOp, SetOperationStmt))
    6458                 :             :         {
    6459                 :           0 :                 SetOperationStmt *op = (SetOperationStmt *) setOp;
    6460                 :           0 :                 int                     subindent;
    6461                 :           0 :                 bool            save_colnamesvisible;
    6462                 :             : 
    6463                 :             :                 /*
    6464                 :             :                  * We force parens when nesting two SetOperationStmts, except when the
    6465                 :             :                  * lefthand input is another setop of the same kind.  Syntactically,
    6466                 :             :                  * we could omit parens in rather more cases, but it seems best to use
    6467                 :             :                  * parens to flag cases where the setop operator changes.  If we use
    6468                 :             :                  * parens, we also increase the indentation level for the child query.
    6469                 :             :                  *
    6470                 :             :                  * There are some cases in which parens are needed around a leaf query
    6471                 :             :                  * too, but those are more easily handled at the next level down (see
    6472                 :             :                  * code above).
    6473                 :             :                  */
    6474         [ #  # ]:           0 :                 if (IsA(op->larg, SetOperationStmt))
    6475                 :             :                 {
    6476                 :           0 :                         SetOperationStmt *lop = (SetOperationStmt *) op->larg;
    6477                 :             : 
    6478   [ #  #  #  # ]:           0 :                         if (op->op == lop->op && op->all == lop->all)
    6479                 :           0 :                                 need_paren = false;
    6480                 :             :                         else
    6481                 :           0 :                                 need_paren = true;
    6482                 :           0 :                 }
    6483                 :             :                 else
    6484                 :           0 :                         need_paren = false;
    6485                 :             : 
    6486         [ #  # ]:           0 :                 if (need_paren)
    6487                 :             :                 {
    6488                 :           0 :                         appendStringInfoChar(buf, '(');
    6489                 :           0 :                         subindent = PRETTYINDENT_STD;
    6490                 :           0 :                         appendContextKeyword(context, "", subindent, 0, 0);
    6491                 :           0 :                 }
    6492                 :             :                 else
    6493                 :           0 :                         subindent = 0;
    6494                 :             : 
    6495                 :           0 :                 get_setop_query(op->larg, query, context);
    6496                 :             : 
    6497         [ #  # ]:           0 :                 if (need_paren)
    6498                 :           0 :                         appendContextKeyword(context, ") ", -subindent, 0, 0);
    6499         [ #  # ]:           0 :                 else if (PRETTY_INDENT(context))
    6500                 :           0 :                         appendContextKeyword(context, "", -subindent, 0, 0);
    6501                 :             :                 else
    6502                 :           0 :                         appendStringInfoChar(buf, ' ');
    6503                 :             : 
    6504   [ #  #  #  # ]:           0 :                 switch (op->op)
    6505                 :             :                 {
    6506                 :             :                         case SETOP_UNION:
    6507                 :           0 :                                 appendStringInfoString(buf, "UNION ");
    6508                 :           0 :                                 break;
    6509                 :             :                         case SETOP_INTERSECT:
    6510                 :           0 :                                 appendStringInfoString(buf, "INTERSECT ");
    6511                 :           0 :                                 break;
    6512                 :             :                         case SETOP_EXCEPT:
    6513                 :           0 :                                 appendStringInfoString(buf, "EXCEPT ");
    6514                 :           0 :                                 break;
    6515                 :             :                         default:
    6516   [ #  #  #  # ]:           0 :                                 elog(ERROR, "unrecognized set op: %d",
    6517                 :             :                                          (int) op->op);
    6518                 :           0 :                 }
    6519         [ #  # ]:           0 :                 if (op->all)
    6520                 :           0 :                         appendStringInfoString(buf, "ALL ");
    6521                 :             : 
    6522                 :             :                 /* Always parenthesize if RHS is another setop */
    6523                 :           0 :                 need_paren = IsA(op->rarg, SetOperationStmt);
    6524                 :             : 
    6525                 :             :                 /*
    6526                 :             :                  * The indentation code here is deliberately a bit different from that
    6527                 :             :                  * for the lefthand input, because we want the line breaks in
    6528                 :             :                  * different places.
    6529                 :             :                  */
    6530         [ #  # ]:           0 :                 if (need_paren)
    6531                 :             :                 {
    6532                 :           0 :                         appendStringInfoChar(buf, '(');
    6533                 :           0 :                         subindent = PRETTYINDENT_STD;
    6534                 :           0 :                 }
    6535                 :             :                 else
    6536                 :           0 :                         subindent = 0;
    6537                 :           0 :                 appendContextKeyword(context, "", subindent, 0, 0);
    6538                 :             : 
    6539                 :             :                 /*
    6540                 :             :                  * The output column names of the RHS sub-select don't matter.
    6541                 :             :                  */
    6542                 :           0 :                 save_colnamesvisible = context->colNamesVisible;
    6543                 :           0 :                 context->colNamesVisible = false;
    6544                 :             : 
    6545                 :           0 :                 get_setop_query(op->rarg, query, context);
    6546                 :             : 
    6547                 :           0 :                 context->colNamesVisible = save_colnamesvisible;
    6548                 :             : 
    6549         [ #  # ]:           0 :                 if (PRETTY_INDENT(context))
    6550                 :           0 :                         context->indentLevel -= subindent;
    6551         [ #  # ]:           0 :                 if (need_paren)
    6552                 :           0 :                         appendContextKeyword(context, ")", 0, 0, 0);
    6553                 :           0 :         }
    6554                 :             :         else
    6555                 :             :         {
    6556   [ #  #  #  # ]:           0 :                 elog(ERROR, "unrecognized node type: %d",
    6557                 :             :                          (int) nodeTag(setOp));
    6558                 :             :         }
    6559                 :           0 : }
    6560                 :             : 
    6561                 :             : /*
    6562                 :             :  * Display a sort/group clause.
    6563                 :             :  *
    6564                 :             :  * Also returns the expression tree, so caller need not find it again.
    6565                 :             :  */
    6566                 :             : static Node *
    6567                 :           0 : get_rule_sortgroupclause(Index ref, List *tlist, bool force_colno,
    6568                 :             :                                                  deparse_context *context)
    6569                 :             : {
    6570                 :           0 :         StringInfo      buf = context->buf;
    6571                 :           0 :         TargetEntry *tle;
    6572                 :           0 :         Node       *expr;
    6573                 :             : 
    6574                 :           0 :         tle = get_sortgroupref_tle(ref, tlist);
    6575                 :           0 :         expr = (Node *) tle->expr;
    6576                 :             : 
    6577                 :             :         /*
    6578                 :             :          * Use column-number form if requested by caller.  Otherwise, if
    6579                 :             :          * expression is a constant, force it to be dumped with an explicit cast
    6580                 :             :          * as decoration --- this is because a simple integer constant is
    6581                 :             :          * ambiguous (and will be misinterpreted by findTargetlistEntrySQL92()) if
    6582                 :             :          * we dump it without any decoration.  Similarly, if it's just a Var,
    6583                 :             :          * there is risk of misinterpretation if the column name is reassigned in
    6584                 :             :          * the SELECT list, so we may need to force table qualification.  And, if
    6585                 :             :          * it's anything more complex than a simple Var, then force extra parens
    6586                 :             :          * around it, to ensure it can't be misinterpreted as a cube() or rollup()
    6587                 :             :          * construct.
    6588                 :             :          */
    6589         [ #  # ]:           0 :         if (force_colno)
    6590                 :             :         {
    6591         [ #  # ]:           0 :                 Assert(!tle->resjunk);
    6592                 :           0 :                 appendStringInfo(buf, "%d", tle->resno);
    6593                 :           0 :         }
    6594         [ #  # ]:           0 :         else if (!expr)
    6595                 :             :                  /* do nothing, probably can't happen */ ;
    6596         [ #  # ]:           0 :         else if (IsA(expr, Const))
    6597                 :           0 :                 get_const_expr((Const *) expr, context, 1);
    6598         [ #  # ]:           0 :         else if (IsA(expr, Var))
    6599                 :             :         {
    6600                 :             :                 /* Tell get_variable to check for name conflict */
    6601                 :           0 :                 bool            save_varinorderby = context->varInOrderBy;
    6602                 :             : 
    6603                 :           0 :                 context->varInOrderBy = true;
    6604                 :           0 :                 (void) get_variable((Var *) expr, 0, false, context);
    6605                 :           0 :                 context->varInOrderBy = save_varinorderby;
    6606                 :           0 :         }
    6607                 :             :         else
    6608                 :             :         {
    6609                 :             :                 /*
    6610                 :             :                  * We must force parens for function-like expressions even if
    6611                 :             :                  * PRETTY_PAREN is off, since those are the ones in danger of
    6612                 :             :                  * misparsing. For other expressions we need to force them only if
    6613                 :             :                  * PRETTY_PAREN is on, since otherwise the expression will output them
    6614                 :             :                  * itself. (We can't skip the parens.)
    6615                 :             :                  */
    6616                 :           0 :                 bool            need_paren = (PRETTY_PAREN(context)
    6617         [ #  # ]:           0 :                                                                   || IsA(expr, FuncExpr)
    6618         [ #  # ]:           0 :                                                                   || IsA(expr, Aggref)
    6619         [ #  # ]:           0 :                                                                   || IsA(expr, WindowFunc)
    6620         [ #  # ]:           0 :                                                                   || IsA(expr, JsonConstructorExpr));
    6621                 :             : 
    6622         [ #  # ]:           0 :                 if (need_paren)
    6623                 :           0 :                         appendStringInfoChar(context->buf, '(');
    6624                 :           0 :                 get_rule_expr(expr, context, true);
    6625         [ #  # ]:           0 :                 if (need_paren)
    6626                 :           0 :                         appendStringInfoChar(context->buf, ')');
    6627                 :           0 :         }
    6628                 :             : 
    6629                 :           0 :         return expr;
    6630                 :           0 : }
    6631                 :             : 
    6632                 :             : /*
    6633                 :             :  * Display a GroupingSet
    6634                 :             :  */
    6635                 :             : static void
    6636                 :           0 : get_rule_groupingset(GroupingSet *gset, List *targetlist,
    6637                 :             :                                          bool omit_parens, deparse_context *context)
    6638                 :             : {
    6639                 :           0 :         ListCell   *l;
    6640                 :           0 :         StringInfo      buf = context->buf;
    6641                 :           0 :         bool            omit_child_parens = true;
    6642                 :           0 :         char       *sep = "";
    6643                 :             : 
    6644   [ #  #  #  #  :           0 :         switch (gset->kind)
                   #  # ]
    6645                 :             :         {
    6646                 :             :                 case GROUPING_SET_EMPTY:
    6647                 :           0 :                         appendStringInfoString(buf, "()");
    6648                 :           0 :                         return;
    6649                 :             : 
    6650                 :             :                 case GROUPING_SET_SIMPLE:
    6651                 :             :                         {
    6652   [ #  #  #  # ]:           0 :                                 if (!omit_parens || list_length(gset->content) != 1)
    6653                 :           0 :                                         appendStringInfoChar(buf, '(');
    6654                 :             : 
    6655   [ #  #  #  #  :           0 :                                 foreach(l, gset->content)
                   #  # ]
    6656                 :             :                                 {
    6657                 :           0 :                                         Index           ref = lfirst_int(l);
    6658                 :             : 
    6659                 :           0 :                                         appendStringInfoString(buf, sep);
    6660                 :           0 :                                         get_rule_sortgroupclause(ref, targetlist,
    6661                 :           0 :                                                                                          false, context);
    6662                 :           0 :                                         sep = ", ";
    6663                 :           0 :                                 }
    6664                 :             : 
    6665   [ #  #  #  # ]:           0 :                                 if (!omit_parens || list_length(gset->content) != 1)
    6666                 :           0 :                                         appendStringInfoChar(buf, ')');
    6667                 :             :                         }
    6668                 :           0 :                         return;
    6669                 :             : 
    6670                 :             :                 case GROUPING_SET_ROLLUP:
    6671                 :           0 :                         appendStringInfoString(buf, "ROLLUP(");
    6672                 :           0 :                         break;
    6673                 :             :                 case GROUPING_SET_CUBE:
    6674                 :           0 :                         appendStringInfoString(buf, "CUBE(");
    6675                 :           0 :                         break;
    6676                 :             :                 case GROUPING_SET_SETS:
    6677                 :           0 :                         appendStringInfoString(buf, "GROUPING SETS (");
    6678                 :           0 :                         omit_child_parens = false;
    6679                 :           0 :                         break;
    6680                 :             :         }
    6681                 :             : 
    6682   [ #  #  #  #  :           0 :         foreach(l, gset->content)
                   #  # ]
    6683                 :             :         {
    6684                 :           0 :                 appendStringInfoString(buf, sep);
    6685                 :           0 :                 get_rule_groupingset(lfirst(l), targetlist, omit_child_parens, context);
    6686                 :           0 :                 sep = ", ";
    6687                 :           0 :         }
    6688                 :             : 
    6689                 :           0 :         appendStringInfoChar(buf, ')');
    6690         [ #  # ]:           0 : }
    6691                 :             : 
    6692                 :             : /*
    6693                 :             :  * Display an ORDER BY list.
    6694                 :             :  */
    6695                 :             : static void
    6696                 :           0 : get_rule_orderby(List *orderList, List *targetList,
    6697                 :             :                                  bool force_colno, deparse_context *context)
    6698                 :             : {
    6699                 :           0 :         StringInfo      buf = context->buf;
    6700                 :           0 :         const char *sep;
    6701                 :           0 :         ListCell   *l;
    6702                 :             : 
    6703                 :           0 :         sep = "";
    6704   [ #  #  #  #  :           0 :         foreach(l, orderList)
                   #  # ]
    6705                 :             :         {
    6706                 :           0 :                 SortGroupClause *srt = (SortGroupClause *) lfirst(l);
    6707                 :           0 :                 Node       *sortexpr;
    6708                 :           0 :                 Oid                     sortcoltype;
    6709                 :           0 :                 TypeCacheEntry *typentry;
    6710                 :             : 
    6711                 :           0 :                 appendStringInfoString(buf, sep);
    6712                 :           0 :                 sortexpr = get_rule_sortgroupclause(srt->tleSortGroupRef, targetList,
    6713                 :           0 :                                                                                         force_colno, context);
    6714                 :           0 :                 sortcoltype = exprType(sortexpr);
    6715                 :             :                 /* See whether operator is default < or > for datatype */
    6716                 :           0 :                 typentry = lookup_type_cache(sortcoltype,
    6717                 :             :                                                                          TYPECACHE_LT_OPR | TYPECACHE_GT_OPR);
    6718         [ #  # ]:           0 :                 if (srt->sortop == typentry->lt_opr)
    6719                 :             :                 {
    6720                 :             :                         /* ASC is default, so emit nothing for it */
    6721         [ #  # ]:           0 :                         if (srt->nulls_first)
    6722                 :           0 :                                 appendStringInfoString(buf, " NULLS FIRST");
    6723                 :           0 :                 }
    6724         [ #  # ]:           0 :                 else if (srt->sortop == typentry->gt_opr)
    6725                 :             :                 {
    6726                 :           0 :                         appendStringInfoString(buf, " DESC");
    6727                 :             :                         /* DESC defaults to NULLS FIRST */
    6728         [ #  # ]:           0 :                         if (!srt->nulls_first)
    6729                 :           0 :                                 appendStringInfoString(buf, " NULLS LAST");
    6730                 :           0 :                 }
    6731                 :             :                 else
    6732                 :             :                 {
    6733                 :           0 :                         appendStringInfo(buf, " USING %s",
    6734                 :           0 :                                                          generate_operator_name(srt->sortop,
    6735                 :           0 :                                                                                                         sortcoltype,
    6736                 :           0 :                                                                                                         sortcoltype));
    6737                 :             :                         /* be specific to eliminate ambiguity */
    6738         [ #  # ]:           0 :                         if (srt->nulls_first)
    6739                 :           0 :                                 appendStringInfoString(buf, " NULLS FIRST");
    6740                 :             :                         else
    6741                 :           0 :                                 appendStringInfoString(buf, " NULLS LAST");
    6742                 :             :                 }
    6743                 :           0 :                 sep = ", ";
    6744                 :           0 :         }
    6745                 :           0 : }
    6746                 :             : 
    6747                 :             : /*
    6748                 :             :  * Display a WINDOW clause.
    6749                 :             :  *
    6750                 :             :  * Note that the windowClause list might contain only anonymous window
    6751                 :             :  * specifications, in which case we should print nothing here.
    6752                 :             :  */
    6753                 :             : static void
    6754                 :           0 : get_rule_windowclause(Query *query, deparse_context *context)
    6755                 :             : {
    6756                 :           0 :         StringInfo      buf = context->buf;
    6757                 :           0 :         const char *sep;
    6758                 :           0 :         ListCell   *l;
    6759                 :             : 
    6760                 :           0 :         sep = NULL;
    6761   [ #  #  #  #  :           0 :         foreach(l, query->windowClause)
                   #  # ]
    6762                 :             :         {
    6763                 :           0 :                 WindowClause *wc = (WindowClause *) lfirst(l);
    6764                 :             : 
    6765         [ #  # ]:           0 :                 if (wc->name == NULL)
    6766                 :           0 :                         continue;                       /* ignore anonymous windows */
    6767                 :             : 
    6768         [ #  # ]:           0 :                 if (sep == NULL)
    6769                 :           0 :                         appendContextKeyword(context, " WINDOW ",
    6770                 :             :                                                                  -PRETTYINDENT_STD, PRETTYINDENT_STD, 1);
    6771                 :             :                 else
    6772                 :           0 :                         appendStringInfoString(buf, sep);
    6773                 :             : 
    6774                 :           0 :                 appendStringInfo(buf, "%s AS ", quote_identifier(wc->name));
    6775                 :             : 
    6776                 :           0 :                 get_rule_windowspec(wc, query->targetList, context);
    6777                 :             : 
    6778                 :           0 :                 sep = ", ";
    6779      [ #  #  # ]:           0 :         }
    6780                 :           0 : }
    6781                 :             : 
    6782                 :             : /*
    6783                 :             :  * Display a window definition
    6784                 :             :  */
    6785                 :             : static void
    6786                 :           0 : get_rule_windowspec(WindowClause *wc, List *targetList,
    6787                 :             :                                         deparse_context *context)
    6788                 :             : {
    6789                 :           0 :         StringInfo      buf = context->buf;
    6790                 :           0 :         bool            needspace = false;
    6791                 :           0 :         const char *sep;
    6792                 :           0 :         ListCell   *l;
    6793                 :             : 
    6794                 :           0 :         appendStringInfoChar(buf, '(');
    6795         [ #  # ]:           0 :         if (wc->refname)
    6796                 :             :         {
    6797                 :           0 :                 appendStringInfoString(buf, quote_identifier(wc->refname));
    6798                 :           0 :                 needspace = true;
    6799                 :           0 :         }
    6800                 :             :         /* partition clauses are always inherited, so only print if no refname */
    6801   [ #  #  #  # ]:           0 :         if (wc->partitionClause && !wc->refname)
    6802                 :             :         {
    6803         [ #  # ]:           0 :                 if (needspace)
    6804                 :           0 :                         appendStringInfoChar(buf, ' ');
    6805                 :           0 :                 appendStringInfoString(buf, "PARTITION BY ");
    6806                 :           0 :                 sep = "";
    6807   [ #  #  #  #  :           0 :                 foreach(l, wc->partitionClause)
                   #  # ]
    6808                 :             :                 {
    6809                 :           0 :                         SortGroupClause *grp = (SortGroupClause *) lfirst(l);
    6810                 :             : 
    6811                 :           0 :                         appendStringInfoString(buf, sep);
    6812                 :           0 :                         get_rule_sortgroupclause(grp->tleSortGroupRef, targetList,
    6813                 :           0 :                                                                          false, context);
    6814                 :           0 :                         sep = ", ";
    6815                 :           0 :                 }
    6816                 :           0 :                 needspace = true;
    6817                 :           0 :         }
    6818                 :             :         /* print ordering clause only if not inherited */
    6819   [ #  #  #  # ]:           0 :         if (wc->orderClause && !wc->copiedOrder)
    6820                 :             :         {
    6821         [ #  # ]:           0 :                 if (needspace)
    6822                 :           0 :                         appendStringInfoChar(buf, ' ');
    6823                 :           0 :                 appendStringInfoString(buf, "ORDER BY ");
    6824                 :           0 :                 get_rule_orderby(wc->orderClause, targetList, false, context);
    6825                 :           0 :                 needspace = true;
    6826                 :           0 :         }
    6827                 :             :         /* framing clause is never inherited, so print unless it's default */
    6828         [ #  # ]:           0 :         if (wc->frameOptions & FRAMEOPTION_NONDEFAULT)
    6829                 :             :         {
    6830         [ #  # ]:           0 :                 if (needspace)
    6831                 :           0 :                         appendStringInfoChar(buf, ' ');
    6832                 :           0 :                 get_window_frame_options(wc->frameOptions,
    6833                 :           0 :                                                                  wc->startOffset, wc->endOffset,
    6834                 :           0 :                                                                  context);
    6835                 :           0 :         }
    6836                 :           0 :         appendStringInfoChar(buf, ')');
    6837                 :           0 : }
    6838                 :             : 
    6839                 :             : /*
    6840                 :             :  * Append the description of a window's framing options to context->buf
    6841                 :             :  */
    6842                 :             : static void
    6843                 :           0 : get_window_frame_options(int frameOptions,
    6844                 :             :                                                  Node *startOffset, Node *endOffset,
    6845                 :             :                                                  deparse_context *context)
    6846                 :             : {
    6847                 :           0 :         StringInfo      buf = context->buf;
    6848                 :             : 
    6849         [ #  # ]:           0 :         if (frameOptions & FRAMEOPTION_NONDEFAULT)
    6850                 :             :         {
    6851         [ #  # ]:           0 :                 if (frameOptions & FRAMEOPTION_RANGE)
    6852                 :           0 :                         appendStringInfoString(buf, "RANGE ");
    6853         [ #  # ]:           0 :                 else if (frameOptions & FRAMEOPTION_ROWS)
    6854                 :           0 :                         appendStringInfoString(buf, "ROWS ");
    6855         [ #  # ]:           0 :                 else if (frameOptions & FRAMEOPTION_GROUPS)
    6856                 :           0 :                         appendStringInfoString(buf, "GROUPS ");
    6857                 :             :                 else
    6858                 :           0 :                         Assert(false);
    6859         [ #  # ]:           0 :                 if (frameOptions & FRAMEOPTION_BETWEEN)
    6860                 :           0 :                         appendStringInfoString(buf, "BETWEEN ");
    6861         [ #  # ]:           0 :                 if (frameOptions & FRAMEOPTION_START_UNBOUNDED_PRECEDING)
    6862                 :           0 :                         appendStringInfoString(buf, "UNBOUNDED PRECEDING ");
    6863         [ #  # ]:           0 :                 else if (frameOptions & FRAMEOPTION_START_CURRENT_ROW)
    6864                 :           0 :                         appendStringInfoString(buf, "CURRENT ROW ");
    6865         [ #  # ]:           0 :                 else if (frameOptions & FRAMEOPTION_START_OFFSET)
    6866                 :             :                 {
    6867                 :           0 :                         get_rule_expr(startOffset, context, false);
    6868         [ #  # ]:           0 :                         if (frameOptions & FRAMEOPTION_START_OFFSET_PRECEDING)
    6869                 :           0 :                                 appendStringInfoString(buf, " PRECEDING ");
    6870         [ #  # ]:           0 :                         else if (frameOptions & FRAMEOPTION_START_OFFSET_FOLLOWING)
    6871                 :           0 :                                 appendStringInfoString(buf, " FOLLOWING ");
    6872                 :             :                         else
    6873                 :           0 :                                 Assert(false);
    6874                 :           0 :                 }
    6875                 :             :                 else
    6876                 :           0 :                         Assert(false);
    6877         [ #  # ]:           0 :                 if (frameOptions & FRAMEOPTION_BETWEEN)
    6878                 :             :                 {
    6879                 :           0 :                         appendStringInfoString(buf, "AND ");
    6880         [ #  # ]:           0 :                         if (frameOptions & FRAMEOPTION_END_UNBOUNDED_FOLLOWING)
    6881                 :           0 :                                 appendStringInfoString(buf, "UNBOUNDED FOLLOWING ");
    6882         [ #  # ]:           0 :                         else if (frameOptions & FRAMEOPTION_END_CURRENT_ROW)
    6883                 :           0 :                                 appendStringInfoString(buf, "CURRENT ROW ");
    6884         [ #  # ]:           0 :                         else if (frameOptions & FRAMEOPTION_END_OFFSET)
    6885                 :             :                         {
    6886                 :           0 :                                 get_rule_expr(endOffset, context, false);
    6887         [ #  # ]:           0 :                                 if (frameOptions & FRAMEOPTION_END_OFFSET_PRECEDING)
    6888                 :           0 :                                         appendStringInfoString(buf, " PRECEDING ");
    6889         [ #  # ]:           0 :                                 else if (frameOptions & FRAMEOPTION_END_OFFSET_FOLLOWING)
    6890                 :           0 :                                         appendStringInfoString(buf, " FOLLOWING ");
    6891                 :             :                                 else
    6892                 :           0 :                                         Assert(false);
    6893                 :           0 :                         }
    6894                 :             :                         else
    6895                 :           0 :                                 Assert(false);
    6896                 :           0 :                 }
    6897         [ #  # ]:           0 :                 if (frameOptions & FRAMEOPTION_EXCLUDE_CURRENT_ROW)
    6898                 :           0 :                         appendStringInfoString(buf, "EXCLUDE CURRENT ROW ");
    6899         [ #  # ]:           0 :                 else if (frameOptions & FRAMEOPTION_EXCLUDE_GROUP)
    6900                 :           0 :                         appendStringInfoString(buf, "EXCLUDE GROUP ");
    6901         [ #  # ]:           0 :                 else if (frameOptions & FRAMEOPTION_EXCLUDE_TIES)
    6902                 :           0 :                         appendStringInfoString(buf, "EXCLUDE TIES ");
    6903                 :             :                 /* we will now have a trailing space; remove it */
    6904                 :           0 :                 buf->data[--(buf->len)] = '\0';
    6905                 :           0 :         }
    6906                 :           0 : }
    6907                 :             : 
    6908                 :             : /*
    6909                 :             :  * Return the description of a window's framing options as a palloc'd string
    6910                 :             :  */
    6911                 :             : char *
    6912                 :           0 : get_window_frame_options_for_explain(int frameOptions,
    6913                 :             :                                                                          Node *startOffset, Node *endOffset,
    6914                 :             :                                                                          List *dpcontext, bool forceprefix)
    6915                 :             : {
    6916                 :           0 :         StringInfoData buf;
    6917                 :           0 :         deparse_context context;
    6918                 :             : 
    6919                 :           0 :         initStringInfo(&buf);
    6920                 :           0 :         context.buf = &buf;
    6921                 :           0 :         context.namespaces = dpcontext;
    6922                 :           0 :         context.resultDesc = NULL;
    6923                 :           0 :         context.targetList = NIL;
    6924                 :           0 :         context.windowClause = NIL;
    6925                 :           0 :         context.varprefix = forceprefix;
    6926                 :           0 :         context.prettyFlags = 0;
    6927                 :           0 :         context.wrapColumn = WRAP_COLUMN_DEFAULT;
    6928                 :           0 :         context.indentLevel = 0;
    6929                 :           0 :         context.colNamesVisible = true;
    6930                 :           0 :         context.inGroupBy = false;
    6931                 :           0 :         context.varInOrderBy = false;
    6932                 :           0 :         context.appendparents = NULL;
    6933                 :             : 
    6934                 :           0 :         get_window_frame_options(frameOptions, startOffset, endOffset, &context);
    6935                 :             : 
    6936                 :           0 :         return buf.data;
    6937                 :           0 : }
    6938                 :             : 
    6939                 :             : /* ----------
    6940                 :             :  * get_insert_query_def                 - Parse back an INSERT parsetree
    6941                 :             :  * ----------
    6942                 :             :  */
    6943                 :             : static void
    6944                 :           0 : get_insert_query_def(Query *query, deparse_context *context)
    6945                 :             : {
    6946                 :           0 :         StringInfo      buf = context->buf;
    6947                 :           0 :         RangeTblEntry *select_rte = NULL;
    6948                 :           0 :         RangeTblEntry *values_rte = NULL;
    6949                 :           0 :         RangeTblEntry *rte;
    6950                 :           0 :         char       *sep;
    6951                 :           0 :         ListCell   *l;
    6952                 :           0 :         List       *strippedexprs;
    6953                 :             : 
    6954                 :             :         /* Insert the WITH clause if given */
    6955                 :           0 :         get_with_clause(query, context);
    6956                 :             : 
    6957                 :             :         /*
    6958                 :             :          * If it's an INSERT ... SELECT or multi-row VALUES, there will be a
    6959                 :             :          * single RTE for the SELECT or VALUES.  Plain VALUES has neither.
    6960                 :             :          */
    6961   [ #  #  #  #  :           0 :         foreach(l, query->rtable)
                   #  # ]
    6962                 :             :         {
    6963                 :           0 :                 rte = (RangeTblEntry *) lfirst(l);
    6964                 :             : 
    6965         [ #  # ]:           0 :                 if (rte->rtekind == RTE_SUBQUERY)
    6966                 :             :                 {
    6967         [ #  # ]:           0 :                         if (select_rte)
    6968   [ #  #  #  # ]:           0 :                                 elog(ERROR, "too many subquery RTEs in INSERT");
    6969                 :           0 :                         select_rte = rte;
    6970                 :           0 :                 }
    6971                 :             : 
    6972         [ #  # ]:           0 :                 if (rte->rtekind == RTE_VALUES)
    6973                 :             :                 {
    6974         [ #  # ]:           0 :                         if (values_rte)
    6975   [ #  #  #  # ]:           0 :                                 elog(ERROR, "too many values RTEs in INSERT");
    6976                 :           0 :                         values_rte = rte;
    6977                 :           0 :                 }
    6978                 :           0 :         }
    6979   [ #  #  #  # ]:           0 :         if (select_rte && values_rte)
    6980   [ #  #  #  # ]:           0 :                 elog(ERROR, "both subquery and values RTEs in INSERT");
    6981                 :             : 
    6982                 :             :         /*
    6983                 :             :          * Start the query with INSERT INTO relname
    6984                 :             :          */
    6985                 :           0 :         rte = rt_fetch(query->resultRelation, query->rtable);
    6986         [ #  # ]:           0 :         Assert(rte->rtekind == RTE_RELATION);
    6987                 :             : 
    6988         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
    6989                 :             :         {
    6990                 :           0 :                 context->indentLevel += PRETTYINDENT_STD;
    6991                 :           0 :                 appendStringInfoChar(buf, ' ');
    6992                 :           0 :         }
    6993                 :           0 :         appendStringInfo(buf, "INSERT INTO %s",
    6994                 :           0 :                                          generate_relation_name(rte->relid, NIL));
    6995                 :             : 
    6996                 :             :         /* Print the relation alias, if needed; INSERT requires explicit AS */
    6997                 :           0 :         get_rte_alias(rte, query->resultRelation, true, context);
    6998                 :             : 
    6999                 :             :         /* always want a space here */
    7000                 :           0 :         appendStringInfoChar(buf, ' ');
    7001                 :             : 
    7002                 :             :         /*
    7003                 :             :          * Add the insert-column-names list.  Any indirection decoration needed on
    7004                 :             :          * the column names can be inferred from the top targetlist.
    7005                 :             :          */
    7006                 :           0 :         strippedexprs = NIL;
    7007                 :           0 :         sep = "";
    7008         [ #  # ]:           0 :         if (query->targetList)
    7009                 :           0 :                 appendStringInfoChar(buf, '(');
    7010   [ #  #  #  #  :           0 :         foreach(l, query->targetList)
                   #  # ]
    7011                 :             :         {
    7012                 :           0 :                 TargetEntry *tle = (TargetEntry *) lfirst(l);
    7013                 :             : 
    7014         [ #  # ]:           0 :                 if (tle->resjunk)
    7015                 :           0 :                         continue;                       /* ignore junk entries */
    7016                 :             : 
    7017                 :           0 :                 appendStringInfoString(buf, sep);
    7018                 :           0 :                 sep = ", ";
    7019                 :             : 
    7020                 :             :                 /*
    7021                 :             :                  * Put out name of target column; look in the catalogs, not at
    7022                 :             :                  * tle->resname, since resname will fail to track RENAME.
    7023                 :             :                  */
    7024                 :           0 :                 appendStringInfoString(buf,
    7025                 :           0 :                                                            quote_identifier(get_attname(rte->relid,
    7026                 :           0 :                                                                                                                         tle->resno,
    7027                 :             :                                                                                                                         false)));
    7028                 :             : 
    7029                 :             :                 /*
    7030                 :             :                  * Print any indirection needed (subfields or subscripts), and strip
    7031                 :             :                  * off the top-level nodes representing the indirection assignments.
    7032                 :             :                  * Add the stripped expressions to strippedexprs.  (If it's a
    7033                 :             :                  * single-VALUES statement, the stripped expressions are the VALUES to
    7034                 :             :                  * print below.  Otherwise they're just Vars and not really
    7035                 :             :                  * interesting.)
    7036                 :             :                  */
    7037                 :           0 :                 strippedexprs = lappend(strippedexprs,
    7038                 :           0 :                                                                 processIndirection((Node *) tle->expr,
    7039                 :           0 :                                                                                                    context));
    7040      [ #  #  # ]:           0 :         }
    7041         [ #  # ]:           0 :         if (query->targetList)
    7042                 :           0 :                 appendStringInfoString(buf, ") ");
    7043                 :             : 
    7044         [ #  # ]:           0 :         if (query->override)
    7045                 :             :         {
    7046         [ #  # ]:           0 :                 if (query->override == OVERRIDING_SYSTEM_VALUE)
    7047                 :           0 :                         appendStringInfoString(buf, "OVERRIDING SYSTEM VALUE ");
    7048         [ #  # ]:           0 :                 else if (query->override == OVERRIDING_USER_VALUE)
    7049                 :           0 :                         appendStringInfoString(buf, "OVERRIDING USER VALUE ");
    7050                 :           0 :         }
    7051                 :             : 
    7052         [ #  # ]:           0 :         if (select_rte)
    7053                 :             :         {
    7054                 :             :                 /* Add the SELECT */
    7055                 :           0 :                 get_query_def(select_rte->subquery, buf, context->namespaces, NULL,
    7056                 :             :                                           false,
    7057                 :           0 :                                           context->prettyFlags, context->wrapColumn,
    7058                 :           0 :                                           context->indentLevel);
    7059                 :           0 :         }
    7060         [ #  # ]:           0 :         else if (values_rte)
    7061                 :             :         {
    7062                 :             :                 /* Add the multi-VALUES expression lists */
    7063                 :           0 :                 get_values_def(values_rte->values_lists, context);
    7064                 :           0 :         }
    7065         [ #  # ]:           0 :         else if (strippedexprs)
    7066                 :             :         {
    7067                 :             :                 /* Add the single-VALUES expression list */
    7068                 :           0 :                 appendContextKeyword(context, "VALUES (",
    7069                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 2);
    7070                 :           0 :                 get_rule_list_toplevel(strippedexprs, context, false);
    7071                 :           0 :                 appendStringInfoChar(buf, ')');
    7072                 :           0 :         }
    7073                 :             :         else
    7074                 :             :         {
    7075                 :             :                 /* No expressions, so it must be DEFAULT VALUES */
    7076                 :           0 :                 appendStringInfoString(buf, "DEFAULT VALUES");
    7077                 :             :         }
    7078                 :             : 
    7079                 :             :         /* Add ON CONFLICT if present */
    7080         [ #  # ]:           0 :         if (query->onConflict)
    7081                 :             :         {
    7082                 :           0 :                 OnConflictExpr *confl = query->onConflict;
    7083                 :             : 
    7084                 :           0 :                 appendStringInfoString(buf, " ON CONFLICT");
    7085                 :             : 
    7086         [ #  # ]:           0 :                 if (confl->arbiterElems)
    7087                 :             :                 {
    7088                 :             :                         /* Add the single-VALUES expression list */
    7089                 :           0 :                         appendStringInfoChar(buf, '(');
    7090                 :           0 :                         get_rule_expr((Node *) confl->arbiterElems, context, false);
    7091                 :           0 :                         appendStringInfoChar(buf, ')');
    7092                 :             : 
    7093                 :             :                         /* Add a WHERE clause (for partial indexes) if given */
    7094         [ #  # ]:           0 :                         if (confl->arbiterWhere != NULL)
    7095                 :             :                         {
    7096                 :           0 :                                 bool            save_varprefix;
    7097                 :             : 
    7098                 :             :                                 /*
    7099                 :             :                                  * Force non-prefixing of Vars, since parser assumes that they
    7100                 :             :                                  * belong to target relation.  WHERE clause does not use
    7101                 :             :                                  * InferenceElem, so this is separately required.
    7102                 :             :                                  */
    7103                 :           0 :                                 save_varprefix = context->varprefix;
    7104                 :           0 :                                 context->varprefix = false;
    7105                 :             : 
    7106                 :           0 :                                 appendContextKeyword(context, " WHERE ",
    7107                 :             :                                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 1);
    7108                 :           0 :                                 get_rule_expr(confl->arbiterWhere, context, false);
    7109                 :             : 
    7110                 :           0 :                                 context->varprefix = save_varprefix;
    7111                 :           0 :                         }
    7112                 :           0 :                 }
    7113         [ #  # ]:           0 :                 else if (OidIsValid(confl->constraint))
    7114                 :             :                 {
    7115                 :           0 :                         char       *constraint = get_constraint_name(confl->constraint);
    7116                 :             : 
    7117         [ #  # ]:           0 :                         if (!constraint)
    7118   [ #  #  #  # ]:           0 :                                 elog(ERROR, "cache lookup failed for constraint %u",
    7119                 :             :                                          confl->constraint);
    7120                 :           0 :                         appendStringInfo(buf, " ON CONSTRAINT %s",
    7121                 :           0 :                                                          quote_identifier(constraint));
    7122                 :           0 :                 }
    7123                 :             : 
    7124         [ #  # ]:           0 :                 if (confl->action == ONCONFLICT_NOTHING)
    7125                 :             :                 {
    7126                 :           0 :                         appendStringInfoString(buf, " DO NOTHING");
    7127                 :           0 :                 }
    7128                 :             :                 else
    7129                 :             :                 {
    7130                 :           0 :                         appendStringInfoString(buf, " DO UPDATE SET ");
    7131                 :             :                         /* Deparse targetlist */
    7132                 :           0 :                         get_update_query_targetlist_def(query, confl->onConflictSet,
    7133                 :           0 :                                                                                         context, rte);
    7134                 :             : 
    7135                 :             :                         /* Add a WHERE clause if given */
    7136         [ #  # ]:           0 :                         if (confl->onConflictWhere != NULL)
    7137                 :             :                         {
    7138                 :           0 :                                 appendContextKeyword(context, " WHERE ",
    7139                 :             :                                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 1);
    7140                 :           0 :                                 get_rule_expr(confl->onConflictWhere, context, false);
    7141                 :           0 :                         }
    7142                 :             :                 }
    7143                 :           0 :         }
    7144                 :             : 
    7145                 :             :         /* Add RETURNING if present */
    7146         [ #  # ]:           0 :         if (query->returningList)
    7147                 :           0 :                 get_returning_clause(query, context);
    7148                 :           0 : }
    7149                 :             : 
    7150                 :             : 
    7151                 :             : /* ----------
    7152                 :             :  * get_update_query_def                 - Parse back an UPDATE parsetree
    7153                 :             :  * ----------
    7154                 :             :  */
    7155                 :             : static void
    7156                 :           0 : get_update_query_def(Query *query, deparse_context *context)
    7157                 :             : {
    7158                 :           0 :         StringInfo      buf = context->buf;
    7159                 :           0 :         RangeTblEntry *rte;
    7160                 :             : 
    7161                 :             :         /* Insert the WITH clause if given */
    7162                 :           0 :         get_with_clause(query, context);
    7163                 :             : 
    7164                 :             :         /*
    7165                 :             :          * Start the query with UPDATE relname SET
    7166                 :             :          */
    7167                 :           0 :         rte = rt_fetch(query->resultRelation, query->rtable);
    7168         [ #  # ]:           0 :         Assert(rte->rtekind == RTE_RELATION);
    7169         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
    7170                 :             :         {
    7171                 :           0 :                 appendStringInfoChar(buf, ' ');
    7172                 :           0 :                 context->indentLevel += PRETTYINDENT_STD;
    7173                 :           0 :         }
    7174                 :           0 :         appendStringInfo(buf, "UPDATE %s%s",
    7175                 :           0 :                                          only_marker(rte),
    7176                 :           0 :                                          generate_relation_name(rte->relid, NIL));
    7177                 :             : 
    7178                 :             :         /* Print the relation alias, if needed */
    7179                 :           0 :         get_rte_alias(rte, query->resultRelation, false, context);
    7180                 :             : 
    7181                 :           0 :         appendStringInfoString(buf, " SET ");
    7182                 :             : 
    7183                 :             :         /* Deparse targetlist */
    7184                 :           0 :         get_update_query_targetlist_def(query, query->targetList, context, rte);
    7185                 :             : 
    7186                 :             :         /* Add the FROM clause if needed */
    7187                 :           0 :         get_from_clause(query, " FROM ", context);
    7188                 :             : 
    7189                 :             :         /* Add a WHERE clause if given */
    7190         [ #  # ]:           0 :         if (query->jointree->quals != NULL)
    7191                 :             :         {
    7192                 :           0 :                 appendContextKeyword(context, " WHERE ",
    7193                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 1);
    7194                 :           0 :                 get_rule_expr(query->jointree->quals, context, false);
    7195                 :           0 :         }
    7196                 :             : 
    7197                 :             :         /* Add RETURNING if present */
    7198         [ #  # ]:           0 :         if (query->returningList)
    7199                 :           0 :                 get_returning_clause(query, context);
    7200                 :           0 : }
    7201                 :             : 
    7202                 :             : 
    7203                 :             : /* ----------
    7204                 :             :  * get_update_query_targetlist_def                      - Parse back an UPDATE targetlist
    7205                 :             :  * ----------
    7206                 :             :  */
    7207                 :             : static void
    7208                 :           0 : get_update_query_targetlist_def(Query *query, List *targetList,
    7209                 :             :                                                                 deparse_context *context, RangeTblEntry *rte)
    7210                 :             : {
    7211                 :           0 :         StringInfo      buf = context->buf;
    7212                 :           0 :         ListCell   *l;
    7213                 :           0 :         ListCell   *next_ma_cell;
    7214                 :           0 :         int                     remaining_ma_columns;
    7215                 :           0 :         const char *sep;
    7216                 :           0 :         SubLink    *cur_ma_sublink;
    7217                 :           0 :         List       *ma_sublinks;
    7218                 :             : 
    7219                 :             :         /*
    7220                 :             :          * Prepare to deal with MULTIEXPR assignments: collect the source SubLinks
    7221                 :             :          * into a list.  We expect them to appear, in ID order, in resjunk tlist
    7222                 :             :          * entries.
    7223                 :             :          */
    7224                 :           0 :         ma_sublinks = NIL;
    7225         [ #  # ]:           0 :         if (query->hasSubLinks)              /* else there can't be any */
    7226                 :             :         {
    7227   [ #  #  #  #  :           0 :                 foreach(l, targetList)
                   #  # ]
    7228                 :             :                 {
    7229                 :           0 :                         TargetEntry *tle = (TargetEntry *) lfirst(l);
    7230                 :             : 
    7231   [ #  #  #  # ]:           0 :                         if (tle->resjunk && IsA(tle->expr, SubLink))
    7232                 :             :                         {
    7233                 :           0 :                                 SubLink    *sl = (SubLink *) tle->expr;
    7234                 :             : 
    7235         [ #  # ]:           0 :                                 if (sl->subLinkType == MULTIEXPR_SUBLINK)
    7236                 :             :                                 {
    7237                 :           0 :                                         ma_sublinks = lappend(ma_sublinks, sl);
    7238         [ #  # ]:           0 :                                         Assert(sl->subLinkId == list_length(ma_sublinks));
    7239                 :           0 :                                 }
    7240                 :           0 :                         }
    7241                 :           0 :                 }
    7242                 :           0 :         }
    7243                 :           0 :         next_ma_cell = list_head(ma_sublinks);
    7244                 :           0 :         cur_ma_sublink = NULL;
    7245                 :           0 :         remaining_ma_columns = 0;
    7246                 :             : 
    7247                 :             :         /* Add the comma separated list of 'attname = value' */
    7248                 :           0 :         sep = "";
    7249   [ #  #  #  #  :           0 :         foreach(l, targetList)
                   #  # ]
    7250                 :             :         {
    7251                 :           0 :                 TargetEntry *tle = (TargetEntry *) lfirst(l);
    7252                 :           0 :                 Node       *expr;
    7253                 :             : 
    7254         [ #  # ]:           0 :                 if (tle->resjunk)
    7255                 :           0 :                         continue;                       /* ignore junk entries */
    7256                 :             : 
    7257                 :             :                 /* Emit separator (OK whether we're in multiassignment or not) */
    7258                 :           0 :                 appendStringInfoString(buf, sep);
    7259                 :           0 :                 sep = ", ";
    7260                 :             : 
    7261                 :             :                 /*
    7262                 :             :                  * Check to see if we're starting a multiassignment group: if so,
    7263                 :             :                  * output a left paren.
    7264                 :             :                  */
    7265   [ #  #  #  # ]:           0 :                 if (next_ma_cell != NULL && cur_ma_sublink == NULL)
    7266                 :             :                 {
    7267                 :             :                         /*
    7268                 :             :                          * We must dig down into the expr to see if it's a PARAM_MULTIEXPR
    7269                 :             :                          * Param.  That could be buried under FieldStores and
    7270                 :             :                          * SubscriptingRefs and CoerceToDomains (cf processIndirection()),
    7271                 :             :                          * and underneath those there could be an implicit type coercion.
    7272                 :             :                          * Because we would ignore implicit type coercions anyway, we
    7273                 :             :                          * don't need to be as careful as processIndirection() is about
    7274                 :             :                          * descending past implicit CoerceToDomains.
    7275                 :             :                          */
    7276                 :           0 :                         expr = (Node *) tle->expr;
    7277         [ #  # ]:           0 :                         while (expr)
    7278                 :             :                         {
    7279         [ #  # ]:           0 :                                 if (IsA(expr, FieldStore))
    7280                 :             :                                 {
    7281                 :           0 :                                         FieldStore *fstore = (FieldStore *) expr;
    7282                 :             : 
    7283                 :           0 :                                         expr = (Node *) linitial(fstore->newvals);
    7284                 :           0 :                                 }
    7285         [ #  # ]:           0 :                                 else if (IsA(expr, SubscriptingRef))
    7286                 :             :                                 {
    7287                 :           0 :                                         SubscriptingRef *sbsref = (SubscriptingRef *) expr;
    7288                 :             : 
    7289         [ #  # ]:           0 :                                         if (sbsref->refassgnexpr == NULL)
    7290                 :           0 :                                                 break;
    7291                 :             : 
    7292                 :           0 :                                         expr = (Node *) sbsref->refassgnexpr;
    7293         [ #  # ]:           0 :                                 }
    7294         [ #  # ]:           0 :                                 else if (IsA(expr, CoerceToDomain))
    7295                 :             :                                 {
    7296                 :           0 :                                         CoerceToDomain *cdomain = (CoerceToDomain *) expr;
    7297                 :             : 
    7298         [ #  # ]:           0 :                                         if (cdomain->coercionformat != COERCE_IMPLICIT_CAST)
    7299                 :           0 :                                                 break;
    7300                 :           0 :                                         expr = (Node *) cdomain->arg;
    7301         [ #  # ]:           0 :                                 }
    7302                 :             :                                 else
    7303                 :           0 :                                         break;
    7304                 :             :                         }
    7305                 :           0 :                         expr = strip_implicit_coercions(expr);
    7306                 :             : 
    7307   [ #  #  #  #  :           0 :                         if (expr && IsA(expr, Param) &&
                   #  # ]
    7308                 :           0 :                                 ((Param *) expr)->paramkind == PARAM_MULTIEXPR)
    7309                 :             :                         {
    7310                 :           0 :                                 cur_ma_sublink = (SubLink *) lfirst(next_ma_cell);
    7311                 :           0 :                                 next_ma_cell = lnext(ma_sublinks, next_ma_cell);
    7312                 :           0 :                                 remaining_ma_columns = count_nonjunk_tlist_entries(((Query *) cur_ma_sublink->subselect)->targetList);
    7313         [ #  # ]:           0 :                                 Assert(((Param *) expr)->paramid ==
    7314                 :             :                                            ((cur_ma_sublink->subLinkId << 16) | 1));
    7315                 :           0 :                                 appendStringInfoChar(buf, '(');
    7316                 :           0 :                         }
    7317                 :           0 :                 }
    7318                 :             : 
    7319                 :             :                 /*
    7320                 :             :                  * Put out name of target column; look in the catalogs, not at
    7321                 :             :                  * tle->resname, since resname will fail to track RENAME.
    7322                 :             :                  */
    7323                 :           0 :                 appendStringInfoString(buf,
    7324                 :           0 :                                                            quote_identifier(get_attname(rte->relid,
    7325                 :           0 :                                                                                                                         tle->resno,
    7326                 :             :                                                                                                                         false)));
    7327                 :             : 
    7328                 :             :                 /*
    7329                 :             :                  * Print any indirection needed (subfields or subscripts), and strip
    7330                 :             :                  * off the top-level nodes representing the indirection assignments.
    7331                 :             :                  */
    7332                 :           0 :                 expr = processIndirection((Node *) tle->expr, context);
    7333                 :             : 
    7334                 :             :                 /*
    7335                 :             :                  * If we're in a multiassignment, skip printing anything more, unless
    7336                 :             :                  * this is the last column; in which case, what we print should be the
    7337                 :             :                  * sublink, not the Param.
    7338                 :             :                  */
    7339         [ #  # ]:           0 :                 if (cur_ma_sublink != NULL)
    7340                 :             :                 {
    7341         [ #  # ]:           0 :                         if (--remaining_ma_columns > 0)
    7342                 :           0 :                                 continue;               /* not the last column of multiassignment */
    7343                 :           0 :                         appendStringInfoChar(buf, ')');
    7344                 :           0 :                         expr = (Node *) cur_ma_sublink;
    7345                 :           0 :                         cur_ma_sublink = NULL;
    7346                 :           0 :                 }
    7347                 :             : 
    7348                 :           0 :                 appendStringInfoString(buf, " = ");
    7349                 :             : 
    7350                 :           0 :                 get_rule_expr(expr, context, false);
    7351         [ #  # ]:           0 :         }
    7352                 :           0 : }
    7353                 :             : 
    7354                 :             : 
    7355                 :             : /* ----------
    7356                 :             :  * get_delete_query_def                 - Parse back a DELETE parsetree
    7357                 :             :  * ----------
    7358                 :             :  */
    7359                 :             : static void
    7360                 :           0 : get_delete_query_def(Query *query, deparse_context *context)
    7361                 :             : {
    7362                 :           0 :         StringInfo      buf = context->buf;
    7363                 :           0 :         RangeTblEntry *rte;
    7364                 :             : 
    7365                 :             :         /* Insert the WITH clause if given */
    7366                 :           0 :         get_with_clause(query, context);
    7367                 :             : 
    7368                 :             :         /*
    7369                 :             :          * Start the query with DELETE FROM relname
    7370                 :             :          */
    7371                 :           0 :         rte = rt_fetch(query->resultRelation, query->rtable);
    7372         [ #  # ]:           0 :         Assert(rte->rtekind == RTE_RELATION);
    7373         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
    7374                 :             :         {
    7375                 :           0 :                 appendStringInfoChar(buf, ' ');
    7376                 :           0 :                 context->indentLevel += PRETTYINDENT_STD;
    7377                 :           0 :         }
    7378                 :           0 :         appendStringInfo(buf, "DELETE FROM %s%s",
    7379                 :           0 :                                          only_marker(rte),
    7380                 :           0 :                                          generate_relation_name(rte->relid, NIL));
    7381                 :             : 
    7382                 :             :         /* Print the relation alias, if needed */
    7383                 :           0 :         get_rte_alias(rte, query->resultRelation, false, context);
    7384                 :             : 
    7385                 :             :         /* Add the USING clause if given */
    7386                 :           0 :         get_from_clause(query, " USING ", context);
    7387                 :             : 
    7388                 :             :         /* Add a WHERE clause if given */
    7389         [ #  # ]:           0 :         if (query->jointree->quals != NULL)
    7390                 :             :         {
    7391                 :           0 :                 appendContextKeyword(context, " WHERE ",
    7392                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 1);
    7393                 :           0 :                 get_rule_expr(query->jointree->quals, context, false);
    7394                 :           0 :         }
    7395                 :             : 
    7396                 :             :         /* Add RETURNING if present */
    7397         [ #  # ]:           0 :         if (query->returningList)
    7398                 :           0 :                 get_returning_clause(query, context);
    7399                 :           0 : }
    7400                 :             : 
    7401                 :             : 
    7402                 :             : /* ----------
    7403                 :             :  * get_merge_query_def                          - Parse back a MERGE parsetree
    7404                 :             :  * ----------
    7405                 :             :  */
    7406                 :             : static void
    7407                 :           0 : get_merge_query_def(Query *query, deparse_context *context)
    7408                 :             : {
    7409                 :           0 :         StringInfo      buf = context->buf;
    7410                 :           0 :         RangeTblEntry *rte;
    7411                 :           0 :         ListCell   *lc;
    7412                 :           0 :         bool            haveNotMatchedBySource;
    7413                 :             : 
    7414                 :             :         /* Insert the WITH clause if given */
    7415                 :           0 :         get_with_clause(query, context);
    7416                 :             : 
    7417                 :             :         /*
    7418                 :             :          * Start the query with MERGE INTO relname
    7419                 :             :          */
    7420                 :           0 :         rte = rt_fetch(query->resultRelation, query->rtable);
    7421         [ #  # ]:           0 :         Assert(rte->rtekind == RTE_RELATION);
    7422         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
    7423                 :             :         {
    7424                 :           0 :                 appendStringInfoChar(buf, ' ');
    7425                 :           0 :                 context->indentLevel += PRETTYINDENT_STD;
    7426                 :           0 :         }
    7427                 :           0 :         appendStringInfo(buf, "MERGE INTO %s%s",
    7428                 :           0 :                                          only_marker(rte),
    7429                 :           0 :                                          generate_relation_name(rte->relid, NIL));
    7430                 :             : 
    7431                 :             :         /* Print the relation alias, if needed */
    7432                 :           0 :         get_rte_alias(rte, query->resultRelation, false, context);
    7433                 :             : 
    7434                 :             :         /* Print the source relation and join clause */
    7435                 :           0 :         get_from_clause(query, " USING ", context);
    7436                 :           0 :         appendContextKeyword(context, " ON ",
    7437                 :             :                                                  -PRETTYINDENT_STD, PRETTYINDENT_STD, 2);
    7438                 :           0 :         get_rule_expr(query->mergeJoinCondition, context, false);
    7439                 :             : 
    7440                 :             :         /*
    7441                 :             :          * Test for any NOT MATCHED BY SOURCE actions.  If there are none, then
    7442                 :             :          * any NOT MATCHED BY TARGET actions are output as "WHEN NOT MATCHED", per
    7443                 :             :          * SQL standard.  Otherwise, we have a non-SQL-standard query, so output
    7444                 :             :          * "BY SOURCE" / "BY TARGET" qualifiers for all NOT MATCHED actions, to be
    7445                 :             :          * more explicit.
    7446                 :             :          */
    7447                 :           0 :         haveNotMatchedBySource = false;
    7448   [ #  #  #  #  :           0 :         foreach(lc, query->mergeActionList)
                   #  # ]
    7449                 :             :         {
    7450                 :           0 :                 MergeAction *action = lfirst_node(MergeAction, lc);
    7451                 :             : 
    7452         [ #  # ]:           0 :                 if (action->matchKind == MERGE_WHEN_NOT_MATCHED_BY_SOURCE)
    7453                 :             :                 {
    7454                 :           0 :                         haveNotMatchedBySource = true;
    7455                 :           0 :                         break;
    7456                 :             :                 }
    7457         [ #  # ]:           0 :         }
    7458                 :             : 
    7459                 :             :         /* Print each merge action */
    7460   [ #  #  #  #  :           0 :         foreach(lc, query->mergeActionList)
                   #  # ]
    7461                 :             :         {
    7462                 :           0 :                 MergeAction *action = lfirst_node(MergeAction, lc);
    7463                 :             : 
    7464                 :           0 :                 appendContextKeyword(context, " WHEN ",
    7465                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 2);
    7466   [ #  #  #  # ]:           0 :                 switch (action->matchKind)
    7467                 :             :                 {
    7468                 :             :                         case MERGE_WHEN_MATCHED:
    7469                 :           0 :                                 appendStringInfoString(buf, "MATCHED");
    7470                 :           0 :                                 break;
    7471                 :             :                         case MERGE_WHEN_NOT_MATCHED_BY_SOURCE:
    7472                 :           0 :                                 appendStringInfoString(buf, "NOT MATCHED BY SOURCE");
    7473                 :           0 :                                 break;
    7474                 :             :                         case MERGE_WHEN_NOT_MATCHED_BY_TARGET:
    7475         [ #  # ]:           0 :                                 if (haveNotMatchedBySource)
    7476                 :           0 :                                         appendStringInfoString(buf, "NOT MATCHED BY TARGET");
    7477                 :             :                                 else
    7478                 :           0 :                                         appendStringInfoString(buf, "NOT MATCHED");
    7479                 :           0 :                                 break;
    7480                 :             :                         default:
    7481   [ #  #  #  # ]:           0 :                                 elog(ERROR, "unrecognized matchKind: %d",
    7482                 :             :                                          (int) action->matchKind);
    7483                 :           0 :                 }
    7484                 :             : 
    7485         [ #  # ]:           0 :                 if (action->qual)
    7486                 :             :                 {
    7487                 :           0 :                         appendContextKeyword(context, " AND ",
    7488                 :             :                                                                  -PRETTYINDENT_STD, PRETTYINDENT_STD, 3);
    7489                 :           0 :                         get_rule_expr(action->qual, context, false);
    7490                 :           0 :                 }
    7491                 :           0 :                 appendContextKeyword(context, " THEN ",
    7492                 :             :                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 3);
    7493                 :             : 
    7494         [ #  # ]:           0 :                 if (action->commandType == CMD_INSERT)
    7495                 :             :                 {
    7496                 :             :                         /* This generally matches get_insert_query_def() */
    7497                 :           0 :                         List       *strippedexprs = NIL;
    7498                 :           0 :                         const char *sep = "";
    7499                 :           0 :                         ListCell   *lc2;
    7500                 :             : 
    7501                 :           0 :                         appendStringInfoString(buf, "INSERT");
    7502                 :             : 
    7503         [ #  # ]:           0 :                         if (action->targetList)
    7504                 :           0 :                                 appendStringInfoString(buf, " (");
    7505   [ #  #  #  #  :           0 :                         foreach(lc2, action->targetList)
                   #  # ]
    7506                 :             :                         {
    7507                 :           0 :                                 TargetEntry *tle = (TargetEntry *) lfirst(lc2);
    7508                 :             : 
    7509         [ #  # ]:           0 :                                 Assert(!tle->resjunk);
    7510                 :             : 
    7511                 :           0 :                                 appendStringInfoString(buf, sep);
    7512                 :           0 :                                 sep = ", ";
    7513                 :             : 
    7514                 :           0 :                                 appendStringInfoString(buf,
    7515                 :           0 :                                                                            quote_identifier(get_attname(rte->relid,
    7516                 :           0 :                                                                                                                                         tle->resno,
    7517                 :             :                                                                                                                                         false)));
    7518                 :           0 :                                 strippedexprs = lappend(strippedexprs,
    7519                 :           0 :                                                                                 processIndirection((Node *) tle->expr,
    7520                 :           0 :                                                                                                                    context));
    7521                 :           0 :                         }
    7522         [ #  # ]:           0 :                         if (action->targetList)
    7523                 :           0 :                                 appendStringInfoChar(buf, ')');
    7524                 :             : 
    7525         [ #  # ]:           0 :                         if (action->override)
    7526                 :             :                         {
    7527         [ #  # ]:           0 :                                 if (action->override == OVERRIDING_SYSTEM_VALUE)
    7528                 :           0 :                                         appendStringInfoString(buf, " OVERRIDING SYSTEM VALUE");
    7529         [ #  # ]:           0 :                                 else if (action->override == OVERRIDING_USER_VALUE)
    7530                 :           0 :                                         appendStringInfoString(buf, " OVERRIDING USER VALUE");
    7531                 :           0 :                         }
    7532                 :             : 
    7533         [ #  # ]:           0 :                         if (strippedexprs)
    7534                 :             :                         {
    7535                 :           0 :                                 appendContextKeyword(context, " VALUES (",
    7536                 :             :                                                                          -PRETTYINDENT_STD, PRETTYINDENT_STD, 4);
    7537                 :           0 :                                 get_rule_list_toplevel(strippedexprs, context, false);
    7538                 :           0 :                                 appendStringInfoChar(buf, ')');
    7539                 :           0 :                         }
    7540                 :             :                         else
    7541                 :           0 :                                 appendStringInfoString(buf, " DEFAULT VALUES");
    7542                 :           0 :                 }
    7543         [ #  # ]:           0 :                 else if (action->commandType == CMD_UPDATE)
    7544                 :             :                 {
    7545                 :           0 :                         appendStringInfoString(buf, "UPDATE SET ");
    7546                 :           0 :                         get_update_query_targetlist_def(query, action->targetList,
    7547                 :           0 :                                                                                         context, rte);
    7548                 :           0 :                 }
    7549         [ #  # ]:           0 :                 else if (action->commandType == CMD_DELETE)
    7550                 :           0 :                         appendStringInfoString(buf, "DELETE");
    7551         [ #  # ]:           0 :                 else if (action->commandType == CMD_NOTHING)
    7552                 :           0 :                         appendStringInfoString(buf, "DO NOTHING");
    7553                 :           0 :         }
    7554                 :             : 
    7555                 :             :         /* Add RETURNING if present */
    7556         [ #  # ]:           0 :         if (query->returningList)
    7557                 :           0 :                 get_returning_clause(query, context);
    7558                 :           0 : }
    7559                 :             : 
    7560                 :             : 
    7561                 :             : /* ----------
    7562                 :             :  * get_utility_query_def                        - Parse back a UTILITY parsetree
    7563                 :             :  * ----------
    7564                 :             :  */
    7565                 :             : static void
    7566                 :           0 : get_utility_query_def(Query *query, deparse_context *context)
    7567                 :             : {
    7568                 :           0 :         StringInfo      buf = context->buf;
    7569                 :             : 
    7570         [ #  # ]:           0 :         if (query->utilityStmt && IsA(query->utilityStmt, NotifyStmt))
    7571                 :             :         {
    7572                 :           0 :                 NotifyStmt *stmt = (NotifyStmt *) query->utilityStmt;
    7573                 :             : 
    7574                 :           0 :                 appendContextKeyword(context, "",
    7575                 :             :                                                          0, PRETTYINDENT_STD, 1);
    7576                 :           0 :                 appendStringInfo(buf, "NOTIFY %s",
    7577                 :           0 :                                                  quote_identifier(stmt->conditionname));
    7578         [ #  # ]:           0 :                 if (stmt->payload)
    7579                 :             :                 {
    7580                 :           0 :                         appendStringInfoString(buf, ", ");
    7581                 :           0 :                         simple_quote_literal(buf, stmt->payload);
    7582                 :           0 :                 }
    7583                 :           0 :         }
    7584                 :             :         else
    7585                 :             :         {
    7586                 :             :                 /* Currently only NOTIFY utility commands can appear in rules */
    7587   [ #  #  #  # ]:           0 :                 elog(ERROR, "unexpected utility statement type");
    7588                 :             :         }
    7589                 :           0 : }
    7590                 :             : 
    7591                 :             : /*
    7592                 :             :  * Display a Var appropriately.
    7593                 :             :  *
    7594                 :             :  * In some cases (currently only when recursing into an unnamed join)
    7595                 :             :  * the Var's varlevelsup has to be interpreted with respect to a context
    7596                 :             :  * above the current one; levelsup indicates the offset.
    7597                 :             :  *
    7598                 :             :  * If istoplevel is true, the Var is at the top level of a SELECT's
    7599                 :             :  * targetlist, which means we need special treatment of whole-row Vars.
    7600                 :             :  * Instead of the normal "tab.*", we'll print "tab.*::typename", which is a
    7601                 :             :  * dirty hack to prevent "tab.*" from being expanded into multiple columns.
    7602                 :             :  * (The parser will strip the useless coercion, so no inefficiency is added in
    7603                 :             :  * dump and reload.)  We used to print just "tab" in such cases, but that is
    7604                 :             :  * ambiguous and will yield the wrong result if "tab" is also a plain column
    7605                 :             :  * name in the query.
    7606                 :             :  *
    7607                 :             :  * Returns the attname of the Var, or NULL if the Var has no attname (because
    7608                 :             :  * it is a whole-row Var or a subplan output reference).
    7609                 :             :  */
    7610                 :             : static char *
    7611                 :           0 : get_variable(Var *var, int levelsup, bool istoplevel, deparse_context *context)
    7612                 :             : {
    7613                 :           0 :         StringInfo      buf = context->buf;
    7614                 :           0 :         RangeTblEntry *rte;
    7615                 :           0 :         AttrNumber      attnum;
    7616                 :           0 :         int                     netlevelsup;
    7617                 :           0 :         deparse_namespace *dpns;
    7618                 :           0 :         int                     varno;
    7619                 :           0 :         AttrNumber      varattno;
    7620                 :           0 :         deparse_columns *colinfo;
    7621                 :           0 :         char       *refname;
    7622                 :           0 :         char       *attname;
    7623                 :           0 :         bool            need_prefix;
    7624                 :             : 
    7625                 :             :         /* Find appropriate nesting depth */
    7626                 :           0 :         netlevelsup = var->varlevelsup + levelsup;
    7627         [ #  # ]:           0 :         if (netlevelsup >= list_length(context->namespaces))
    7628   [ #  #  #  # ]:           0 :                 elog(ERROR, "bogus varlevelsup: %d offset %d",
    7629                 :             :                          var->varlevelsup, levelsup);
    7630                 :           0 :         dpns = (deparse_namespace *) list_nth(context->namespaces,
    7631                 :           0 :                                                                                   netlevelsup);
    7632                 :             : 
    7633                 :             :         /*
    7634                 :             :          * If we have a syntactic referent for the Var, and we're working from a
    7635                 :             :          * parse tree, prefer to use the syntactic referent.  Otherwise, fall back
    7636                 :             :          * on the semantic referent.  (Forcing use of the semantic referent when
    7637                 :             :          * printing plan trees is a design choice that's perhaps more motivated by
    7638                 :             :          * backwards compatibility than anything else.  But it does have the
    7639                 :             :          * advantage of making plans more explicit.)
    7640                 :             :          */
    7641   [ #  #  #  # ]:           0 :         if (var->varnosyn > 0 && dpns->plan == NULL)
    7642                 :             :         {
    7643                 :           0 :                 varno = var->varnosyn;
    7644                 :           0 :                 varattno = var->varattnosyn;
    7645                 :           0 :         }
    7646                 :             :         else
    7647                 :             :         {
    7648                 :           0 :                 varno = var->varno;
    7649                 :           0 :                 varattno = var->varattno;
    7650                 :             :         }
    7651                 :             : 
    7652                 :             :         /*
    7653                 :             :          * Try to find the relevant RTE in this rtable.  In a plan tree, it's
    7654                 :             :          * likely that varno is OUTER_VAR or INNER_VAR, in which case we must dig
    7655                 :             :          * down into the subplans, or INDEX_VAR, which is resolved similarly. Also
    7656                 :             :          * find the aliases previously assigned for this RTE.
    7657                 :             :          */
    7658   [ #  #  #  # ]:           0 :         if (varno >= 1 && varno <= list_length(dpns->rtable))
    7659                 :             :         {
    7660                 :             :                 /*
    7661                 :             :                  * We might have been asked to map child Vars to some parent relation.
    7662                 :             :                  */
    7663   [ #  #  #  # ]:           0 :                 if (context->appendparents && dpns->appendrels)
    7664                 :             :                 {
    7665                 :           0 :                         int                     pvarno = varno;
    7666                 :           0 :                         AttrNumber      pvarattno = varattno;
    7667                 :           0 :                         AppendRelInfo *appinfo = dpns->appendrels[pvarno];
    7668                 :           0 :                         bool            found = false;
    7669                 :             : 
    7670                 :             :                         /* Only map up to inheritance parents, not UNION ALL appendrels */
    7671   [ #  #  #  # ]:           0 :                         while (appinfo &&
    7672                 :           0 :                                    rt_fetch(appinfo->parent_relid,
    7673                 :           0 :                                                         dpns->rtable)->rtekind == RTE_RELATION)
    7674                 :             :                         {
    7675                 :           0 :                                 found = false;
    7676         [ #  # ]:           0 :                                 if (pvarattno > 0)   /* system columns stay as-is */
    7677                 :             :                                 {
    7678         [ #  # ]:           0 :                                         if (pvarattno > appinfo->num_child_cols)
    7679                 :           0 :                                                 break;  /* safety check */
    7680                 :           0 :                                         pvarattno = appinfo->parent_colnos[pvarattno - 1];
    7681         [ #  # ]:           0 :                                         if (pvarattno == 0)
    7682                 :           0 :                                                 break;  /* Var is local to child */
    7683                 :           0 :                                 }
    7684                 :             : 
    7685                 :           0 :                                 pvarno = appinfo->parent_relid;
    7686                 :           0 :                                 found = true;
    7687                 :             : 
    7688                 :             :                                 /* If the parent is itself a child, continue up. */
    7689         [ #  # ]:           0 :                                 Assert(pvarno > 0 && pvarno <= list_length(dpns->rtable));
    7690                 :           0 :                                 appinfo = dpns->appendrels[pvarno];
    7691                 :             :                         }
    7692                 :             : 
    7693                 :             :                         /*
    7694                 :             :                          * If we found an ancestral rel, and that rel is included in
    7695                 :             :                          * appendparents, print that column not the original one.
    7696                 :             :                          */
    7697   [ #  #  #  # ]:           0 :                         if (found && bms_is_member(pvarno, context->appendparents))
    7698                 :             :                         {
    7699                 :           0 :                                 varno = pvarno;
    7700                 :           0 :                                 varattno = pvarattno;
    7701                 :           0 :                         }
    7702                 :           0 :                 }
    7703                 :             : 
    7704                 :           0 :                 rte = rt_fetch(varno, dpns->rtable);
    7705                 :             : 
    7706                 :             :                 /* might be returning old/new column value */
    7707         [ #  # ]:           0 :                 if (var->varreturningtype == VAR_RETURNING_OLD)
    7708                 :           0 :                         refname = dpns->ret_old_alias;
    7709         [ #  # ]:           0 :                 else if (var->varreturningtype == VAR_RETURNING_NEW)
    7710                 :           0 :                         refname = dpns->ret_new_alias;
    7711                 :             :                 else
    7712                 :           0 :                         refname = (char *) list_nth(dpns->rtable_names, varno - 1);
    7713                 :             : 
    7714                 :           0 :                 colinfo = deparse_columns_fetch(varno, dpns);
    7715                 :           0 :                 attnum = varattno;
    7716                 :           0 :         }
    7717                 :             :         else
    7718                 :             :         {
    7719                 :           0 :                 resolve_special_varno((Node *) var, context,
    7720                 :             :                                                           get_special_variable, NULL);
    7721                 :           0 :                 return NULL;
    7722                 :             :         }
    7723                 :             : 
    7724                 :             :         /*
    7725                 :             :          * The planner will sometimes emit Vars referencing resjunk elements of a
    7726                 :             :          * subquery's target list (this is currently only possible if it chooses
    7727                 :             :          * to generate a "physical tlist" for a SubqueryScan or CteScan node).
    7728                 :             :          * Although we prefer to print subquery-referencing Vars using the
    7729                 :             :          * subquery's alias, that's not possible for resjunk items since they have
    7730                 :             :          * no alias.  So in that case, drill down to the subplan and print the
    7731                 :             :          * contents of the referenced tlist item.  This works because in a plan
    7732                 :             :          * tree, such Vars can only occur in a SubqueryScan or CteScan node, and
    7733                 :             :          * we'll have set dpns->inner_plan to reference the child plan node.
    7734                 :             :          */
    7735         [ #  # ]:           0 :         if ((rte->rtekind == RTE_SUBQUERY || rte->rtekind == RTE_CTE) &&
    7736   [ #  #  #  # ]:           0 :                 attnum > list_length(rte->eref->colnames) &&
    7737                 :           0 :                 dpns->inner_plan)
    7738                 :             :         {
    7739                 :           0 :                 TargetEntry *tle;
    7740                 :           0 :                 deparse_namespace save_dpns;
    7741                 :             : 
    7742                 :           0 :                 tle = get_tle_by_resno(dpns->inner_tlist, attnum);
    7743         [ #  # ]:           0 :                 if (!tle)
    7744   [ #  #  #  # ]:           0 :                         elog(ERROR, "invalid attnum %d for relation \"%s\"",
    7745                 :             :                                  attnum, rte->eref->aliasname);
    7746                 :             : 
    7747         [ #  # ]:           0 :                 Assert(netlevelsup == 0);
    7748                 :           0 :                 push_child_plan(dpns, dpns->inner_plan, &save_dpns);
    7749                 :             : 
    7750                 :             :                 /*
    7751                 :             :                  * Force parentheses because our caller probably assumed a Var is a
    7752                 :             :                  * simple expression.
    7753                 :             :                  */
    7754         [ #  # ]:           0 :                 if (!IsA(tle->expr, Var))
    7755                 :           0 :                         appendStringInfoChar(buf, '(');
    7756                 :           0 :                 get_rule_expr((Node *) tle->expr, context, true);
    7757         [ #  # ]:           0 :                 if (!IsA(tle->expr, Var))
    7758                 :           0 :                         appendStringInfoChar(buf, ')');
    7759                 :             : 
    7760                 :           0 :                 pop_child_plan(dpns, &save_dpns);
    7761                 :           0 :                 return NULL;
    7762                 :           0 :         }
    7763                 :             : 
    7764                 :             :         /*
    7765                 :             :          * If it's an unnamed join, look at the expansion of the alias variable.
    7766                 :             :          * If it's a simple reference to one of the input vars, then recursively
    7767                 :             :          * print the name of that var instead.  When it's not a simple reference,
    7768                 :             :          * we have to just print the unqualified join column name.  (This can only
    7769                 :             :          * happen with "dangerous" merged columns in a JOIN USING; we took pains
    7770                 :             :          * previously to make the unqualified column name unique in such cases.)
    7771                 :             :          *
    7772                 :             :          * This wouldn't work in decompiling plan trees, because we don't store
    7773                 :             :          * joinaliasvars lists after planning; but a plan tree should never
    7774                 :             :          * contain a join alias variable.
    7775                 :             :          */
    7776   [ #  #  #  # ]:           0 :         if (rte->rtekind == RTE_JOIN && rte->alias == NULL)
    7777                 :             :         {
    7778         [ #  # ]:           0 :                 if (rte->joinaliasvars == NIL)
    7779   [ #  #  #  # ]:           0 :                         elog(ERROR, "cannot decompile join alias var in plan tree");
    7780         [ #  # ]:           0 :                 if (attnum > 0)
    7781                 :             :                 {
    7782                 :           0 :                         Var                *aliasvar;
    7783                 :             : 
    7784                 :           0 :                         aliasvar = (Var *) list_nth(rte->joinaliasvars, attnum - 1);
    7785                 :             :                         /* we intentionally don't strip implicit coercions here */
    7786   [ #  #  #  # ]:           0 :                         if (aliasvar && IsA(aliasvar, Var))
    7787                 :             :                         {
    7788                 :           0 :                                 return get_variable(aliasvar, var->varlevelsup + levelsup,
    7789                 :           0 :                                                                         istoplevel, context);
    7790                 :             :                         }
    7791         [ #  # ]:           0 :                 }
    7792                 :             : 
    7793                 :             :                 /*
    7794                 :             :                  * Unnamed join has no refname.  (Note: since it's unnamed, there is
    7795                 :             :                  * no way the user could have referenced it to create a whole-row Var
    7796                 :             :                  * for it.  So we don't have to cover that case below.)
    7797                 :             :                  */
    7798         [ #  # ]:           0 :                 Assert(refname == NULL);
    7799                 :           0 :         }
    7800                 :             : 
    7801         [ #  # ]:           0 :         if (attnum == InvalidAttrNumber)
    7802                 :           0 :                 attname = NULL;
    7803         [ #  # ]:           0 :         else if (attnum > 0)
    7804                 :             :         {
    7805                 :             :                 /* Get column name to use from the colinfo struct */
    7806         [ #  # ]:           0 :                 if (attnum > colinfo->num_cols)
    7807   [ #  #  #  # ]:           0 :                         elog(ERROR, "invalid attnum %d for relation \"%s\"",
    7808                 :             :                                  attnum, rte->eref->aliasname);
    7809                 :           0 :                 attname = colinfo->colnames[attnum - 1];
    7810                 :             : 
    7811                 :             :                 /*
    7812                 :             :                  * If we find a Var referencing a dropped column, it seems better to
    7813                 :             :                  * print something (anything) than to fail.  In general this should
    7814                 :             :                  * not happen, but it used to be possible for some cases involving
    7815                 :             :                  * functions returning named composite types, and perhaps there are
    7816                 :             :                  * still bugs out there.
    7817                 :             :                  */
    7818         [ #  # ]:           0 :                 if (attname == NULL)
    7819                 :           0 :                         attname = "?dropped?column?";
    7820                 :           0 :         }
    7821                 :             :         else
    7822                 :             :         {
    7823                 :             :                 /* System column - name is fixed, get it from the catalog */
    7824                 :           0 :                 attname = get_rte_attribute_name(rte, attnum);
    7825                 :             :         }
    7826                 :             : 
    7827   [ #  #  #  # ]:           0 :         need_prefix = (context->varprefix || attname == NULL ||
    7828                 :           0 :                                    var->varreturningtype != VAR_RETURNING_DEFAULT);
    7829                 :             : 
    7830                 :             :         /*
    7831                 :             :          * If we're considering a plain Var in an ORDER BY (but not GROUP BY)
    7832                 :             :          * clause, we may need to add a table-name prefix to prevent
    7833                 :             :          * findTargetlistEntrySQL92 from misinterpreting the name as an
    7834                 :             :          * output-column name.  To avoid cluttering the output with unnecessary
    7835                 :             :          * prefixes, do so only if there is a name match to a SELECT tlist item
    7836                 :             :          * that is different from the Var.
    7837                 :             :          */
    7838   [ #  #  #  #  :           0 :         if (context->varInOrderBy && !context->inGroupBy && !need_prefix)
                   #  # ]
    7839                 :             :         {
    7840                 :           0 :                 int                     colno = 0;
    7841                 :             : 
    7842   [ #  #  #  #  :           0 :                 foreach_node(TargetEntry, tle, context->targetList)
             #  #  #  # ]
    7843                 :             :                 {
    7844                 :           0 :                         char       *colname;
    7845                 :             : 
    7846         [ #  # ]:           0 :                         if (tle->resjunk)
    7847                 :           0 :                                 continue;               /* ignore junk entries */
    7848                 :           0 :                         colno++;
    7849                 :             : 
    7850                 :             :                         /* This must match colname-choosing logic in get_target_list() */
    7851   [ #  #  #  # ]:           0 :                         if (context->resultDesc && colno <= context->resultDesc->natts)
    7852                 :           0 :                                 colname = NameStr(TupleDescAttr(context->resultDesc,
    7853                 :             :                                                                                                 colno - 1)->attname);
    7854                 :             :                         else
    7855                 :           0 :                                 colname = tle->resname;
    7856                 :             : 
    7857   [ #  #  #  #  :           0 :                         if (colname && strcmp(colname, attname) == 0 &&
                   #  # ]
    7858                 :           0 :                                 !equal(var, tle->expr))
    7859                 :             :                         {
    7860                 :           0 :                                 need_prefix = true;
    7861                 :           0 :                                 break;
    7862                 :             :                         }
    7863      [ #  #  # ]:           0 :                 }
    7864                 :           0 :         }
    7865                 :             : 
    7866   [ #  #  #  # ]:           0 :         if (refname && need_prefix)
    7867                 :             :         {
    7868                 :           0 :                 appendStringInfoString(buf, quote_identifier(refname));
    7869                 :           0 :                 appendStringInfoChar(buf, '.');
    7870                 :           0 :         }
    7871         [ #  # ]:           0 :         if (attname)
    7872                 :           0 :                 appendStringInfoString(buf, quote_identifier(attname));
    7873                 :             :         else
    7874                 :             :         {
    7875                 :           0 :                 appendStringInfoChar(buf, '*');
    7876         [ #  # ]:           0 :                 if (istoplevel)
    7877                 :           0 :                         appendStringInfo(buf, "::%s",
    7878                 :           0 :                                                          format_type_with_typemod(var->vartype,
    7879                 :           0 :                                                                                                           var->vartypmod));
    7880                 :             :         }
    7881                 :             : 
    7882                 :           0 :         return attname;
    7883                 :           0 : }
    7884                 :             : 
    7885                 :             : /*
    7886                 :             :  * Deparse a Var which references OUTER_VAR, INNER_VAR, or INDEX_VAR.  This
    7887                 :             :  * routine is actually a callback for resolve_special_varno, which handles
    7888                 :             :  * finding the correct TargetEntry.  We get the expression contained in that
    7889                 :             :  * TargetEntry and just need to deparse it, a job we can throw back on
    7890                 :             :  * get_rule_expr.
    7891                 :             :  */
    7892                 :             : static void
    7893                 :           0 : get_special_variable(Node *node, deparse_context *context, void *callback_arg)
    7894                 :             : {
    7895                 :           0 :         StringInfo      buf = context->buf;
    7896                 :             : 
    7897                 :             :         /*
    7898                 :             :          * For a non-Var referent, force parentheses because our caller probably
    7899                 :             :          * assumed a Var is a simple expression.
    7900                 :             :          */
    7901         [ #  # ]:           0 :         if (!IsA(node, Var))
    7902                 :           0 :                 appendStringInfoChar(buf, '(');
    7903                 :           0 :         get_rule_expr(node, context, true);
    7904         [ #  # ]:           0 :         if (!IsA(node, Var))
    7905                 :           0 :                 appendStringInfoChar(buf, ')');
    7906                 :           0 : }
    7907                 :             : 
    7908                 :             : /*
    7909                 :             :  * Chase through plan references to special varnos (OUTER_VAR, INNER_VAR,
    7910                 :             :  * INDEX_VAR) until we find a real Var or some kind of non-Var node; then,
    7911                 :             :  * invoke the callback provided.
    7912                 :             :  */
    7913                 :             : static void
    7914                 :           0 : resolve_special_varno(Node *node, deparse_context *context,
    7915                 :             :                                           rsv_callback callback, void *callback_arg)
    7916                 :             : {
    7917                 :           0 :         Var                *var;
    7918                 :           0 :         deparse_namespace *dpns;
    7919                 :             : 
    7920                 :             :         /* This function is recursive, so let's be paranoid. */
    7921                 :           0 :         check_stack_depth();
    7922                 :             : 
    7923                 :             :         /* If it's not a Var, invoke the callback. */
    7924         [ #  # ]:           0 :         if (!IsA(node, Var))
    7925                 :             :         {
    7926                 :           0 :                 (*callback) (node, context, callback_arg);
    7927                 :           0 :                 return;
    7928                 :             :         }
    7929                 :             : 
    7930                 :             :         /* Find appropriate nesting depth */
    7931                 :           0 :         var = (Var *) node;
    7932                 :           0 :         dpns = (deparse_namespace *) list_nth(context->namespaces,
    7933                 :           0 :                                                                                   var->varlevelsup);
    7934                 :             : 
    7935                 :             :         /*
    7936                 :             :          * If varno is special, recurse.  (Don't worry about varnosyn; if we're
    7937                 :             :          * here, we already decided not to use that.)
    7938                 :             :          */
    7939   [ #  #  #  # ]:           0 :         if (var->varno == OUTER_VAR && dpns->outer_tlist)
    7940                 :             :         {
    7941                 :           0 :                 TargetEntry *tle;
    7942                 :           0 :                 deparse_namespace save_dpns;
    7943                 :           0 :                 Bitmapset  *save_appendparents;
    7944                 :             : 
    7945                 :           0 :                 tle = get_tle_by_resno(dpns->outer_tlist, var->varattno);
    7946         [ #  # ]:           0 :                 if (!tle)
    7947   [ #  #  #  # ]:           0 :                         elog(ERROR, "bogus varattno for OUTER_VAR var: %d", var->varattno);
    7948                 :             : 
    7949                 :             :                 /*
    7950                 :             :                  * If we're descending to the first child of an Append or MergeAppend,
    7951                 :             :                  * update appendparents.  This will affect deparsing of all Vars
    7952                 :             :                  * appearing within the eventually-resolved subexpression.
    7953                 :             :                  */
    7954                 :           0 :                 save_appendparents = context->appendparents;
    7955                 :             : 
    7956         [ #  # ]:           0 :                 if (IsA(dpns->plan, Append))
    7957                 :           0 :                         context->appendparents = bms_union(context->appendparents,
    7958                 :           0 :                                                                                            ((Append *) dpns->plan)->apprelids);
    7959         [ #  # ]:           0 :                 else if (IsA(dpns->plan, MergeAppend))
    7960                 :           0 :                         context->appendparents = bms_union(context->appendparents,
    7961                 :           0 :                                                                                            ((MergeAppend *) dpns->plan)->apprelids);
    7962                 :             : 
    7963                 :           0 :                 push_child_plan(dpns, dpns->outer_plan, &save_dpns);
    7964                 :           0 :                 resolve_special_varno((Node *) tle->expr, context,
    7965                 :           0 :                                                           callback, callback_arg);
    7966                 :           0 :                 pop_child_plan(dpns, &save_dpns);
    7967                 :           0 :                 context->appendparents = save_appendparents;
    7968                 :             :                 return;
    7969                 :           0 :         }
    7970   [ #  #  #  # ]:           0 :         else if (var->varno == INNER_VAR && dpns->inner_tlist)
    7971                 :             :         {
    7972                 :           0 :                 TargetEntry *tle;
    7973                 :           0 :                 deparse_namespace save_dpns;
    7974                 :             : 
    7975                 :           0 :                 tle = get_tle_by_resno(dpns->inner_tlist, var->varattno);
    7976         [ #  # ]:           0 :                 if (!tle)
    7977   [ #  #  #  # ]:           0 :                         elog(ERROR, "bogus varattno for INNER_VAR var: %d", var->varattno);
    7978                 :             : 
    7979                 :           0 :                 push_child_plan(dpns, dpns->inner_plan, &save_dpns);
    7980                 :           0 :                 resolve_special_varno((Node *) tle->expr, context,
    7981                 :           0 :                                                           callback, callback_arg);
    7982                 :           0 :                 pop_child_plan(dpns, &save_dpns);
    7983                 :             :                 return;
    7984                 :           0 :         }
    7985   [ #  #  #  # ]:           0 :         else if (var->varno == INDEX_VAR && dpns->index_tlist)
    7986                 :             :         {
    7987                 :           0 :                 TargetEntry *tle;
    7988                 :             : 
    7989                 :           0 :                 tle = get_tle_by_resno(dpns->index_tlist, var->varattno);
    7990         [ #  # ]:           0 :                 if (!tle)
    7991   [ #  #  #  # ]:           0 :                         elog(ERROR, "bogus varattno for INDEX_VAR var: %d", var->varattno);
    7992                 :             : 
    7993                 :           0 :                 resolve_special_varno((Node *) tle->expr, context,
    7994                 :           0 :                                                           callback, callback_arg);
    7995                 :             :                 return;
    7996                 :           0 :         }
    7997         [ #  # ]:           0 :         else if (var->varno < 1 || var->varno > list_length(dpns->rtable))
    7998   [ #  #  #  # ]:           0 :                 elog(ERROR, "bogus varno: %d", var->varno);
    7999                 :             : 
    8000                 :             :         /* Not special.  Just invoke the callback. */
    8001                 :           0 :         (*callback) (node, context, callback_arg);
    8002         [ #  # ]:           0 : }
    8003                 :             : 
    8004                 :             : /*
    8005                 :             :  * Get the name of a field of an expression of composite type.  The
    8006                 :             :  * expression is usually a Var, but we handle other cases too.
    8007                 :             :  *
    8008                 :             :  * levelsup is an extra offset to interpret the Var's varlevelsup correctly.
    8009                 :             :  *
    8010                 :             :  * This is fairly straightforward when the expression has a named composite
    8011                 :             :  * type; we need only look up the type in the catalogs.  However, the type
    8012                 :             :  * could also be RECORD.  Since no actual table or view column is allowed to
    8013                 :             :  * have type RECORD, a Var of type RECORD must refer to a JOIN or FUNCTION RTE
    8014                 :             :  * or to a subquery output.  We drill down to find the ultimate defining
    8015                 :             :  * expression and attempt to infer the field name from it.  We ereport if we
    8016                 :             :  * can't determine the name.
    8017                 :             :  *
    8018                 :             :  * Similarly, a PARAM of type RECORD has to refer to some expression of
    8019                 :             :  * a determinable composite type.
    8020                 :             :  */
    8021                 :             : static const char *
    8022                 :           0 : get_name_for_var_field(Var *var, int fieldno,
    8023                 :             :                                            int levelsup, deparse_context *context)
    8024                 :             : {
    8025                 :           0 :         RangeTblEntry *rte;
    8026                 :           0 :         AttrNumber      attnum;
    8027                 :           0 :         int                     netlevelsup;
    8028                 :           0 :         deparse_namespace *dpns;
    8029                 :           0 :         int                     varno;
    8030                 :           0 :         AttrNumber      varattno;
    8031                 :           0 :         TupleDesc       tupleDesc;
    8032                 :           0 :         Node       *expr;
    8033                 :             : 
    8034                 :             :         /*
    8035                 :             :          * If it's a RowExpr that was expanded from a whole-row Var, use the
    8036                 :             :          * column names attached to it.  (We could let get_expr_result_tupdesc()
    8037                 :             :          * handle this, but it's much cheaper to just pull out the name we need.)
    8038                 :             :          */
    8039         [ #  # ]:           0 :         if (IsA(var, RowExpr))
    8040                 :             :         {
    8041                 :           0 :                 RowExpr    *r = (RowExpr *) var;
    8042                 :             : 
    8043   [ #  #  #  # ]:           0 :                 if (fieldno > 0 && fieldno <= list_length(r->colnames))
    8044                 :           0 :                         return strVal(list_nth(r->colnames, fieldno - 1));
    8045         [ #  # ]:           0 :         }
    8046                 :             : 
    8047                 :             :         /*
    8048                 :             :          * If it's a Param of type RECORD, try to find what the Param refers to.
    8049                 :             :          */
    8050         [ #  # ]:           0 :         if (IsA(var, Param))
    8051                 :             :         {
    8052                 :           0 :                 Param      *param = (Param *) var;
    8053                 :           0 :                 ListCell   *ancestor_cell;
    8054                 :             : 
    8055                 :           0 :                 expr = find_param_referent(param, context, &dpns, &ancestor_cell);
    8056         [ #  # ]:           0 :                 if (expr)
    8057                 :             :                 {
    8058                 :             :                         /* Found a match, so recurse to decipher the field name */
    8059                 :           0 :                         deparse_namespace save_dpns;
    8060                 :           0 :                         const char *result;
    8061                 :             : 
    8062                 :           0 :                         push_ancestor_plan(dpns, ancestor_cell, &save_dpns);
    8063                 :           0 :                         result = get_name_for_var_field((Var *) expr, fieldno,
    8064                 :           0 :                                                                                         0, context);
    8065                 :           0 :                         pop_ancestor_plan(dpns, &save_dpns);
    8066                 :           0 :                         return result;
    8067                 :           0 :                 }
    8068         [ #  # ]:           0 :         }
    8069                 :             : 
    8070                 :             :         /*
    8071                 :             :          * If it's a Var of type RECORD, we have to find what the Var refers to;
    8072                 :             :          * if not, we can use get_expr_result_tupdesc().
    8073                 :             :          */
    8074   [ #  #  #  # ]:           0 :         if (!IsA(var, Var) ||
    8075                 :           0 :                 var->vartype != RECORDOID)
    8076                 :             :         {
    8077                 :           0 :                 tupleDesc = get_expr_result_tupdesc((Node *) var, false);
    8078                 :             :                 /* Got the tupdesc, so we can extract the field name */
    8079         [ #  # ]:           0 :                 Assert(fieldno >= 1 && fieldno <= tupleDesc->natts);
    8080                 :           0 :                 return NameStr(TupleDescAttr(tupleDesc, fieldno - 1)->attname);
    8081                 :             :         }
    8082                 :             : 
    8083                 :             :         /* Find appropriate nesting depth */
    8084                 :           0 :         netlevelsup = var->varlevelsup + levelsup;
    8085         [ #  # ]:           0 :         if (netlevelsup >= list_length(context->namespaces))
    8086   [ #  #  #  # ]:           0 :                 elog(ERROR, "bogus varlevelsup: %d offset %d",
    8087                 :             :                          var->varlevelsup, levelsup);
    8088                 :           0 :         dpns = (deparse_namespace *) list_nth(context->namespaces,
    8089                 :           0 :                                                                                   netlevelsup);
    8090                 :             : 
    8091                 :             :         /*
    8092                 :             :          * If we have a syntactic referent for the Var, and we're working from a
    8093                 :             :          * parse tree, prefer to use the syntactic referent.  Otherwise, fall back
    8094                 :             :          * on the semantic referent.  (See comments in get_variable().)
    8095                 :             :          */
    8096   [ #  #  #  # ]:           0 :         if (var->varnosyn > 0 && dpns->plan == NULL)
    8097                 :             :         {
    8098                 :           0 :                 varno = var->varnosyn;
    8099                 :           0 :                 varattno = var->varattnosyn;
    8100                 :           0 :         }
    8101                 :             :         else
    8102                 :             :         {
    8103                 :           0 :                 varno = var->varno;
    8104                 :           0 :                 varattno = var->varattno;
    8105                 :             :         }
    8106                 :             : 
    8107                 :             :         /*
    8108                 :             :          * Try to find the relevant RTE in this rtable.  In a plan tree, it's
    8109                 :             :          * likely that varno is OUTER_VAR or INNER_VAR, in which case we must dig
    8110                 :             :          * down into the subplans, or INDEX_VAR, which is resolved similarly.
    8111                 :             :          *
    8112                 :             :          * Note: unlike get_variable and resolve_special_varno, we need not worry
    8113                 :             :          * about inheritance mapping: a child Var should have the same datatype as
    8114                 :             :          * its parent, and here we're really only interested in the Var's type.
    8115                 :             :          */
    8116   [ #  #  #  # ]:           0 :         if (varno >= 1 && varno <= list_length(dpns->rtable))
    8117                 :             :         {
    8118                 :           0 :                 rte = rt_fetch(varno, dpns->rtable);
    8119                 :           0 :                 attnum = varattno;
    8120                 :           0 :         }
    8121   [ #  #  #  # ]:           0 :         else if (varno == OUTER_VAR && dpns->outer_tlist)
    8122                 :             :         {
    8123                 :           0 :                 TargetEntry *tle;
    8124                 :           0 :                 deparse_namespace save_dpns;
    8125                 :           0 :                 const char *result;
    8126                 :             : 
    8127                 :           0 :                 tle = get_tle_by_resno(dpns->outer_tlist, varattno);
    8128         [ #  # ]:           0 :                 if (!tle)
    8129   [ #  #  #  # ]:           0 :                         elog(ERROR, "bogus varattno for OUTER_VAR var: %d", varattno);
    8130                 :             : 
    8131         [ #  # ]:           0 :                 Assert(netlevelsup == 0);
    8132                 :           0 :                 push_child_plan(dpns, dpns->outer_plan, &save_dpns);
    8133                 :             : 
    8134                 :           0 :                 result = get_name_for_var_field((Var *) tle->expr, fieldno,
    8135                 :           0 :                                                                                 levelsup, context);
    8136                 :             : 
    8137                 :           0 :                 pop_child_plan(dpns, &save_dpns);
    8138                 :           0 :                 return result;
    8139                 :           0 :         }
    8140   [ #  #  #  # ]:           0 :         else if (varno == INNER_VAR && dpns->inner_tlist)
    8141                 :             :         {
    8142                 :           0 :                 TargetEntry *tle;
    8143                 :           0 :                 deparse_namespace save_dpns;
    8144                 :           0 :                 const char *result;
    8145                 :             : 
    8146                 :           0 :                 tle = get_tle_by_resno(dpns->inner_tlist, varattno);
    8147         [ #  # ]:           0 :                 if (!tle)
    8148   [ #  #  #  # ]:           0 :                         elog(ERROR, "bogus varattno for INNER_VAR var: %d", varattno);
    8149                 :             : 
    8150         [ #  # ]:           0 :                 Assert(netlevelsup == 0);
    8151                 :           0 :                 push_child_plan(dpns, dpns->inner_plan, &save_dpns);
    8152                 :             : 
    8153                 :           0 :                 result = get_name_for_var_field((Var *) tle->expr, fieldno,
    8154                 :           0 :                                                                                 levelsup, context);
    8155                 :             : 
    8156                 :           0 :                 pop_child_plan(dpns, &save_dpns);
    8157                 :           0 :                 return result;
    8158                 :           0 :         }
    8159         [ #  # ]:           0 :         else if (varno == INDEX_VAR && dpns->index_tlist)
    8160                 :             :         {
    8161                 :           0 :                 TargetEntry *tle;
    8162                 :           0 :                 const char *result;
    8163                 :             : 
    8164                 :           0 :                 tle = get_tle_by_resno(dpns->index_tlist, varattno);
    8165         [ #  # ]:           0 :                 if (!tle)
    8166   [ #  #  #  # ]:           0 :                         elog(ERROR, "bogus varattno for INDEX_VAR var: %d", varattno);
    8167                 :             : 
    8168         [ #  # ]:           0 :                 Assert(netlevelsup == 0);
    8169                 :             : 
    8170                 :           0 :                 result = get_name_for_var_field((Var *) tle->expr, fieldno,
    8171                 :           0 :                                                                                 levelsup, context);
    8172                 :             : 
    8173                 :           0 :                 return result;
    8174                 :           0 :         }
    8175                 :             :         else
    8176                 :             :         {
    8177   [ #  #  #  # ]:           0 :                 elog(ERROR, "bogus varno: %d", varno);
    8178                 :           0 :                 return NULL;                    /* keep compiler quiet */
    8179                 :             :         }
    8180                 :             : 
    8181         [ #  # ]:           0 :         if (attnum == InvalidAttrNumber)
    8182                 :             :         {
    8183                 :             :                 /* Var is whole-row reference to RTE, so select the right field */
    8184                 :           0 :                 return get_rte_attribute_name(rte, fieldno);
    8185                 :             :         }
    8186                 :             : 
    8187                 :             :         /*
    8188                 :             :          * This part has essentially the same logic as the parser's
    8189                 :             :          * expandRecordVariable() function, but we are dealing with a different
    8190                 :             :          * representation of the input context, and we only need one field name
    8191                 :             :          * not a TupleDesc.  Also, we need special cases for finding subquery and
    8192                 :             :          * CTE subplans when deparsing Plan trees.
    8193                 :             :          */
    8194                 :           0 :         expr = (Node *) var;            /* default if we can't drill down */
    8195                 :             : 
    8196   [ #  #  #  #  :           0 :         switch (rte->rtekind)
                   #  # ]
    8197                 :             :         {
    8198                 :             :                 case RTE_RELATION:
    8199                 :             :                 case RTE_VALUES:
    8200                 :             :                 case RTE_NAMEDTUPLESTORE:
    8201                 :             :                 case RTE_RESULT:
    8202                 :             : 
    8203                 :             :                         /*
    8204                 :             :                          * This case should not occur: a column of a table, values list,
    8205                 :             :                          * or ENR shouldn't have type RECORD.  Fall through and fail (most
    8206                 :             :                          * likely) at the bottom.
    8207                 :             :                          */
    8208                 :           0 :                         break;
    8209                 :             :                 case RTE_SUBQUERY:
    8210                 :             :                         /* Subselect-in-FROM: examine sub-select's output expr */
    8211                 :             :                         {
    8212         [ #  # ]:           0 :                                 if (rte->subquery)
    8213                 :             :                                 {
    8214                 :           0 :                                         TargetEntry *ste = get_tle_by_resno(rte->subquery->targetList,
    8215                 :           0 :                                                                                                                 attnum);
    8216                 :             : 
    8217         [ #  # ]:           0 :                                         if (ste == NULL || ste->resjunk)
    8218   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "subquery %s does not have attribute %d",
    8219                 :             :                                                          rte->eref->aliasname, attnum);
    8220                 :           0 :                                         expr = (Node *) ste->expr;
    8221         [ #  # ]:           0 :                                         if (IsA(expr, Var))
    8222                 :             :                                         {
    8223                 :             :                                                 /*
    8224                 :             :                                                  * Recurse into the sub-select to see what its Var
    8225                 :             :                                                  * refers to. We have to build an additional level of
    8226                 :             :                                                  * namespace to keep in step with varlevelsup in the
    8227                 :             :                                                  * subselect; furthermore, the subquery RTE might be
    8228                 :             :                                                  * from an outer query level, in which case the
    8229                 :             :                                                  * namespace for the subselect must have that outer
    8230                 :             :                                                  * level as parent namespace.
    8231                 :             :                                                  */
    8232                 :           0 :                                                 List       *save_nslist = context->namespaces;
    8233                 :           0 :                                                 List       *parent_namespaces;
    8234                 :           0 :                                                 deparse_namespace mydpns;
    8235                 :           0 :                                                 const char *result;
    8236                 :             : 
    8237                 :           0 :                                                 parent_namespaces = list_copy_tail(context->namespaces,
    8238                 :           0 :                                                                                                                    netlevelsup);
    8239                 :             : 
    8240                 :           0 :                                                 set_deparse_for_query(&mydpns, rte->subquery,
    8241                 :           0 :                                                                                           parent_namespaces);
    8242                 :             : 
    8243                 :           0 :                                                 context->namespaces = lcons(&mydpns, parent_namespaces);
    8244                 :             : 
    8245                 :           0 :                                                 result = get_name_for_var_field((Var *) expr, fieldno,
    8246                 :           0 :                                                                                                                 0, context);
    8247                 :             : 
    8248                 :           0 :                                                 context->namespaces = save_nslist;
    8249                 :             : 
    8250                 :           0 :                                                 return result;
    8251                 :           0 :                                         }
    8252                 :             :                                         /* else fall through to inspect the expression */
    8253         [ #  # ]:           0 :                                 }
    8254                 :             :                                 else
    8255                 :             :                                 {
    8256                 :             :                                         /*
    8257                 :             :                                          * We're deparsing a Plan tree so we don't have complete
    8258                 :             :                                          * RTE entries (in particular, rte->subquery is NULL). But
    8259                 :             :                                          * the only place we'd normally see a Var directly
    8260                 :             :                                          * referencing a SUBQUERY RTE is in a SubqueryScan plan
    8261                 :             :                                          * node, and we can look into the child plan's tlist
    8262                 :             :                                          * instead.  An exception occurs if the subquery was
    8263                 :             :                                          * proven empty and optimized away: then we'd find such a
    8264                 :             :                                          * Var in a childless Result node, and there's nothing in
    8265                 :             :                                          * the plan tree that would let us figure out what it had
    8266                 :             :                                          * originally referenced.  In that case, fall back on
    8267                 :             :                                          * printing "fN", analogously to the default column names
    8268                 :             :                                          * for RowExprs.
    8269                 :             :                                          */
    8270                 :           0 :                                         TargetEntry *tle;
    8271                 :           0 :                                         deparse_namespace save_dpns;
    8272                 :           0 :                                         const char *result;
    8273                 :             : 
    8274         [ #  # ]:           0 :                                         if (!dpns->inner_plan)
    8275                 :             :                                         {
    8276                 :           0 :                                                 char       *dummy_name = palloc(32);
    8277                 :             : 
    8278         [ #  # ]:           0 :                                                 Assert(dpns->plan && IsA(dpns->plan, Result));
    8279                 :           0 :                                                 snprintf(dummy_name, 32, "f%d", fieldno);
    8280                 :           0 :                                                 return dummy_name;
    8281                 :           0 :                                         }
    8282         [ #  # ]:           0 :                                         Assert(dpns->plan && IsA(dpns->plan, SubqueryScan));
    8283                 :             : 
    8284                 :           0 :                                         tle = get_tle_by_resno(dpns->inner_tlist, attnum);
    8285         [ #  # ]:           0 :                                         if (!tle)
    8286   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "bogus varattno for subquery var: %d",
    8287                 :             :                                                          attnum);
    8288         [ #  # ]:           0 :                                         Assert(netlevelsup == 0);
    8289                 :           0 :                                         push_child_plan(dpns, dpns->inner_plan, &save_dpns);
    8290                 :             : 
    8291                 :           0 :                                         result = get_name_for_var_field((Var *) tle->expr, fieldno,
    8292                 :           0 :                                                                                                         levelsup, context);
    8293                 :             : 
    8294                 :           0 :                                         pop_child_plan(dpns, &save_dpns);
    8295                 :           0 :                                         return result;
    8296                 :           0 :                                 }
    8297                 :             :                         }
    8298                 :           0 :                         break;
    8299                 :             :                 case RTE_JOIN:
    8300                 :             :                         /* Join RTE --- recursively inspect the alias variable */
    8301         [ #  # ]:           0 :                         if (rte->joinaliasvars == NIL)
    8302   [ #  #  #  # ]:           0 :                                 elog(ERROR, "cannot decompile join alias var in plan tree");
    8303         [ #  # ]:           0 :                         Assert(attnum > 0 && attnum <= list_length(rte->joinaliasvars));
    8304                 :           0 :                         expr = (Node *) list_nth(rte->joinaliasvars, attnum - 1);
    8305         [ #  # ]:           0 :                         Assert(expr != NULL);
    8306                 :             :                         /* we intentionally don't strip implicit coercions here */
    8307         [ #  # ]:           0 :                         if (IsA(expr, Var))
    8308                 :           0 :                                 return get_name_for_var_field((Var *) expr, fieldno,
    8309                 :           0 :                                                                                           var->varlevelsup + levelsup,
    8310                 :           0 :                                                                                           context);
    8311                 :             :                         /* else fall through to inspect the expression */
    8312                 :           0 :                         break;
    8313                 :             :                 case RTE_FUNCTION:
    8314                 :             :                 case RTE_TABLEFUNC:
    8315                 :             : 
    8316                 :             :                         /*
    8317                 :             :                          * We couldn't get here unless a function is declared with one of
    8318                 :             :                          * its result columns as RECORD, which is not allowed.
    8319                 :             :                          */
    8320                 :           0 :                         break;
    8321                 :             :                 case RTE_CTE:
    8322                 :             :                         /* CTE reference: examine subquery's output expr */
    8323                 :             :                         {
    8324                 :           0 :                                 CommonTableExpr *cte = NULL;
    8325                 :           0 :                                 Index           ctelevelsup;
    8326                 :           0 :                                 ListCell   *lc;
    8327                 :             : 
    8328                 :             :                                 /*
    8329                 :             :                                  * Try to find the referenced CTE using the namespace stack.
    8330                 :             :                                  */
    8331                 :           0 :                                 ctelevelsup = rte->ctelevelsup + netlevelsup;
    8332         [ #  # ]:           0 :                                 if (ctelevelsup >= list_length(context->namespaces))
    8333                 :           0 :                                         lc = NULL;
    8334                 :             :                                 else
    8335                 :             :                                 {
    8336                 :           0 :                                         deparse_namespace *ctedpns;
    8337                 :             : 
    8338                 :           0 :                                         ctedpns = (deparse_namespace *)
    8339                 :           0 :                                                 list_nth(context->namespaces, ctelevelsup);
    8340   [ #  #  #  #  :           0 :                                         foreach(lc, ctedpns->ctes)
                   #  # ]
    8341                 :             :                                         {
    8342                 :           0 :                                                 cte = (CommonTableExpr *) lfirst(lc);
    8343         [ #  # ]:           0 :                                                 if (strcmp(cte->ctename, rte->ctename) == 0)
    8344                 :           0 :                                                         break;
    8345                 :           0 :                                         }
    8346                 :           0 :                                 }
    8347         [ #  # ]:           0 :                                 if (lc != NULL)
    8348                 :             :                                 {
    8349                 :           0 :                                         Query      *ctequery = (Query *) cte->ctequery;
    8350   [ #  #  #  # ]:           0 :                                         TargetEntry *ste = get_tle_by_resno(GetCTETargetList(cte),
    8351                 :           0 :                                                                                                                 attnum);
    8352                 :             : 
    8353         [ #  # ]:           0 :                                         if (ste == NULL || ste->resjunk)
    8354   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "CTE %s does not have attribute %d",
    8355                 :             :                                                          rte->eref->aliasname, attnum);
    8356                 :           0 :                                         expr = (Node *) ste->expr;
    8357         [ #  # ]:           0 :                                         if (IsA(expr, Var))
    8358                 :             :                                         {
    8359                 :             :                                                 /*
    8360                 :             :                                                  * Recurse into the CTE to see what its Var refers to.
    8361                 :             :                                                  * We have to build an additional level of namespace
    8362                 :             :                                                  * to keep in step with varlevelsup in the CTE;
    8363                 :             :                                                  * furthermore it could be an outer CTE (compare
    8364                 :             :                                                  * SUBQUERY case above).
    8365                 :             :                                                  */
    8366                 :           0 :                                                 List       *save_nslist = context->namespaces;
    8367                 :           0 :                                                 List       *parent_namespaces;
    8368                 :           0 :                                                 deparse_namespace mydpns;
    8369                 :           0 :                                                 const char *result;
    8370                 :             : 
    8371                 :           0 :                                                 parent_namespaces = list_copy_tail(context->namespaces,
    8372                 :           0 :                                                                                                                    ctelevelsup);
    8373                 :             : 
    8374                 :           0 :                                                 set_deparse_for_query(&mydpns, ctequery,
    8375                 :           0 :                                                                                           parent_namespaces);
    8376                 :             : 
    8377                 :           0 :                                                 context->namespaces = lcons(&mydpns, parent_namespaces);
    8378                 :             : 
    8379                 :           0 :                                                 result = get_name_for_var_field((Var *) expr, fieldno,
    8380                 :           0 :                                                                                                                 0, context);
    8381                 :             : 
    8382                 :           0 :                                                 context->namespaces = save_nslist;
    8383                 :             : 
    8384                 :           0 :                                                 return result;
    8385                 :           0 :                                         }
    8386                 :             :                                         /* else fall through to inspect the expression */
    8387         [ #  # ]:           0 :                                 }
    8388                 :             :                                 else
    8389                 :             :                                 {
    8390                 :             :                                         /*
    8391                 :             :                                          * We're deparsing a Plan tree so we don't have a CTE
    8392                 :             :                                          * list.  But the only places we'd normally see a Var
    8393                 :             :                                          * directly referencing a CTE RTE are in CteScan or
    8394                 :             :                                          * WorkTableScan plan nodes.  For those cases,
    8395                 :             :                                          * set_deparse_plan arranged for dpns->inner_plan to be
    8396                 :             :                                          * the plan node that emits the CTE or RecursiveUnion
    8397                 :             :                                          * result, and we can look at its tlist instead.  As
    8398                 :             :                                          * above, this can fail if the CTE has been proven empty,
    8399                 :             :                                          * in which case fall back to "fN".
    8400                 :             :                                          */
    8401                 :           0 :                                         TargetEntry *tle;
    8402                 :           0 :                                         deparse_namespace save_dpns;
    8403                 :           0 :                                         const char *result;
    8404                 :             : 
    8405         [ #  # ]:           0 :                                         if (!dpns->inner_plan)
    8406                 :             :                                         {
    8407                 :           0 :                                                 char       *dummy_name = palloc(32);
    8408                 :             : 
    8409         [ #  # ]:           0 :                                                 Assert(dpns->plan && IsA(dpns->plan, Result));
    8410                 :           0 :                                                 snprintf(dummy_name, 32, "f%d", fieldno);
    8411                 :           0 :                                                 return dummy_name;
    8412                 :           0 :                                         }
    8413   [ #  #  #  # ]:           0 :                                         Assert(dpns->plan && (IsA(dpns->plan, CteScan) ||
    8414                 :             :                                                                                   IsA(dpns->plan, WorkTableScan)));
    8415                 :             : 
    8416                 :           0 :                                         tle = get_tle_by_resno(dpns->inner_tlist, attnum);
    8417         [ #  # ]:           0 :                                         if (!tle)
    8418   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "bogus varattno for subquery var: %d",
    8419                 :             :                                                          attnum);
    8420         [ #  # ]:           0 :                                         Assert(netlevelsup == 0);
    8421                 :           0 :                                         push_child_plan(dpns, dpns->inner_plan, &save_dpns);
    8422                 :             : 
    8423                 :           0 :                                         result = get_name_for_var_field((Var *) tle->expr, fieldno,
    8424                 :           0 :                                                                                                         levelsup, context);
    8425                 :             : 
    8426                 :           0 :                                         pop_child_plan(dpns, &save_dpns);
    8427                 :           0 :                                         return result;
    8428                 :           0 :                                 }
    8429         [ #  # ]:           0 :                         }
    8430                 :           0 :                         break;
    8431                 :             :                 case RTE_GROUP:
    8432                 :             : 
    8433                 :             :                         /*
    8434                 :             :                          * We couldn't get here: any Vars that reference the RTE_GROUP RTE
    8435                 :             :                          * should have been replaced with the underlying grouping
    8436                 :             :                          * expressions.
    8437                 :             :                          */
    8438                 :             :                         break;
    8439                 :             :         }
    8440                 :             : 
    8441                 :             :         /*
    8442                 :             :          * We now have an expression we can't expand any more, so see if
    8443                 :             :          * get_expr_result_tupdesc() can do anything with it.
    8444                 :             :          */
    8445                 :           0 :         tupleDesc = get_expr_result_tupdesc(expr, false);
    8446                 :             :         /* Got the tupdesc, so we can extract the field name */
    8447         [ #  # ]:           0 :         Assert(fieldno >= 1 && fieldno <= tupleDesc->natts);
    8448                 :           0 :         return NameStr(TupleDescAttr(tupleDesc, fieldno - 1)->attname);
    8449                 :           0 : }
    8450                 :             : 
    8451                 :             : /*
    8452                 :             :  * Try to find the referenced expression for a PARAM_EXEC Param that might
    8453                 :             :  * reference a parameter supplied by an upper NestLoop or SubPlan plan node.
    8454                 :             :  *
    8455                 :             :  * If successful, return the expression and set *dpns_p and *ancestor_cell_p
    8456                 :             :  * appropriately for calling push_ancestor_plan().  If no referent can be
    8457                 :             :  * found, return NULL.
    8458                 :             :  */
    8459                 :             : static Node *
    8460                 :           0 : find_param_referent(Param *param, deparse_context *context,
    8461                 :             :                                         deparse_namespace **dpns_p, ListCell **ancestor_cell_p)
    8462                 :             : {
    8463                 :             :         /* Initialize output parameters to prevent compiler warnings */
    8464                 :           0 :         *dpns_p = NULL;
    8465                 :           0 :         *ancestor_cell_p = NULL;
    8466                 :             : 
    8467                 :             :         /*
    8468                 :             :          * If it's a PARAM_EXEC parameter, look for a matching NestLoopParam or
    8469                 :             :          * SubPlan argument.  This will necessarily be in some ancestor of the
    8470                 :             :          * current expression's Plan node.
    8471                 :             :          */
    8472         [ #  # ]:           0 :         if (param->paramkind == PARAM_EXEC)
    8473                 :             :         {
    8474                 :           0 :                 deparse_namespace *dpns;
    8475                 :           0 :                 Plan       *child_plan;
    8476                 :           0 :                 ListCell   *lc;
    8477                 :             : 
    8478                 :           0 :                 dpns = (deparse_namespace *) linitial(context->namespaces);
    8479                 :           0 :                 child_plan = dpns->plan;
    8480                 :             : 
    8481   [ #  #  #  #  :           0 :                 foreach(lc, dpns->ancestors)
             #  #  #  # ]
    8482                 :             :                 {
    8483                 :           0 :                         Node       *ancestor = (Node *) lfirst(lc);
    8484                 :           0 :                         ListCell   *lc2;
    8485                 :             : 
    8486                 :             :                         /*
    8487                 :             :                          * NestLoops transmit params to their inner child only.
    8488                 :             :                          */
    8489   [ #  #  #  # ]:           0 :                         if (IsA(ancestor, NestLoop) &&
    8490                 :           0 :                                 child_plan == innerPlan(ancestor))
    8491                 :             :                         {
    8492                 :           0 :                                 NestLoop   *nl = (NestLoop *) ancestor;
    8493                 :             : 
    8494   [ #  #  #  #  :           0 :                                 foreach(lc2, nl->nestParams)
             #  #  #  # ]
    8495                 :             :                                 {
    8496                 :           0 :                                         NestLoopParam *nlp = (NestLoopParam *) lfirst(lc2);
    8497                 :             : 
    8498         [ #  # ]:           0 :                                         if (nlp->paramno == param->paramid)
    8499                 :             :                                         {
    8500                 :             :                                                 /* Found a match, so return it */
    8501                 :           0 :                                                 *dpns_p = dpns;
    8502                 :           0 :                                                 *ancestor_cell_p = lc;
    8503                 :           0 :                                                 return (Node *) nlp->paramval;
    8504                 :             :                                         }
    8505         [ #  # ]:           0 :                                 }
    8506         [ #  # ]:           0 :                         }
    8507                 :             : 
    8508                 :             :                         /*
    8509                 :             :                          * If ancestor is a SubPlan, check the arguments it provides.
    8510                 :             :                          */
    8511         [ #  # ]:           0 :                         if (IsA(ancestor, SubPlan))
    8512                 :             :                         {
    8513                 :           0 :                                 SubPlan    *subplan = (SubPlan *) ancestor;
    8514                 :           0 :                                 ListCell   *lc3;
    8515                 :           0 :                                 ListCell   *lc4;
    8516                 :             : 
    8517   [ #  #  #  #  :           0 :                                 forboth(lc3, subplan->parParam, lc4, subplan->args)
          #  #  #  #  #  
             #  #  #  #  
                      # ]
    8518                 :             :                                 {
    8519                 :           0 :                                         int                     paramid = lfirst_int(lc3);
    8520                 :           0 :                                         Node       *arg = (Node *) lfirst(lc4);
    8521                 :             : 
    8522         [ #  # ]:           0 :                                         if (paramid == param->paramid)
    8523                 :             :                                         {
    8524                 :             :                                                 /*
    8525                 :             :                                                  * Found a match, so return it.  But, since Vars in
    8526                 :             :                                                  * the arg are to be evaluated in the surrounding
    8527                 :             :                                                  * context, we have to point to the next ancestor item
    8528                 :             :                                                  * that is *not* a SubPlan.
    8529                 :             :                                                  */
    8530                 :           0 :                                                 ListCell   *rest;
    8531                 :             : 
    8532   [ #  #  #  #  :           0 :                                                 for_each_cell(rest, dpns->ancestors,
             #  #  #  # ]
    8533                 :             :                                                                           lnext(dpns->ancestors, lc))
    8534                 :             :                                                 {
    8535                 :           0 :                                                         Node       *ancestor2 = (Node *) lfirst(rest);
    8536                 :             : 
    8537         [ #  # ]:           0 :                                                         if (!IsA(ancestor2, SubPlan))
    8538                 :             :                                                         {
    8539                 :           0 :                                                                 *dpns_p = dpns;
    8540                 :           0 :                                                                 *ancestor_cell_p = rest;
    8541                 :           0 :                                                                 return arg;
    8542                 :             :                                                         }
    8543         [ #  # ]:           0 :                                                 }
    8544   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "SubPlan cannot be outermost ancestor");
    8545         [ #  # ]:           0 :                                         }
    8546         [ #  # ]:           0 :                                 }
    8547                 :             : 
    8548                 :             :                                 /* SubPlan isn't a kind of Plan, so skip the rest */
    8549                 :           0 :                                 continue;
    8550                 :           0 :                         }
    8551                 :             : 
    8552                 :             :                         /*
    8553                 :             :                          * We need not consider the ancestor's initPlan list, since
    8554                 :             :                          * initplans never have any parParams.
    8555                 :             :                          */
    8556                 :             : 
    8557                 :             :                         /* No luck, crawl up to next ancestor */
    8558                 :           0 :                         child_plan = (Plan *) ancestor;
    8559      [ #  #  # ]:           0 :                 }
    8560      [ #  #  # ]:           0 :         }
    8561                 :             : 
    8562                 :             :         /* No referent found */
    8563                 :           0 :         return NULL;
    8564                 :           0 : }
    8565                 :             : 
    8566                 :             : /*
    8567                 :             :  * Try to find a subplan/initplan that emits the value for a PARAM_EXEC Param.
    8568                 :             :  *
    8569                 :             :  * If successful, return the generating subplan/initplan and set *column_p
    8570                 :             :  * to the subplan's 0-based output column number.
    8571                 :             :  * Otherwise, return NULL.
    8572                 :             :  */
    8573                 :             : static SubPlan *
    8574                 :           0 : find_param_generator(Param *param, deparse_context *context, int *column_p)
    8575                 :             : {
    8576                 :             :         /* Initialize output parameter to prevent compiler warnings */
    8577                 :           0 :         *column_p = 0;
    8578                 :             : 
    8579                 :             :         /*
    8580                 :             :          * If it's a PARAM_EXEC parameter, search the current plan node as well as
    8581                 :             :          * ancestor nodes looking for a subplan or initplan that emits the value
    8582                 :             :          * for the Param.  It could appear in the setParams of an initplan or
    8583                 :             :          * MULTIEXPR_SUBLINK subplan, or in the paramIds of an ancestral SubPlan.
    8584                 :             :          */
    8585         [ #  # ]:           0 :         if (param->paramkind == PARAM_EXEC)
    8586                 :             :         {
    8587                 :           0 :                 SubPlan    *result;
    8588                 :           0 :                 deparse_namespace *dpns;
    8589                 :           0 :                 ListCell   *lc;
    8590                 :             : 
    8591                 :           0 :                 dpns = (deparse_namespace *) linitial(context->namespaces);
    8592                 :             : 
    8593                 :             :                 /* First check the innermost plan node's initplans */
    8594                 :           0 :                 result = find_param_generator_initplan(param, dpns->plan, column_p);
    8595         [ #  # ]:           0 :                 if (result)
    8596                 :           0 :                         return result;
    8597                 :             : 
    8598                 :             :                 /*
    8599                 :             :                  * The plan's targetlist might contain MULTIEXPR_SUBLINK SubPlans,
    8600                 :             :                  * which can be referenced by Params elsewhere in the targetlist.
    8601                 :             :                  * (Such Params should always be in the same targetlist, so there's no
    8602                 :             :                  * need to do this work at upper plan nodes.)
    8603                 :             :                  */
    8604   [ #  #  #  #  :           0 :                 foreach_node(TargetEntry, tle, dpns->plan->targetlist)
          #  #  #  #  #  
                #  #  # ]
    8605                 :             :                 {
    8606   [ #  #  #  # ]:           0 :                         if (tle->expr && IsA(tle->expr, SubPlan))
    8607                 :             :                         {
    8608                 :           0 :                                 SubPlan    *subplan = (SubPlan *) tle->expr;
    8609                 :             : 
    8610         [ #  # ]:           0 :                                 if (subplan->subLinkType == MULTIEXPR_SUBLINK)
    8611                 :             :                                 {
    8612   [ #  #  #  #  :           0 :                                         foreach_int(paramid, subplan->setParam)
          #  #  #  #  #  
                #  #  # ]
    8613                 :             :                                         {
    8614         [ #  # ]:           0 :                                                 if (paramid == param->paramid)
    8615                 :             :                                                 {
    8616                 :             :                                                         /* Found a match, so return it. */
    8617                 :           0 :                                                         *column_p = foreach_current_index(paramid);
    8618                 :           0 :                                                         return subplan;
    8619                 :             :                                                 }
    8620                 :           0 :                                         }
    8621                 :           0 :                                 }
    8622         [ #  # ]:           0 :                         }
    8623                 :           0 :                 }
    8624                 :             : 
    8625                 :             :                 /* No luck, so check the ancestor nodes */
    8626   [ #  #  #  #  :           0 :                 foreach(lc, dpns->ancestors)
             #  #  #  # ]
    8627                 :             :                 {
    8628                 :           0 :                         Node       *ancestor = (Node *) lfirst(lc);
    8629                 :             : 
    8630                 :             :                         /*
    8631                 :             :                          * If ancestor is a SubPlan, check the paramIds it provides.
    8632                 :             :                          */
    8633         [ #  # ]:           0 :                         if (IsA(ancestor, SubPlan))
    8634                 :             :                         {
    8635                 :           0 :                                 SubPlan    *subplan = (SubPlan *) ancestor;
    8636                 :             : 
    8637   [ #  #  #  #  :           0 :                                 foreach_int(paramid, subplan->paramIds)
          #  #  #  #  #  
                #  #  # ]
    8638                 :             :                                 {
    8639         [ #  # ]:           0 :                                         if (paramid == param->paramid)
    8640                 :             :                                         {
    8641                 :             :                                                 /* Found a match, so return it. */
    8642                 :           0 :                                                 *column_p = foreach_current_index(paramid);
    8643                 :           0 :                                                 return subplan;
    8644                 :             :                                         }
    8645                 :           0 :                                 }
    8646                 :             : 
    8647                 :             :                                 /* SubPlan isn't a kind of Plan, so skip the rest */
    8648                 :           0 :                                 continue;
    8649                 :           0 :                         }
    8650                 :             : 
    8651                 :             :                         /*
    8652                 :             :                          * Otherwise, it's some kind of Plan node, so check its initplans.
    8653                 :             :                          */
    8654                 :           0 :                         result = find_param_generator_initplan(param, (Plan *) ancestor,
    8655                 :           0 :                                                                                                    column_p);
    8656         [ #  # ]:           0 :                         if (result)
    8657                 :           0 :                                 return result;
    8658                 :             : 
    8659                 :             :                         /* No luck, crawl up to next ancestor */
    8660      [ #  #  # ]:           0 :                 }
    8661      [ #  #  # ]:           0 :         }
    8662                 :             : 
    8663                 :             :         /* No generator found */
    8664                 :           0 :         return NULL;
    8665                 :           0 : }
    8666                 :             : 
    8667                 :             : /*
    8668                 :             :  * Subroutine for find_param_generator: search one Plan node's initplans
    8669                 :             :  */
    8670                 :             : static SubPlan *
    8671                 :           0 : find_param_generator_initplan(Param *param, Plan *plan, int *column_p)
    8672                 :             : {
    8673   [ #  #  #  #  :           0 :         foreach_node(SubPlan, subplan, plan->initPlan)
          #  #  #  #  #  
             #  #  #  # ]
    8674                 :             :         {
    8675   [ #  #  #  #  :           0 :                 foreach_int(paramid, subplan->setParam)
          #  #  #  #  #  
                #  #  # ]
    8676                 :             :                 {
    8677         [ #  # ]:           0 :                         if (paramid == param->paramid)
    8678                 :             :                         {
    8679                 :             :                                 /* Found a match, so return it. */
    8680                 :           0 :                                 *column_p = foreach_current_index(paramid);
    8681                 :           0 :                                 return subplan;
    8682                 :             :                         }
    8683                 :           0 :                 }
    8684                 :           0 :         }
    8685                 :           0 :         return NULL;
    8686                 :           0 : }
    8687                 :             : 
    8688                 :             : /*
    8689                 :             :  * Display a Param appropriately.
    8690                 :             :  */
    8691                 :             : static void
    8692                 :           0 : get_parameter(Param *param, deparse_context *context)
    8693                 :             : {
    8694                 :           0 :         Node       *expr;
    8695                 :           0 :         deparse_namespace *dpns;
    8696                 :           0 :         ListCell   *ancestor_cell;
    8697                 :           0 :         SubPlan    *subplan;
    8698                 :           0 :         int                     column;
    8699                 :             : 
    8700                 :             :         /*
    8701                 :             :          * If it's a PARAM_EXEC parameter, try to locate the expression from which
    8702                 :             :          * the parameter was computed.  This stanza handles only cases in which
    8703                 :             :          * the Param represents an input to the subplan we are currently in.
    8704                 :             :          */
    8705                 :           0 :         expr = find_param_referent(param, context, &dpns, &ancestor_cell);
    8706         [ #  # ]:           0 :         if (expr)
    8707                 :             :         {
    8708                 :             :                 /* Found a match, so print it */
    8709                 :           0 :                 deparse_namespace save_dpns;
    8710                 :           0 :                 bool            save_varprefix;
    8711                 :           0 :                 bool            need_paren;
    8712                 :             : 
    8713                 :             :                 /* Switch attention to the ancestor plan node */
    8714                 :           0 :                 push_ancestor_plan(dpns, ancestor_cell, &save_dpns);
    8715                 :             : 
    8716                 :             :                 /*
    8717                 :             :                  * Force prefixing of Vars, since they won't belong to the relation
    8718                 :             :                  * being scanned in the original plan node.
    8719                 :             :                  */
    8720                 :           0 :                 save_varprefix = context->varprefix;
    8721                 :           0 :                 context->varprefix = true;
    8722                 :             : 
    8723                 :             :                 /*
    8724                 :             :                  * A Param's expansion is typically a Var, Aggref, GroupingFunc, or
    8725                 :             :                  * upper-level Param, which wouldn't need extra parentheses.
    8726                 :             :                  * Otherwise, insert parens to ensure the expression looks atomic.
    8727                 :             :                  */
    8728         [ #  # ]:           0 :                 need_paren = !(IsA(expr, Var) ||
    8729         [ #  # ]:           0 :                                            IsA(expr, Aggref) ||
    8730         [ #  # ]:           0 :                                            IsA(expr, GroupingFunc) ||
    8731                 :           0 :                                            IsA(expr, Param));
    8732         [ #  # ]:           0 :                 if (need_paren)
    8733                 :           0 :                         appendStringInfoChar(context->buf, '(');
    8734                 :             : 
    8735                 :           0 :                 get_rule_expr(expr, context, false);
    8736                 :             : 
    8737         [ #  # ]:           0 :                 if (need_paren)
    8738                 :           0 :                         appendStringInfoChar(context->buf, ')');
    8739                 :             : 
    8740                 :           0 :                 context->varprefix = save_varprefix;
    8741                 :             : 
    8742                 :           0 :                 pop_ancestor_plan(dpns, &save_dpns);
    8743                 :             : 
    8744                 :             :                 return;
    8745                 :           0 :         }
    8746                 :             : 
    8747                 :             :         /*
    8748                 :             :          * Alternatively, maybe it's a subplan output, which we print as a
    8749                 :             :          * reference to the subplan.  (We could drill down into the subplan and
    8750                 :             :          * print the relevant targetlist expression, but that has been deemed too
    8751                 :             :          * confusing since it would violate normal SQL scope rules.  Also, we're
    8752                 :             :          * relying on this reference to show that the testexpr containing the
    8753                 :             :          * Param has anything to do with that subplan at all.)
    8754                 :             :          */
    8755                 :           0 :         subplan = find_param_generator(param, context, &column);
    8756         [ #  # ]:           0 :         if (subplan)
    8757                 :             :         {
    8758                 :           0 :                 const char *nameprefix;
    8759                 :             : 
    8760         [ #  # ]:           0 :                 if (subplan->isInitPlan)
    8761                 :           0 :                         nameprefix = "InitPlan ";
    8762                 :             :                 else
    8763                 :           0 :                         nameprefix = "SubPlan ";
    8764                 :             : 
    8765                 :           0 :                 appendStringInfo(context->buf, "(%s%s%s).col%d",
    8766                 :           0 :                                                  subplan->useHashTable ? "hashed " : "",
    8767                 :           0 :                                                  nameprefix,
    8768                 :           0 :                                                  subplan->plan_name, column + 1);
    8769                 :             : 
    8770                 :             :                 return;
    8771                 :           0 :         }
    8772                 :             : 
    8773                 :             :         /*
    8774                 :             :          * If it's an external parameter, see if the outermost namespace provides
    8775                 :             :          * function argument names.
    8776                 :             :          */
    8777   [ #  #  #  # ]:           0 :         if (param->paramkind == PARAM_EXTERN && context->namespaces != NIL)
    8778                 :             :         {
    8779                 :           0 :                 dpns = llast(context->namespaces);
    8780         [ #  # ]:           0 :                 if (dpns->argnames &&
    8781   [ #  #  #  # ]:           0 :                         param->paramid > 0 &&
    8782                 :           0 :                         param->paramid <= dpns->numargs)
    8783                 :             :                 {
    8784                 :           0 :                         char       *argname = dpns->argnames[param->paramid - 1];
    8785                 :             : 
    8786         [ #  # ]:           0 :                         if (argname)
    8787                 :             :                         {
    8788                 :           0 :                                 bool            should_qualify = false;
    8789                 :           0 :                                 ListCell   *lc;
    8790                 :             : 
    8791                 :             :                                 /*
    8792                 :             :                                  * Qualify the parameter name if there are any other deparse
    8793                 :             :                                  * namespaces with range tables.  This avoids qualifying in
    8794                 :             :                                  * trivial cases like "RETURN a + b", but makes it safe in all
    8795                 :             :                                  * other cases.
    8796                 :             :                                  */
    8797   [ #  #  #  #  :           0 :                                 foreach(lc, context->namespaces)
                   #  # ]
    8798                 :             :                                 {
    8799                 :           0 :                                         deparse_namespace *depns = lfirst(lc);
    8800                 :             : 
    8801         [ #  # ]:           0 :                                         if (depns->rtable_names != NIL)
    8802                 :             :                                         {
    8803                 :           0 :                                                 should_qualify = true;
    8804                 :           0 :                                                 break;
    8805                 :             :                                         }
    8806         [ #  # ]:           0 :                                 }
    8807         [ #  # ]:           0 :                                 if (should_qualify)
    8808                 :             :                                 {
    8809                 :           0 :                                         appendStringInfoString(context->buf, quote_identifier(dpns->funcname));
    8810                 :           0 :                                         appendStringInfoChar(context->buf, '.');
    8811                 :           0 :                                 }
    8812                 :             : 
    8813                 :           0 :                                 appendStringInfoString(context->buf, quote_identifier(argname));
    8814                 :             :                                 return;
    8815                 :           0 :                         }
    8816         [ #  # ]:           0 :                 }
    8817                 :           0 :         }
    8818                 :             : 
    8819                 :             :         /*
    8820                 :             :          * Not PARAM_EXEC, or couldn't find referent: just print $N.
    8821                 :             :          *
    8822                 :             :          * It's a bug if we get here for anything except PARAM_EXTERN Params, but
    8823                 :             :          * in production builds printing $N seems more useful than failing.
    8824                 :             :          */
    8825         [ #  # ]:           0 :         Assert(param->paramkind == PARAM_EXTERN);
    8826                 :             : 
    8827                 :           0 :         appendStringInfo(context->buf, "$%d", param->paramid);
    8828         [ #  # ]:           0 : }
    8829                 :             : 
    8830                 :             : /*
    8831                 :             :  * get_simple_binary_op_name
    8832                 :             :  *
    8833                 :             :  * helper function for isSimpleNode
    8834                 :             :  * will return single char binary operator name, or NULL if it's not
    8835                 :             :  */
    8836                 :             : static const char *
    8837                 :           0 : get_simple_binary_op_name(OpExpr *expr)
    8838                 :             : {
    8839                 :           0 :         List       *args = expr->args;
    8840                 :             : 
    8841         [ #  # ]:           0 :         if (list_length(args) == 2)
    8842                 :             :         {
    8843                 :             :                 /* binary operator */
    8844                 :           0 :                 Node       *arg1 = (Node *) linitial(args);
    8845                 :           0 :                 Node       *arg2 = (Node *) lsecond(args);
    8846                 :           0 :                 const char *op;
    8847                 :             : 
    8848                 :           0 :                 op = generate_operator_name(expr->opno, exprType(arg1), exprType(arg2));
    8849         [ #  # ]:           0 :                 if (strlen(op) == 1)
    8850                 :           0 :                         return op;
    8851         [ #  # ]:           0 :         }
    8852                 :           0 :         return NULL;
    8853                 :           0 : }
    8854                 :             : 
    8855                 :             : 
    8856                 :             : /*
    8857                 :             :  * isSimpleNode - check if given node is simple (doesn't need parenthesizing)
    8858                 :             :  *
    8859                 :             :  *      true   : simple in the context of parent node's type
    8860                 :             :  *      false  : not simple
    8861                 :             :  */
    8862                 :             : static bool
    8863                 :           0 : isSimpleNode(Node *node, Node *parentNode, int prettyFlags)
    8864                 :             : {
    8865         [ #  # ]:           0 :         if (!node)
    8866                 :           0 :                 return false;
    8867                 :             : 
    8868   [ #  #  #  #  :           0 :         switch (nodeTag(node))
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
    8869                 :             :         {
    8870                 :             :                 case T_Var:
    8871                 :             :                 case T_Const:
    8872                 :             :                 case T_Param:
    8873                 :             :                 case T_CoerceToDomainValue:
    8874                 :             :                 case T_SetToDefault:
    8875                 :             :                 case T_CurrentOfExpr:
    8876                 :             :                         /* single words: always simple */
    8877                 :           0 :                         return true;
    8878                 :             : 
    8879                 :             :                 case T_SubscriptingRef:
    8880                 :             :                 case T_ArrayExpr:
    8881                 :             :                 case T_RowExpr:
    8882                 :             :                 case T_CoalesceExpr:
    8883                 :             :                 case T_MinMaxExpr:
    8884                 :             :                 case T_SQLValueFunction:
    8885                 :             :                 case T_XmlExpr:
    8886                 :             :                 case T_NextValueExpr:
    8887                 :             :                 case T_NullIfExpr:
    8888                 :             :                 case T_Aggref:
    8889                 :             :                 case T_GroupingFunc:
    8890                 :             :                 case T_WindowFunc:
    8891                 :             :                 case T_MergeSupportFunc:
    8892                 :             :                 case T_FuncExpr:
    8893                 :             :                 case T_JsonConstructorExpr:
    8894                 :             :                 case T_JsonExpr:
    8895                 :             :                         /* function-like: name(..) or name[..] */
    8896                 :           0 :                         return true;
    8897                 :             : 
    8898                 :             :                         /* CASE keywords act as parentheses */
    8899                 :             :                 case T_CaseExpr:
    8900                 :           0 :                         return true;
    8901                 :             : 
    8902                 :             :                 case T_FieldSelect:
    8903                 :             : 
    8904                 :             :                         /*
    8905                 :             :                          * appears simple since . has top precedence, unless parent is
    8906                 :             :                          * T_FieldSelect itself!
    8907                 :             :                          */
    8908                 :           0 :                         return !IsA(parentNode, FieldSelect);
    8909                 :             : 
    8910                 :             :                 case T_FieldStore:
    8911                 :             : 
    8912                 :             :                         /*
    8913                 :             :                          * treat like FieldSelect (probably doesn't matter)
    8914                 :             :                          */
    8915                 :           0 :                         return !IsA(parentNode, FieldStore);
    8916                 :             : 
    8917                 :             :                 case T_CoerceToDomain:
    8918                 :             :                         /* maybe simple, check args */
    8919                 :           0 :                         return isSimpleNode((Node *) ((CoerceToDomain *) node)->arg,
    8920                 :           0 :                                                                 node, prettyFlags);
    8921                 :             :                 case T_RelabelType:
    8922                 :           0 :                         return isSimpleNode((Node *) ((RelabelType *) node)->arg,
    8923                 :           0 :                                                                 node, prettyFlags);
    8924                 :             :                 case T_CoerceViaIO:
    8925                 :           0 :                         return isSimpleNode((Node *) ((CoerceViaIO *) node)->arg,
    8926                 :           0 :                                                                 node, prettyFlags);
    8927                 :             :                 case T_ArrayCoerceExpr:
    8928                 :           0 :                         return isSimpleNode((Node *) ((ArrayCoerceExpr *) node)->arg,
    8929                 :           0 :                                                                 node, prettyFlags);
    8930                 :             :                 case T_ConvertRowtypeExpr:
    8931                 :           0 :                         return isSimpleNode((Node *) ((ConvertRowtypeExpr *) node)->arg,
    8932                 :           0 :                                                                 node, prettyFlags);
    8933                 :             :                 case T_ReturningExpr:
    8934                 :           0 :                         return isSimpleNode((Node *) ((ReturningExpr *) node)->retexpr,
    8935                 :           0 :                                                                 node, prettyFlags);
    8936                 :             : 
    8937                 :             :                 case T_OpExpr:
    8938                 :             :                         {
    8939                 :             :                                 /* depends on parent node type; needs further checking */
    8940   [ #  #  #  # ]:           0 :                                 if (prettyFlags & PRETTYFLAG_PAREN && IsA(parentNode, OpExpr))
    8941                 :             :                                 {
    8942                 :           0 :                                         const char *op;
    8943                 :           0 :                                         const char *parentOp;
    8944                 :           0 :                                         bool            is_lopriop;
    8945                 :           0 :                                         bool            is_hipriop;
    8946                 :           0 :                                         bool            is_lopriparent;
    8947                 :           0 :                                         bool            is_hipriparent;
    8948                 :             : 
    8949                 :           0 :                                         op = get_simple_binary_op_name((OpExpr *) node);
    8950         [ #  # ]:           0 :                                         if (!op)
    8951                 :           0 :                                                 return false;
    8952                 :             : 
    8953                 :             :                                         /* We know only the basic operators + - and * / % */
    8954                 :           0 :                                         is_lopriop = (strchr("+-", *op) != NULL);
    8955                 :           0 :                                         is_hipriop = (strchr("*/%", *op) != NULL);
    8956   [ #  #  #  # ]:           0 :                                         if (!(is_lopriop || is_hipriop))
    8957                 :           0 :                                                 return false;
    8958                 :             : 
    8959                 :           0 :                                         parentOp = get_simple_binary_op_name((OpExpr *) parentNode);
    8960         [ #  # ]:           0 :                                         if (!parentOp)
    8961                 :           0 :                                                 return false;
    8962                 :             : 
    8963                 :           0 :                                         is_lopriparent = (strchr("+-", *parentOp) != NULL);
    8964                 :           0 :                                         is_hipriparent = (strchr("*/%", *parentOp) != NULL);
    8965   [ #  #  #  # ]:           0 :                                         if (!(is_lopriparent || is_hipriparent))
    8966                 :           0 :                                                 return false;
    8967                 :             : 
    8968   [ #  #  #  # ]:           0 :                                         if (is_hipriop && is_lopriparent)
    8969                 :           0 :                                                 return true;    /* op binds tighter than parent */
    8970                 :             : 
    8971   [ #  #  #  # ]:           0 :                                         if (is_lopriop && is_hipriparent)
    8972                 :           0 :                                                 return false;
    8973                 :             : 
    8974                 :             :                                         /*
    8975                 :             :                                          * Operators are same priority --- can skip parens only if
    8976                 :             :                                          * we have (a - b) - c, not a - (b - c).
    8977                 :             :                                          */
    8978         [ #  # ]:           0 :                                         if (node == (Node *) linitial(((OpExpr *) parentNode)->args))
    8979                 :           0 :                                                 return true;
    8980                 :             : 
    8981                 :           0 :                                         return false;
    8982                 :           0 :                                 }
    8983                 :             :                                 /* else do the same stuff as for T_SubLink et al. */
    8984                 :           0 :                         }
    8985                 :             :                         /* FALLTHROUGH */
    8986                 :             : 
    8987                 :             :                 case T_SubLink:
    8988                 :             :                 case T_NullTest:
    8989                 :             :                 case T_BooleanTest:
    8990                 :             :                 case T_DistinctExpr:
    8991                 :             :                 case T_JsonIsPredicate:
    8992      [ #  #  # ]:           0 :                         switch (nodeTag(parentNode))
    8993                 :             :                         {
    8994                 :             :                                 case T_FuncExpr:
    8995                 :             :                                         {
    8996                 :             :                                                 /* special handling for casts and COERCE_SQL_SYNTAX */
    8997                 :           0 :                                                 CoercionForm type = ((FuncExpr *) parentNode)->funcformat;
    8998                 :             : 
    8999         [ #  # ]:           0 :                                                 if (type == COERCE_EXPLICIT_CAST ||
    9000   [ #  #  #  # ]:           0 :                                                         type == COERCE_IMPLICIT_CAST ||
    9001                 :           0 :                                                         type == COERCE_SQL_SYNTAX)
    9002                 :           0 :                                                         return false;
    9003                 :           0 :                                                 return true;    /* own parentheses */
    9004                 :           0 :                                         }
    9005                 :             :                                 case T_BoolExpr:        /* lower precedence */
    9006                 :             :                                 case T_SubscriptingRef: /* other separators */
    9007                 :             :                                 case T_ArrayExpr:       /* other separators */
    9008                 :             :                                 case T_RowExpr: /* other separators */
    9009                 :             :                                 case T_CoalesceExpr:    /* own parentheses */
    9010                 :             :                                 case T_MinMaxExpr:      /* own parentheses */
    9011                 :             :                                 case T_XmlExpr: /* own parentheses */
    9012                 :             :                                 case T_NullIfExpr:      /* other separators */
    9013                 :             :                                 case T_Aggref:  /* own parentheses */
    9014                 :             :                                 case T_GroupingFunc:    /* own parentheses */
    9015                 :             :                                 case T_WindowFunc:      /* own parentheses */
    9016                 :             :                                 case T_CaseExpr:        /* other separators */
    9017                 :           0 :                                         return true;
    9018                 :             :                                 default:
    9019                 :           0 :                                         return false;
    9020                 :             :                         }
    9021                 :             : 
    9022                 :             :                 case T_BoolExpr:
    9023   [ #  #  #  # ]:           0 :                         switch (nodeTag(parentNode))
    9024                 :             :                         {
    9025                 :             :                                 case T_BoolExpr:
    9026         [ #  # ]:           0 :                                         if (prettyFlags & PRETTYFLAG_PAREN)
    9027                 :             :                                         {
    9028                 :           0 :                                                 BoolExprType type;
    9029                 :           0 :                                                 BoolExprType parentType;
    9030                 :             : 
    9031                 :           0 :                                                 type = ((BoolExpr *) node)->boolop;
    9032                 :           0 :                                                 parentType = ((BoolExpr *) parentNode)->boolop;
    9033      [ #  #  # ]:           0 :                                                 switch (type)
    9034                 :             :                                                 {
    9035                 :             :                                                         case NOT_EXPR:
    9036                 :             :                                                         case AND_EXPR:
    9037   [ #  #  #  # ]:           0 :                                                                 if (parentType == AND_EXPR || parentType == OR_EXPR)
    9038                 :           0 :                                                                         return true;
    9039                 :           0 :                                                                 break;
    9040                 :             :                                                         case OR_EXPR:
    9041         [ #  # ]:           0 :                                                                 if (parentType == OR_EXPR)
    9042                 :           0 :                                                                         return true;
    9043                 :           0 :                                                                 break;
    9044                 :             :                                                 }
    9045      [ #  #  # ]:           0 :                                         }
    9046                 :           0 :                                         return false;
    9047                 :             :                                 case T_FuncExpr:
    9048                 :             :                                         {
    9049                 :             :                                                 /* special handling for casts and COERCE_SQL_SYNTAX */
    9050                 :           0 :                                                 CoercionForm type = ((FuncExpr *) parentNode)->funcformat;
    9051                 :             : 
    9052         [ #  # ]:           0 :                                                 if (type == COERCE_EXPLICIT_CAST ||
    9053   [ #  #  #  # ]:           0 :                                                         type == COERCE_IMPLICIT_CAST ||
    9054                 :           0 :                                                         type == COERCE_SQL_SYNTAX)
    9055                 :           0 :                                                         return false;
    9056                 :           0 :                                                 return true;    /* own parentheses */
    9057                 :           0 :                                         }
    9058                 :             :                                 case T_SubscriptingRef: /* other separators */
    9059                 :             :                                 case T_ArrayExpr:       /* other separators */
    9060                 :             :                                 case T_RowExpr: /* other separators */
    9061                 :             :                                 case T_CoalesceExpr:    /* own parentheses */
    9062                 :             :                                 case T_MinMaxExpr:      /* own parentheses */
    9063                 :             :                                 case T_XmlExpr: /* own parentheses */
    9064                 :             :                                 case T_NullIfExpr:      /* other separators */
    9065                 :             :                                 case T_Aggref:  /* own parentheses */
    9066                 :             :                                 case T_GroupingFunc:    /* own parentheses */
    9067                 :             :                                 case T_WindowFunc:      /* own parentheses */
    9068                 :             :                                 case T_CaseExpr:        /* other separators */
    9069                 :             :                                 case T_JsonExpr:        /* own parentheses */
    9070                 :           0 :                                         return true;
    9071                 :             :                                 default:
    9072                 :           0 :                                         return false;
    9073                 :             :                         }
    9074                 :             : 
    9075                 :             :                 case T_JsonValueExpr:
    9076                 :             :                         /* maybe simple, check args */
    9077                 :           0 :                         return isSimpleNode((Node *) ((JsonValueExpr *) node)->raw_expr,
    9078                 :           0 :                                                                 node, prettyFlags);
    9079                 :             : 
    9080                 :             :                 default:
    9081                 :           0 :                         break;
    9082                 :             :         }
    9083                 :             :         /* those we don't know: in dubio complexo */
    9084                 :           0 :         return false;
    9085                 :           0 : }
    9086                 :             : 
    9087                 :             : 
    9088                 :             : /*
    9089                 :             :  * appendContextKeyword - append a keyword to buffer
    9090                 :             :  *
    9091                 :             :  * If prettyPrint is enabled, perform a line break, and adjust indentation.
    9092                 :             :  * Otherwise, just append the keyword.
    9093                 :             :  */
    9094                 :             : static void
    9095                 :           0 : appendContextKeyword(deparse_context *context, const char *str,
    9096                 :             :                                          int indentBefore, int indentAfter, int indentPlus)
    9097                 :             : {
    9098                 :           0 :         StringInfo      buf = context->buf;
    9099                 :             : 
    9100         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
    9101                 :             :         {
    9102                 :           0 :                 int                     indentAmount;
    9103                 :             : 
    9104                 :           0 :                 context->indentLevel += indentBefore;
    9105                 :             : 
    9106                 :             :                 /* remove any trailing spaces currently in the buffer ... */
    9107                 :           0 :                 removeStringInfoSpaces(buf);
    9108                 :             :                 /* ... then add a newline and some spaces */
    9109                 :           0 :                 appendStringInfoChar(buf, '\n');
    9110                 :             : 
    9111         [ #  # ]:           0 :                 if (context->indentLevel < PRETTYINDENT_LIMIT)
    9112         [ #  # ]:           0 :                         indentAmount = Max(context->indentLevel, 0) + indentPlus;
    9113                 :             :                 else
    9114                 :             :                 {
    9115                 :             :                         /*
    9116                 :             :                          * If we're indented more than PRETTYINDENT_LIMIT characters, try
    9117                 :             :                          * to conserve horizontal space by reducing the per-level
    9118                 :             :                          * indentation.  For best results the scale factor here should
    9119                 :             :                          * divide all the indent amounts that get added to indentLevel
    9120                 :             :                          * (PRETTYINDENT_STD, etc).  It's important that the indentation
    9121                 :             :                          * not grow unboundedly, else deeply-nested trees use O(N^2)
    9122                 :             :                          * whitespace; so we also wrap modulo PRETTYINDENT_LIMIT.
    9123                 :             :                          */
    9124                 :           0 :                         indentAmount = PRETTYINDENT_LIMIT +
    9125                 :           0 :                                 (context->indentLevel - PRETTYINDENT_LIMIT) /
    9126                 :             :                                 (PRETTYINDENT_STD / 2);
    9127                 :           0 :                         indentAmount %= PRETTYINDENT_LIMIT;
    9128                 :             :                         /* scale/wrap logic affects indentLevel, but not indentPlus */
    9129                 :           0 :                         indentAmount += indentPlus;
    9130                 :             :                 }
    9131                 :           0 :                 appendStringInfoSpaces(buf, indentAmount);
    9132                 :             : 
    9133                 :           0 :                 appendStringInfoString(buf, str);
    9134                 :             : 
    9135                 :           0 :                 context->indentLevel += indentAfter;
    9136         [ #  # ]:           0 :                 if (context->indentLevel < 0)
    9137                 :           0 :                         context->indentLevel = 0;
    9138                 :           0 :         }
    9139                 :             :         else
    9140                 :           0 :                 appendStringInfoString(buf, str);
    9141                 :           0 : }
    9142                 :             : 
    9143                 :             : /*
    9144                 :             :  * removeStringInfoSpaces - delete trailing spaces from a buffer.
    9145                 :             :  *
    9146                 :             :  * Possibly this should move to stringinfo.c at some point.
    9147                 :             :  */
    9148                 :             : static void
    9149                 :           0 : removeStringInfoSpaces(StringInfo str)
    9150                 :             : {
    9151   [ #  #  #  # ]:           0 :         while (str->len > 0 && str->data[str->len - 1] == ' ')
    9152                 :           0 :                 str->data[--(str->len)] = '\0';
    9153                 :           0 : }
    9154                 :             : 
    9155                 :             : 
    9156                 :             : /*
    9157                 :             :  * get_rule_expr_paren  - deparse expr using get_rule_expr,
    9158                 :             :  * embracing the string with parentheses if necessary for prettyPrint.
    9159                 :             :  *
    9160                 :             :  * Never embrace if prettyFlags=0, because it's done in the calling node.
    9161                 :             :  *
    9162                 :             :  * Any node that does *not* embrace its argument node by sql syntax (with
    9163                 :             :  * parentheses, non-operator keywords like CASE/WHEN/ON, or comma etc) should
    9164                 :             :  * use get_rule_expr_paren instead of get_rule_expr so parentheses can be
    9165                 :             :  * added.
    9166                 :             :  */
    9167                 :             : static void
    9168                 :           0 : get_rule_expr_paren(Node *node, deparse_context *context,
    9169                 :             :                                         bool showimplicit, Node *parentNode)
    9170                 :             : {
    9171                 :           0 :         bool            need_paren;
    9172                 :             : 
    9173         [ #  # ]:           0 :         need_paren = PRETTY_PAREN(context) &&
    9174                 :           0 :                 !isSimpleNode(node, parentNode, context->prettyFlags);
    9175                 :             : 
    9176         [ #  # ]:           0 :         if (need_paren)
    9177                 :           0 :                 appendStringInfoChar(context->buf, '(');
    9178                 :             : 
    9179                 :           0 :         get_rule_expr(node, context, showimplicit);
    9180                 :             : 
    9181         [ #  # ]:           0 :         if (need_paren)
    9182                 :           0 :                 appendStringInfoChar(context->buf, ')');
    9183                 :           0 : }
    9184                 :             : 
    9185                 :             : static void
    9186                 :           0 : get_json_behavior(JsonBehavior *behavior, deparse_context *context,
    9187                 :             :                                   const char *on)
    9188                 :             : {
    9189                 :             :         /*
    9190                 :             :          * The order of array elements must correspond to the order of
    9191                 :             :          * JsonBehaviorType members.
    9192                 :             :          */
    9193                 :           0 :         const char *behavior_names[] =
    9194                 :             :         {
    9195                 :             :                 " NULL",
    9196                 :             :                 " ERROR",
    9197                 :             :                 " EMPTY",
    9198                 :             :                 " TRUE",
    9199                 :             :                 " FALSE",
    9200                 :             :                 " UNKNOWN",
    9201                 :             :                 " EMPTY ARRAY",
    9202                 :             :                 " EMPTY OBJECT",
    9203                 :             :                 " DEFAULT "
    9204                 :             :         };
    9205                 :             : 
    9206         [ #  # ]:           0 :         if ((int) behavior->btype < 0 || behavior->btype >= lengthof(behavior_names))
    9207   [ #  #  #  # ]:           0 :                 elog(ERROR, "invalid json behavior type: %d", behavior->btype);
    9208                 :             : 
    9209                 :           0 :         appendStringInfoString(context->buf, behavior_names[behavior->btype]);
    9210                 :             : 
    9211         [ #  # ]:           0 :         if (behavior->btype == JSON_BEHAVIOR_DEFAULT)
    9212                 :           0 :                 get_rule_expr(behavior->expr, context, false);
    9213                 :             : 
    9214                 :           0 :         appendStringInfo(context->buf, " ON %s", on);
    9215                 :           0 : }
    9216                 :             : 
    9217                 :             : /*
    9218                 :             :  * get_json_expr_options
    9219                 :             :  *
    9220                 :             :  * Parse back common options for JSON_QUERY, JSON_VALUE, JSON_EXISTS and
    9221                 :             :  * JSON_TABLE columns.
    9222                 :             :  */
    9223                 :             : static void
    9224                 :           0 : get_json_expr_options(JsonExpr *jsexpr, deparse_context *context,
    9225                 :             :                                           JsonBehaviorType default_behavior)
    9226                 :             : {
    9227         [ #  # ]:           0 :         if (jsexpr->op == JSON_QUERY_OP)
    9228                 :             :         {
    9229         [ #  # ]:           0 :                 if (jsexpr->wrapper == JSW_CONDITIONAL)
    9230                 :           0 :                         appendStringInfoString(context->buf, " WITH CONDITIONAL WRAPPER");
    9231         [ #  # ]:           0 :                 else if (jsexpr->wrapper == JSW_UNCONDITIONAL)
    9232                 :           0 :                         appendStringInfoString(context->buf, " WITH UNCONDITIONAL WRAPPER");
    9233                 :             :                 /* The default */
    9234   [ #  #  #  # ]:           0 :                 else if (jsexpr->wrapper == JSW_NONE || jsexpr->wrapper == JSW_UNSPEC)
    9235                 :           0 :                         appendStringInfoString(context->buf, " WITHOUT WRAPPER");
    9236                 :             : 
    9237         [ #  # ]:           0 :                 if (jsexpr->omit_quotes)
    9238                 :           0 :                         appendStringInfoString(context->buf, " OMIT QUOTES");
    9239                 :             :                 /* The default */
    9240                 :             :                 else
    9241                 :           0 :                         appendStringInfoString(context->buf, " KEEP QUOTES");
    9242                 :           0 :         }
    9243                 :             : 
    9244   [ #  #  #  # ]:           0 :         if (jsexpr->on_empty && jsexpr->on_empty->btype != default_behavior)
    9245                 :           0 :                 get_json_behavior(jsexpr->on_empty, context, "EMPTY");
    9246                 :             : 
    9247   [ #  #  #  # ]:           0 :         if (jsexpr->on_error && jsexpr->on_error->btype != default_behavior)
    9248                 :           0 :                 get_json_behavior(jsexpr->on_error, context, "ERROR");
    9249                 :           0 : }
    9250                 :             : 
    9251                 :             : /* ----------
    9252                 :             :  * get_rule_expr                        - Parse back an expression
    9253                 :             :  *
    9254                 :             :  * Note: showimplicit determines whether we display any implicit cast that
    9255                 :             :  * is present at the top of the expression tree.  It is a passed argument,
    9256                 :             :  * not a field of the context struct, because we change the value as we
    9257                 :             :  * recurse down into the expression.  In general we suppress implicit casts
    9258                 :             :  * when the result type is known with certainty (eg, the arguments of an
    9259                 :             :  * OR must be boolean).  We display implicit casts for arguments of functions
    9260                 :             :  * and operators, since this is needed to be certain that the same function
    9261                 :             :  * or operator will be chosen when the expression is re-parsed.
    9262                 :             :  * ----------
    9263                 :             :  */
    9264                 :             : static void
    9265                 :           0 : get_rule_expr(Node *node, deparse_context *context,
    9266                 :             :                           bool showimplicit)
    9267                 :             : {
    9268                 :           0 :         StringInfo      buf = context->buf;
    9269                 :             : 
    9270         [ #  # ]:           0 :         if (node == NULL)
    9271                 :           0 :                 return;
    9272                 :             : 
    9273                 :             :         /* Guard against excessively long or deeply-nested queries */
    9274         [ #  # ]:           0 :         CHECK_FOR_INTERRUPTS();
    9275                 :           0 :         check_stack_depth();
    9276                 :             : 
    9277                 :             :         /*
    9278                 :             :          * Each level of get_rule_expr must emit an indivisible term
    9279                 :             :          * (parenthesized if necessary) to ensure result is reparsed into the same
    9280                 :             :          * expression tree.  The only exception is that when the input is a List,
    9281                 :             :          * we emit the component items comma-separated with no surrounding
    9282                 :             :          * decoration; this is convenient for most callers.
    9283                 :             :          */
    9284   [ #  #  #  #  :           0 :         switch (nodeTag(node))
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
    9285                 :             :         {
    9286                 :             :                 case T_Var:
    9287                 :           0 :                         (void) get_variable((Var *) node, 0, false, context);
    9288                 :           0 :                         break;
    9289                 :             : 
    9290                 :             :                 case T_Const:
    9291                 :           0 :                         get_const_expr((Const *) node, context, 0);
    9292                 :           0 :                         break;
    9293                 :             : 
    9294                 :             :                 case T_Param:
    9295                 :           0 :                         get_parameter((Param *) node, context);
    9296                 :           0 :                         break;
    9297                 :             : 
    9298                 :             :                 case T_Aggref:
    9299                 :           0 :                         get_agg_expr((Aggref *) node, context, (Aggref *) node);
    9300                 :           0 :                         break;
    9301                 :             : 
    9302                 :             :                 case T_GroupingFunc:
    9303                 :             :                         {
    9304                 :           0 :                                 GroupingFunc *gexpr = (GroupingFunc *) node;
    9305                 :             : 
    9306                 :           0 :                                 appendStringInfoString(buf, "GROUPING(");
    9307                 :           0 :                                 get_rule_expr((Node *) gexpr->args, context, true);
    9308                 :           0 :                                 appendStringInfoChar(buf, ')');
    9309                 :           0 :                         }
    9310                 :           0 :                         break;
    9311                 :             : 
    9312                 :             :                 case T_WindowFunc:
    9313                 :           0 :                         get_windowfunc_expr((WindowFunc *) node, context);
    9314                 :           0 :                         break;
    9315                 :             : 
    9316                 :             :                 case T_MergeSupportFunc:
    9317                 :           0 :                         appendStringInfoString(buf, "MERGE_ACTION()");
    9318                 :           0 :                         break;
    9319                 :             : 
    9320                 :             :                 case T_SubscriptingRef:
    9321                 :             :                         {
    9322                 :           0 :                                 SubscriptingRef *sbsref = (SubscriptingRef *) node;
    9323                 :           0 :                                 bool            need_parens;
    9324                 :             : 
    9325                 :             :                                 /*
    9326                 :             :                                  * If the argument is a CaseTestExpr, we must be inside a
    9327                 :             :                                  * FieldStore, ie, we are assigning to an element of an array
    9328                 :             :                                  * within a composite column.  Since we already punted on
    9329                 :             :                                  * displaying the FieldStore's target information, just punt
    9330                 :             :                                  * here too, and display only the assignment source
    9331                 :             :                                  * expression.
    9332                 :             :                                  */
    9333         [ #  # ]:           0 :                                 if (IsA(sbsref->refexpr, CaseTestExpr))
    9334                 :             :                                 {
    9335         [ #  # ]:           0 :                                         Assert(sbsref->refassgnexpr);
    9336                 :           0 :                                         get_rule_expr((Node *) sbsref->refassgnexpr,
    9337                 :           0 :                                                                   context, showimplicit);
    9338                 :           0 :                                         break;
    9339                 :             :                                 }
    9340                 :             : 
    9341                 :             :                                 /*
    9342                 :             :                                  * Parenthesize the argument unless it's a simple Var or a
    9343                 :             :                                  * FieldSelect.  (In particular, if it's another
    9344                 :             :                                  * SubscriptingRef, we *must* parenthesize to avoid
    9345                 :             :                                  * confusion.)
    9346                 :             :                                  */
    9347         [ #  # ]:           0 :                                 need_parens = !IsA(sbsref->refexpr, Var) &&
    9348                 :           0 :                                         !IsA(sbsref->refexpr, FieldSelect);
    9349         [ #  # ]:           0 :                                 if (need_parens)
    9350                 :           0 :                                         appendStringInfoChar(buf, '(');
    9351                 :           0 :                                 get_rule_expr((Node *) sbsref->refexpr, context, showimplicit);
    9352         [ #  # ]:           0 :                                 if (need_parens)
    9353                 :           0 :                                         appendStringInfoChar(buf, ')');
    9354                 :             : 
    9355                 :             :                                 /*
    9356                 :             :                                  * If there's a refassgnexpr, we want to print the node in the
    9357                 :             :                                  * format "container[subscripts] := refassgnexpr".  This is
    9358                 :             :                                  * not legal SQL, so decompilation of INSERT or UPDATE
    9359                 :             :                                  * statements should always use processIndirection as part of
    9360                 :             :                                  * the statement-level syntax.  We should only see this when
    9361                 :             :                                  * EXPLAIN tries to print the targetlist of a plan resulting
    9362                 :             :                                  * from such a statement.
    9363                 :             :                                  */
    9364         [ #  # ]:           0 :                                 if (sbsref->refassgnexpr)
    9365                 :             :                                 {
    9366                 :           0 :                                         Node       *refassgnexpr;
    9367                 :             : 
    9368                 :             :                                         /*
    9369                 :             :                                          * Use processIndirection to print this node's subscripts
    9370                 :             :                                          * as well as any additional field selections or
    9371                 :             :                                          * subscripting in immediate descendants.  It returns the
    9372                 :             :                                          * RHS expr that is actually being "assigned".
    9373                 :             :                                          */
    9374                 :           0 :                                         refassgnexpr = processIndirection(node, context);
    9375                 :           0 :                                         appendStringInfoString(buf, " := ");
    9376                 :           0 :                                         get_rule_expr(refassgnexpr, context, showimplicit);
    9377                 :           0 :                                 }
    9378                 :             :                                 else
    9379                 :             :                                 {
    9380                 :             :                                         /* Just an ordinary container fetch, so print subscripts */
    9381                 :           0 :                                         printSubscripts(sbsref, context);
    9382                 :             :                                 }
    9383         [ #  # ]:           0 :                         }
    9384                 :           0 :                         break;
    9385                 :             : 
    9386                 :             :                 case T_FuncExpr:
    9387                 :           0 :                         get_func_expr((FuncExpr *) node, context, showimplicit);
    9388                 :           0 :                         break;
    9389                 :             : 
    9390                 :             :                 case T_NamedArgExpr:
    9391                 :             :                         {
    9392                 :           0 :                                 NamedArgExpr *na = (NamedArgExpr *) node;
    9393                 :             : 
    9394                 :           0 :                                 appendStringInfo(buf, "%s => ", quote_identifier(na->name));
    9395                 :           0 :                                 get_rule_expr((Node *) na->arg, context, showimplicit);
    9396                 :           0 :                         }
    9397                 :           0 :                         break;
    9398                 :             : 
    9399                 :             :                 case T_OpExpr:
    9400                 :           0 :                         get_oper_expr((OpExpr *) node, context);
    9401                 :           0 :                         break;
    9402                 :             : 
    9403                 :             :                 case T_DistinctExpr:
    9404                 :             :                         {
    9405                 :           0 :                                 DistinctExpr *expr = (DistinctExpr *) node;
    9406                 :           0 :                                 List       *args = expr->args;
    9407                 :           0 :                                 Node       *arg1 = (Node *) linitial(args);
    9408                 :           0 :                                 Node       *arg2 = (Node *) lsecond(args);
    9409                 :             : 
    9410         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
    9411                 :           0 :                                         appendStringInfoChar(buf, '(');
    9412                 :           0 :                                 get_rule_expr_paren(arg1, context, true, node);
    9413                 :           0 :                                 appendStringInfoString(buf, " IS DISTINCT FROM ");
    9414                 :           0 :                                 get_rule_expr_paren(arg2, context, true, node);
    9415         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
    9416                 :           0 :                                         appendStringInfoChar(buf, ')');
    9417                 :           0 :                         }
    9418                 :           0 :                         break;
    9419                 :             : 
    9420                 :             :                 case T_NullIfExpr:
    9421                 :             :                         {
    9422                 :           0 :                                 NullIfExpr *nullifexpr = (NullIfExpr *) node;
    9423                 :             : 
    9424                 :           0 :                                 appendStringInfoString(buf, "NULLIF(");
    9425                 :           0 :                                 get_rule_expr((Node *) nullifexpr->args, context, true);
    9426                 :           0 :                                 appendStringInfoChar(buf, ')');
    9427                 :           0 :                         }
    9428                 :           0 :                         break;
    9429                 :             : 
    9430                 :             :                 case T_ScalarArrayOpExpr:
    9431                 :             :                         {
    9432                 :           0 :                                 ScalarArrayOpExpr *expr = (ScalarArrayOpExpr *) node;
    9433                 :           0 :                                 List       *args = expr->args;
    9434                 :           0 :                                 Node       *arg1 = (Node *) linitial(args);
    9435                 :           0 :                                 Node       *arg2 = (Node *) lsecond(args);
    9436                 :             : 
    9437         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
    9438                 :           0 :                                         appendStringInfoChar(buf, '(');
    9439                 :           0 :                                 get_rule_expr_paren(arg1, context, true, node);
    9440                 :           0 :                                 appendStringInfo(buf, " %s %s (",
    9441                 :           0 :                                                                  generate_operator_name(expr->opno,
    9442                 :           0 :                                                                                                                 exprType(arg1),
    9443                 :           0 :                                                                                                                 get_base_element_type(exprType(arg2))),
    9444                 :           0 :                                                                  expr->useOr ? "ANY" : "ALL");
    9445                 :           0 :                                 get_rule_expr_paren(arg2, context, true, node);
    9446                 :             : 
    9447                 :             :                                 /*
    9448                 :             :                                  * There's inherent ambiguity in "x op ANY/ALL (y)" when y is
    9449                 :             :                                  * a bare sub-SELECT.  Since we're here, the sub-SELECT must
    9450                 :             :                                  * be meant as a scalar sub-SELECT yielding an array value to
    9451                 :             :                                  * be used in ScalarArrayOpExpr; but the grammar will
    9452                 :             :                                  * preferentially interpret such a construct as an ANY/ALL
    9453                 :             :                                  * SubLink.  To prevent misparsing the output that way, insert
    9454                 :             :                                  * a dummy coercion (which will be stripped by parse analysis,
    9455                 :             :                                  * so no inefficiency is added in dump and reload).  This is
    9456                 :             :                                  * indeed most likely what the user wrote to get the construct
    9457                 :             :                                  * accepted in the first place.
    9458                 :             :                                  */
    9459   [ #  #  #  # ]:           0 :                                 if (IsA(arg2, SubLink) &&
    9460                 :           0 :                                         ((SubLink *) arg2)->subLinkType == EXPR_SUBLINK)
    9461                 :           0 :                                         appendStringInfo(buf, "::%s",
    9462                 :           0 :                                                                          format_type_with_typemod(exprType(arg2),
    9463                 :           0 :                                                                                                                           exprTypmod(arg2)));
    9464                 :           0 :                                 appendStringInfoChar(buf, ')');
    9465         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
    9466                 :           0 :                                         appendStringInfoChar(buf, ')');
    9467                 :           0 :                         }
    9468                 :           0 :                         break;
    9469                 :             : 
    9470                 :             :                 case T_BoolExpr:
    9471                 :             :                         {
    9472                 :           0 :                                 BoolExpr   *expr = (BoolExpr *) node;
    9473                 :           0 :                                 Node       *first_arg = linitial(expr->args);
    9474                 :           0 :                                 ListCell   *arg;
    9475                 :             : 
    9476   [ #  #  #  # ]:           0 :                                 switch (expr->boolop)
    9477                 :             :                                 {
    9478                 :             :                                         case AND_EXPR:
    9479         [ #  # ]:           0 :                                                 if (!PRETTY_PAREN(context))
    9480                 :           0 :                                                         appendStringInfoChar(buf, '(');
    9481                 :           0 :                                                 get_rule_expr_paren(first_arg, context,
    9482                 :           0 :                                                                                         false, node);
    9483   [ #  #  #  #  :           0 :                                                 for_each_from(arg, expr->args, 1)
                   #  # ]
    9484                 :             :                                                 {
    9485                 :           0 :                                                         appendStringInfoString(buf, " AND ");
    9486                 :           0 :                                                         get_rule_expr_paren((Node *) lfirst(arg), context,
    9487                 :           0 :                                                                                                 false, node);
    9488                 :           0 :                                                 }
    9489         [ #  # ]:           0 :                                                 if (!PRETTY_PAREN(context))
    9490                 :           0 :                                                         appendStringInfoChar(buf, ')');
    9491                 :           0 :                                                 break;
    9492                 :             : 
    9493                 :             :                                         case OR_EXPR:
    9494         [ #  # ]:           0 :                                                 if (!PRETTY_PAREN(context))
    9495                 :           0 :                                                         appendStringInfoChar(buf, '(');
    9496                 :           0 :                                                 get_rule_expr_paren(first_arg, context,
    9497                 :           0 :                                                                                         false, node);
    9498   [ #  #  #  #  :           0 :                                                 for_each_from(arg, expr->args, 1)
                   #  # ]
    9499                 :             :                                                 {
    9500                 :           0 :                                                         appendStringInfoString(buf, " OR ");
    9501                 :           0 :                                                         get_rule_expr_paren((Node *) lfirst(arg), context,
    9502                 :           0 :                                                                                                 false, node);
    9503                 :           0 :                                                 }
    9504         [ #  # ]:           0 :                                                 if (!PRETTY_PAREN(context))
    9505                 :           0 :                                                         appendStringInfoChar(buf, ')');
    9506                 :           0 :                                                 break;
    9507                 :             : 
    9508                 :             :                                         case NOT_EXPR:
    9509         [ #  # ]:           0 :                                                 if (!PRETTY_PAREN(context))
    9510                 :           0 :                                                         appendStringInfoChar(buf, '(');
    9511                 :           0 :                                                 appendStringInfoString(buf, "NOT ");
    9512                 :           0 :                                                 get_rule_expr_paren(first_arg, context,
    9513                 :           0 :                                                                                         false, node);
    9514         [ #  # ]:           0 :                                                 if (!PRETTY_PAREN(context))
    9515                 :           0 :                                                         appendStringInfoChar(buf, ')');
    9516                 :           0 :                                                 break;
    9517                 :             : 
    9518                 :             :                                         default:
    9519   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "unrecognized boolop: %d",
    9520                 :             :                                                          (int) expr->boolop);
    9521                 :           0 :                                 }
    9522                 :           0 :                         }
    9523                 :           0 :                         break;
    9524                 :             : 
    9525                 :             :                 case T_SubLink:
    9526                 :           0 :                         get_sublink_expr((SubLink *) node, context);
    9527                 :           0 :                         break;
    9528                 :             : 
    9529                 :             :                 case T_SubPlan:
    9530                 :             :                         {
    9531                 :           0 :                                 SubPlan    *subplan = (SubPlan *) node;
    9532                 :             : 
    9533                 :             :                                 /*
    9534                 :             :                                  * We cannot see an already-planned subplan in rule deparsing,
    9535                 :             :                                  * only while EXPLAINing a query plan.  We don't try to
    9536                 :             :                                  * reconstruct the original SQL, just reference the subplan
    9537                 :             :                                  * that appears elsewhere in EXPLAIN's result.  It does seem
    9538                 :             :                                  * useful to show the subLinkType and testexpr (if any), and
    9539                 :             :                                  * we also note whether the subplan will be hashed.
    9540                 :             :                                  */
    9541   [ #  #  #  #  :           0 :                                 switch (subplan->subLinkType)
             #  #  #  #  
                      # ]
    9542                 :             :                                 {
    9543                 :             :                                         case EXISTS_SUBLINK:
    9544                 :           0 :                                                 appendStringInfoString(buf, "EXISTS(");
    9545         [ #  # ]:           0 :                                                 Assert(subplan->testexpr == NULL);
    9546                 :           0 :                                                 break;
    9547                 :             :                                         case ALL_SUBLINK:
    9548                 :           0 :                                                 appendStringInfoString(buf, "(ALL ");
    9549         [ #  # ]:           0 :                                                 Assert(subplan->testexpr != NULL);
    9550                 :           0 :                                                 break;
    9551                 :             :                                         case ANY_SUBLINK:
    9552                 :           0 :                                                 appendStringInfoString(buf, "(ANY ");
    9553         [ #  # ]:           0 :                                                 Assert(subplan->testexpr != NULL);
    9554                 :           0 :                                                 break;
    9555                 :             :                                         case ROWCOMPARE_SUBLINK:
    9556                 :             :                                                 /* Parenthesizing the testexpr seems sufficient */
    9557                 :           0 :                                                 appendStringInfoChar(buf, '(');
    9558         [ #  # ]:           0 :                                                 Assert(subplan->testexpr != NULL);
    9559                 :           0 :                                                 break;
    9560                 :             :                                         case EXPR_SUBLINK:
    9561                 :             :                                                 /* No need to decorate these subplan references */
    9562                 :           0 :                                                 appendStringInfoChar(buf, '(');
    9563         [ #  # ]:           0 :                                                 Assert(subplan->testexpr == NULL);
    9564                 :           0 :                                                 break;
    9565                 :             :                                         case MULTIEXPR_SUBLINK:
    9566                 :             :                                                 /* MULTIEXPR isn't executed in the normal way */
    9567                 :           0 :                                                 appendStringInfoString(buf, "(rescan ");
    9568         [ #  # ]:           0 :                                                 Assert(subplan->testexpr == NULL);
    9569                 :           0 :                                                 break;
    9570                 :             :                                         case ARRAY_SUBLINK:
    9571                 :           0 :                                                 appendStringInfoString(buf, "ARRAY(");
    9572         [ #  # ]:           0 :                                                 Assert(subplan->testexpr == NULL);
    9573                 :           0 :                                                 break;
    9574                 :             :                                         case CTE_SUBLINK:
    9575                 :             :                                                 /* This case is unreachable within expressions */
    9576                 :           0 :                                                 appendStringInfoString(buf, "CTE(");
    9577         [ #  # ]:           0 :                                                 Assert(subplan->testexpr == NULL);
    9578                 :           0 :                                                 break;
    9579                 :             :                                 }
    9580                 :             : 
    9581         [ #  # ]:           0 :                                 if (subplan->testexpr != NULL)
    9582                 :             :                                 {
    9583                 :           0 :                                         deparse_namespace *dpns;
    9584                 :             : 
    9585                 :             :                                         /*
    9586                 :             :                                          * Push SubPlan into ancestors list while deparsing
    9587                 :             :                                          * testexpr, so that we can handle PARAM_EXEC references
    9588                 :             :                                          * to the SubPlan's paramIds.  (This makes it look like
    9589                 :             :                                          * the SubPlan is an "ancestor" of the current plan node,
    9590                 :             :                                          * which is a little weird, but it does no harm.)  In this
    9591                 :             :                                          * path, we don't need to mention the SubPlan explicitly,
    9592                 :             :                                          * because the referencing Params will show its existence.
    9593                 :             :                                          */
    9594                 :           0 :                                         dpns = (deparse_namespace *) linitial(context->namespaces);
    9595                 :           0 :                                         dpns->ancestors = lcons(subplan, dpns->ancestors);
    9596                 :             : 
    9597                 :           0 :                                         get_rule_expr(subplan->testexpr, context, showimplicit);
    9598                 :           0 :                                         appendStringInfoChar(buf, ')');
    9599                 :             : 
    9600                 :           0 :                                         dpns->ancestors = list_delete_first(dpns->ancestors);
    9601                 :           0 :                                 }
    9602                 :             :                                 else
    9603                 :             :                                 {
    9604                 :           0 :                                         const char *nameprefix;
    9605                 :             : 
    9606                 :             :                                         /* No referencing Params, so show the SubPlan's name */
    9607         [ #  # ]:           0 :                                         if (subplan->isInitPlan)
    9608                 :           0 :                                                 nameprefix = "InitPlan ";
    9609                 :             :                                         else
    9610                 :           0 :                                                 nameprefix = "SubPlan ";
    9611         [ #  # ]:           0 :                                         if (subplan->useHashTable)
    9612                 :           0 :                                                 appendStringInfo(buf, "hashed %s%s)",
    9613                 :           0 :                                                                                  nameprefix, subplan->plan_name);
    9614                 :             :                                         else
    9615                 :           0 :                                                 appendStringInfo(buf, "%s%s)",
    9616                 :           0 :                                                                                  nameprefix, subplan->plan_name);
    9617                 :           0 :                                 }
    9618                 :           0 :                         }
    9619                 :           0 :                         break;
    9620                 :             : 
    9621                 :             :                 case T_AlternativeSubPlan:
    9622                 :             :                         {
    9623                 :           0 :                                 AlternativeSubPlan *asplan = (AlternativeSubPlan *) node;
    9624                 :           0 :                                 ListCell   *lc;
    9625                 :             : 
    9626                 :             :                                 /*
    9627                 :             :                                  * This case cannot be reached in normal usage, since no
    9628                 :             :                                  * AlternativeSubPlan can appear either in parsetrees or
    9629                 :             :                                  * finished plan trees.  We keep it just in case somebody
    9630                 :             :                                  * wants to use this code to print planner data structures.
    9631                 :             :                                  */
    9632                 :           0 :                                 appendStringInfoString(buf, "(alternatives: ");
    9633   [ #  #  #  #  :           0 :                                 foreach(lc, asplan->subplans)
                   #  # ]
    9634                 :             :                                 {
    9635                 :           0 :                                         SubPlan    *splan = lfirst_node(SubPlan, lc);
    9636                 :           0 :                                         const char *nameprefix;
    9637                 :             : 
    9638         [ #  # ]:           0 :                                         if (splan->isInitPlan)
    9639                 :           0 :                                                 nameprefix = "InitPlan ";
    9640                 :             :                                         else
    9641                 :           0 :                                                 nameprefix = "SubPlan ";
    9642         [ #  # ]:           0 :                                         if (splan->useHashTable)
    9643                 :           0 :                                                 appendStringInfo(buf, "hashed %s%s", nameprefix,
    9644                 :           0 :                                                                                  splan->plan_name);
    9645                 :             :                                         else
    9646                 :           0 :                                                 appendStringInfo(buf, "%s%s", nameprefix,
    9647                 :           0 :                                                                                  splan->plan_name);
    9648         [ #  # ]:           0 :                                         if (lnext(asplan->subplans, lc))
    9649                 :           0 :                                                 appendStringInfoString(buf, " or ");
    9650                 :           0 :                                 }
    9651                 :           0 :                                 appendStringInfoChar(buf, ')');
    9652                 :           0 :                         }
    9653                 :           0 :                         break;
    9654                 :             : 
    9655                 :             :                 case T_FieldSelect:
    9656                 :             :                         {
    9657                 :           0 :                                 FieldSelect *fselect = (FieldSelect *) node;
    9658                 :           0 :                                 Node       *arg = (Node *) fselect->arg;
    9659                 :           0 :                                 int                     fno = fselect->fieldnum;
    9660                 :           0 :                                 const char *fieldname;
    9661                 :           0 :                                 bool            need_parens;
    9662                 :             : 
    9663                 :             :                                 /*
    9664                 :             :                                  * Parenthesize the argument unless it's a SubscriptingRef or
    9665                 :             :                                  * another FieldSelect.  Note in particular that it would be
    9666                 :             :                                  * WRONG to not parenthesize a Var argument; simplicity is not
    9667                 :             :                                  * the issue here, having the right number of names is.
    9668                 :             :                                  */
    9669         [ #  # ]:           0 :                                 need_parens = !IsA(arg, SubscriptingRef) &&
    9670                 :           0 :                                         !IsA(arg, FieldSelect);
    9671         [ #  # ]:           0 :                                 if (need_parens)
    9672                 :           0 :                                         appendStringInfoChar(buf, '(');
    9673                 :           0 :                                 get_rule_expr(arg, context, true);
    9674         [ #  # ]:           0 :                                 if (need_parens)
    9675                 :           0 :                                         appendStringInfoChar(buf, ')');
    9676                 :             : 
    9677                 :             :                                 /*
    9678                 :             :                                  * Get and print the field name.
    9679                 :             :                                  */
    9680                 :           0 :                                 fieldname = get_name_for_var_field((Var *) arg, fno,
    9681                 :           0 :                                                                                                    0, context);
    9682                 :           0 :                                 appendStringInfo(buf, ".%s", quote_identifier(fieldname));
    9683                 :           0 :                         }
    9684                 :           0 :                         break;
    9685                 :             : 
    9686                 :             :                 case T_FieldStore:
    9687                 :             :                         {
    9688                 :           0 :                                 FieldStore *fstore = (FieldStore *) node;
    9689                 :           0 :                                 bool            need_parens;
    9690                 :             : 
    9691                 :             :                                 /*
    9692                 :             :                                  * There is no good way to represent a FieldStore as real SQL,
    9693                 :             :                                  * so decompilation of INSERT or UPDATE statements should
    9694                 :             :                                  * always use processIndirection as part of the
    9695                 :             :                                  * statement-level syntax.  We should only get here when
    9696                 :             :                                  * EXPLAIN tries to print the targetlist of a plan resulting
    9697                 :             :                                  * from such a statement.  The plan case is even harder than
    9698                 :             :                                  * ordinary rules would be, because the planner tries to
    9699                 :             :                                  * collapse multiple assignments to the same field or subfield
    9700                 :             :                                  * into one FieldStore; so we can see a list of target fields
    9701                 :             :                                  * not just one, and the arguments could be FieldStores
    9702                 :             :                                  * themselves.  We don't bother to try to print the target
    9703                 :             :                                  * field names; we just print the source arguments, with a
    9704                 :             :                                  * ROW() around them if there's more than one.  This isn't
    9705                 :             :                                  * terribly complete, but it's probably good enough for
    9706                 :             :                                  * EXPLAIN's purposes; especially since anything more would be
    9707                 :             :                                  * either hopelessly confusing or an even poorer
    9708                 :             :                                  * representation of what the plan is actually doing.
    9709                 :             :                                  */
    9710                 :           0 :                                 need_parens = (list_length(fstore->newvals) != 1);
    9711         [ #  # ]:           0 :                                 if (need_parens)
    9712                 :           0 :                                         appendStringInfoString(buf, "ROW(");
    9713                 :           0 :                                 get_rule_expr((Node *) fstore->newvals, context, showimplicit);
    9714         [ #  # ]:           0 :                                 if (need_parens)
    9715                 :           0 :                                         appendStringInfoChar(buf, ')');
    9716                 :           0 :                         }
    9717                 :           0 :                         break;
    9718                 :             : 
    9719                 :             :                 case T_RelabelType:
    9720                 :             :                         {
    9721                 :           0 :                                 RelabelType *relabel = (RelabelType *) node;
    9722                 :           0 :                                 Node       *arg = (Node *) relabel->arg;
    9723                 :             : 
    9724   [ #  #  #  # ]:           0 :                                 if (relabel->relabelformat == COERCE_IMPLICIT_CAST &&
    9725                 :           0 :                                         !showimplicit)
    9726                 :             :                                 {
    9727                 :             :                                         /* don't show the implicit cast */
    9728                 :           0 :                                         get_rule_expr_paren(arg, context, false, node);
    9729                 :           0 :                                 }
    9730                 :             :                                 else
    9731                 :             :                                 {
    9732                 :           0 :                                         get_coercion_expr(arg, context,
    9733                 :           0 :                                                                           relabel->resulttype,
    9734                 :           0 :                                                                           relabel->resulttypmod,
    9735                 :           0 :                                                                           node);
    9736                 :             :                                 }
    9737                 :           0 :                         }
    9738                 :           0 :                         break;
    9739                 :             : 
    9740                 :             :                 case T_CoerceViaIO:
    9741                 :             :                         {
    9742                 :           0 :                                 CoerceViaIO *iocoerce = (CoerceViaIO *) node;
    9743                 :           0 :                                 Node       *arg = (Node *) iocoerce->arg;
    9744                 :             : 
    9745   [ #  #  #  # ]:           0 :                                 if (iocoerce->coerceformat == COERCE_IMPLICIT_CAST &&
    9746                 :           0 :                                         !showimplicit)
    9747                 :             :                                 {
    9748                 :             :                                         /* don't show the implicit cast */
    9749                 :           0 :                                         get_rule_expr_paren(arg, context, false, node);
    9750                 :           0 :                                 }
    9751                 :             :                                 else
    9752                 :             :                                 {
    9753                 :           0 :                                         get_coercion_expr(arg, context,
    9754                 :           0 :                                                                           iocoerce->resulttype,
    9755                 :             :                                                                           -1,
    9756                 :           0 :                                                                           node);
    9757                 :             :                                 }
    9758                 :           0 :                         }
    9759                 :           0 :                         break;
    9760                 :             : 
    9761                 :             :                 case T_ArrayCoerceExpr:
    9762                 :             :                         {
    9763                 :           0 :                                 ArrayCoerceExpr *acoerce = (ArrayCoerceExpr *) node;
    9764                 :           0 :                                 Node       *arg = (Node *) acoerce->arg;
    9765                 :             : 
    9766   [ #  #  #  # ]:           0 :                                 if (acoerce->coerceformat == COERCE_IMPLICIT_CAST &&
    9767                 :           0 :                                         !showimplicit)
    9768                 :             :                                 {
    9769                 :             :                                         /* don't show the implicit cast */
    9770                 :           0 :                                         get_rule_expr_paren(arg, context, false, node);
    9771                 :           0 :                                 }
    9772                 :             :                                 else
    9773                 :             :                                 {
    9774                 :           0 :                                         get_coercion_expr(arg, context,
    9775                 :           0 :                                                                           acoerce->resulttype,
    9776                 :           0 :                                                                           acoerce->resulttypmod,
    9777                 :           0 :                                                                           node);
    9778                 :             :                                 }
    9779                 :           0 :                         }
    9780                 :           0 :                         break;
    9781                 :             : 
    9782                 :             :                 case T_ConvertRowtypeExpr:
    9783                 :             :                         {
    9784                 :           0 :                                 ConvertRowtypeExpr *convert = (ConvertRowtypeExpr *) node;
    9785                 :           0 :                                 Node       *arg = (Node *) convert->arg;
    9786                 :             : 
    9787   [ #  #  #  # ]:           0 :                                 if (convert->convertformat == COERCE_IMPLICIT_CAST &&
    9788                 :           0 :                                         !showimplicit)
    9789                 :             :                                 {
    9790                 :             :                                         /* don't show the implicit cast */
    9791                 :           0 :                                         get_rule_expr_paren(arg, context, false, node);
    9792                 :           0 :                                 }
    9793                 :             :                                 else
    9794                 :             :                                 {
    9795                 :           0 :                                         get_coercion_expr(arg, context,
    9796                 :           0 :                                                                           convert->resulttype, -1,
    9797                 :           0 :                                                                           node);
    9798                 :             :                                 }
    9799                 :           0 :                         }
    9800                 :           0 :                         break;
    9801                 :             : 
    9802                 :             :                 case T_CollateExpr:
    9803                 :             :                         {
    9804                 :           0 :                                 CollateExpr *collate = (CollateExpr *) node;
    9805                 :           0 :                                 Node       *arg = (Node *) collate->arg;
    9806                 :             : 
    9807         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
    9808                 :           0 :                                         appendStringInfoChar(buf, '(');
    9809                 :           0 :                                 get_rule_expr_paren(arg, context, showimplicit, node);
    9810                 :           0 :                                 appendStringInfo(buf, " COLLATE %s",
    9811                 :           0 :                                                                  generate_collation_name(collate->collOid));
    9812         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
    9813                 :           0 :                                         appendStringInfoChar(buf, ')');
    9814                 :           0 :                         }
    9815                 :           0 :                         break;
    9816                 :             : 
    9817                 :             :                 case T_CaseExpr:
    9818                 :             :                         {
    9819                 :           0 :                                 CaseExpr   *caseexpr = (CaseExpr *) node;
    9820                 :           0 :                                 ListCell   *temp;
    9821                 :             : 
    9822                 :           0 :                                 appendContextKeyword(context, "CASE",
    9823                 :             :                                                                          0, PRETTYINDENT_VAR, 0);
    9824         [ #  # ]:           0 :                                 if (caseexpr->arg)
    9825                 :             :                                 {
    9826                 :           0 :                                         appendStringInfoChar(buf, ' ');
    9827                 :           0 :                                         get_rule_expr((Node *) caseexpr->arg, context, true);
    9828                 :           0 :                                 }
    9829   [ #  #  #  #  :           0 :                                 foreach(temp, caseexpr->args)
                   #  # ]
    9830                 :             :                                 {
    9831                 :           0 :                                         CaseWhen   *when = (CaseWhen *) lfirst(temp);
    9832                 :           0 :                                         Node       *w = (Node *) when->expr;
    9833                 :             : 
    9834         [ #  # ]:           0 :                                         if (caseexpr->arg)
    9835                 :             :                                         {
    9836                 :             :                                                 /*
    9837                 :             :                                                  * The parser should have produced WHEN clauses of the
    9838                 :             :                                                  * form "CaseTestExpr = RHS", possibly with an
    9839                 :             :                                                  * implicit coercion inserted above the CaseTestExpr.
    9840                 :             :                                                  * For accurate decompilation of rules it's essential
    9841                 :             :                                                  * that we show just the RHS.  However in an
    9842                 :             :                                                  * expression that's been through the optimizer, the
    9843                 :             :                                                  * WHEN clause could be almost anything (since the
    9844                 :             :                                                  * equality operator could have been expanded into an
    9845                 :             :                                                  * inline function).  If we don't recognize the form
    9846                 :             :                                                  * of the WHEN clause, just punt and display it as-is.
    9847                 :             :                                                  */
    9848         [ #  # ]:           0 :                                                 if (IsA(w, OpExpr))
    9849                 :             :                                                 {
    9850                 :           0 :                                                         List       *args = ((OpExpr *) w)->args;
    9851                 :             : 
    9852   [ #  #  #  # ]:           0 :                                                         if (list_length(args) == 2 &&
    9853                 :           0 :                                                                 IsA(strip_implicit_coercions(linitial(args)),
    9854                 :             :                                                                         CaseTestExpr))
    9855                 :           0 :                                                                 w = (Node *) lsecond(args);
    9856                 :           0 :                                                 }
    9857                 :           0 :                                         }
    9858                 :             : 
    9859         [ #  # ]:           0 :                                         if (!PRETTY_INDENT(context))
    9860                 :           0 :                                                 appendStringInfoChar(buf, ' ');
    9861                 :           0 :                                         appendContextKeyword(context, "WHEN ",
    9862                 :             :                                                                                  0, 0, 0);
    9863                 :           0 :                                         get_rule_expr(w, context, false);
    9864                 :           0 :                                         appendStringInfoString(buf, " THEN ");
    9865                 :           0 :                                         get_rule_expr((Node *) when->result, context, true);
    9866                 :           0 :                                 }
    9867         [ #  # ]:           0 :                                 if (!PRETTY_INDENT(context))
    9868                 :           0 :                                         appendStringInfoChar(buf, ' ');
    9869                 :           0 :                                 appendContextKeyword(context, "ELSE ",
    9870                 :             :                                                                          0, 0, 0);
    9871                 :           0 :                                 get_rule_expr((Node *) caseexpr->defresult, context, true);
    9872         [ #  # ]:           0 :                                 if (!PRETTY_INDENT(context))
    9873                 :           0 :                                         appendStringInfoChar(buf, ' ');
    9874                 :           0 :                                 appendContextKeyword(context, "END",
    9875                 :             :                                                                          -PRETTYINDENT_VAR, 0, 0);
    9876                 :           0 :                         }
    9877                 :           0 :                         break;
    9878                 :             : 
    9879                 :             :                 case T_CaseTestExpr:
    9880                 :             :                         {
    9881                 :             :                                 /*
    9882                 :             :                                  * Normally we should never get here, since for expressions
    9883                 :             :                                  * that can contain this node type we attempt to avoid
    9884                 :             :                                  * recursing to it.  But in an optimized expression we might
    9885                 :             :                                  * be unable to avoid that (see comments for CaseExpr).  If we
    9886                 :             :                                  * do see one, print it as CASE_TEST_EXPR.
    9887                 :             :                                  */
    9888                 :           0 :                                 appendStringInfoString(buf, "CASE_TEST_EXPR");
    9889                 :             :                         }
    9890                 :           0 :                         break;
    9891                 :             : 
    9892                 :             :                 case T_ArrayExpr:
    9893                 :             :                         {
    9894                 :           0 :                                 ArrayExpr  *arrayexpr = (ArrayExpr *) node;
    9895                 :             : 
    9896                 :           0 :                                 appendStringInfoString(buf, "ARRAY[");
    9897                 :           0 :                                 get_rule_expr((Node *) arrayexpr->elements, context, true);
    9898                 :           0 :                                 appendStringInfoChar(buf, ']');
    9899                 :             : 
    9900                 :             :                                 /*
    9901                 :             :                                  * If the array isn't empty, we assume its elements are
    9902                 :             :                                  * coerced to the desired type.  If it's empty, though, we
    9903                 :             :                                  * need an explicit coercion to the array type.
    9904                 :             :                                  */
    9905         [ #  # ]:           0 :                                 if (arrayexpr->elements == NIL)
    9906                 :           0 :                                         appendStringInfo(buf, "::%s",
    9907                 :           0 :                                                                          format_type_with_typemod(arrayexpr->array_typeid, -1));
    9908                 :           0 :                         }
    9909                 :           0 :                         break;
    9910                 :             : 
    9911                 :             :                 case T_RowExpr:
    9912                 :             :                         {
    9913                 :           0 :                                 RowExpr    *rowexpr = (RowExpr *) node;
    9914                 :           0 :                                 TupleDesc       tupdesc = NULL;
    9915                 :           0 :                                 ListCell   *arg;
    9916                 :           0 :                                 int                     i;
    9917                 :           0 :                                 char       *sep;
    9918                 :             : 
    9919                 :             :                                 /*
    9920                 :             :                                  * If it's a named type and not RECORD, we may have to skip
    9921                 :             :                                  * dropped columns and/or claim there are NULLs for added
    9922                 :             :                                  * columns.
    9923                 :             :                                  */
    9924         [ #  # ]:           0 :                                 if (rowexpr->row_typeid != RECORDOID)
    9925                 :             :                                 {
    9926                 :           0 :                                         tupdesc = lookup_rowtype_tupdesc(rowexpr->row_typeid, -1);
    9927         [ #  # ]:           0 :                                         Assert(list_length(rowexpr->args) <= tupdesc->natts);
    9928                 :           0 :                                 }
    9929                 :             : 
    9930                 :             :                                 /*
    9931                 :             :                                  * SQL99 allows "ROW" to be omitted when there is more than
    9932                 :             :                                  * one column, but for simplicity we always print it.
    9933                 :             :                                  */
    9934                 :           0 :                                 appendStringInfoString(buf, "ROW(");
    9935                 :           0 :                                 sep = "";
    9936                 :           0 :                                 i = 0;
    9937   [ #  #  #  #  :           0 :                                 foreach(arg, rowexpr->args)
                   #  # ]
    9938                 :             :                                 {
    9939                 :           0 :                                         Node       *e = (Node *) lfirst(arg);
    9940                 :             : 
    9941   [ #  #  #  # ]:           0 :                                         if (tupdesc == NULL ||
    9942                 :           0 :                                                 !TupleDescCompactAttr(tupdesc, i)->attisdropped)
    9943                 :             :                                         {
    9944                 :           0 :                                                 appendStringInfoString(buf, sep);
    9945                 :             :                                                 /* Whole-row Vars need special treatment here */
    9946                 :           0 :                                                 get_rule_expr_toplevel(e, context, true);
    9947                 :           0 :                                                 sep = ", ";
    9948                 :           0 :                                         }
    9949                 :           0 :                                         i++;
    9950                 :           0 :                                 }
    9951         [ #  # ]:           0 :                                 if (tupdesc != NULL)
    9952                 :             :                                 {
    9953         [ #  # ]:           0 :                                         while (i < tupdesc->natts)
    9954                 :             :                                         {
    9955         [ #  # ]:           0 :                                                 if (!TupleDescCompactAttr(tupdesc, i)->attisdropped)
    9956                 :             :                                                 {
    9957                 :           0 :                                                         appendStringInfoString(buf, sep);
    9958                 :           0 :                                                         appendStringInfoString(buf, "NULL");
    9959                 :           0 :                                                         sep = ", ";
    9960                 :           0 :                                                 }
    9961                 :           0 :                                                 i++;
    9962                 :             :                                         }
    9963                 :             : 
    9964         [ #  # ]:           0 :                                         ReleaseTupleDesc(tupdesc);
    9965                 :           0 :                                 }
    9966                 :           0 :                                 appendStringInfoChar(buf, ')');
    9967         [ #  # ]:           0 :                                 if (rowexpr->row_format == COERCE_EXPLICIT_CAST)
    9968                 :           0 :                                         appendStringInfo(buf, "::%s",
    9969                 :           0 :                                                                          format_type_with_typemod(rowexpr->row_typeid, -1));
    9970                 :           0 :                         }
    9971                 :           0 :                         break;
    9972                 :             : 
    9973                 :             :                 case T_RowCompareExpr:
    9974                 :             :                         {
    9975                 :           0 :                                 RowCompareExpr *rcexpr = (RowCompareExpr *) node;
    9976                 :             : 
    9977                 :             :                                 /*
    9978                 :             :                                  * SQL99 allows "ROW" to be omitted when there is more than
    9979                 :             :                                  * one column, but for simplicity we always print it.  Within
    9980                 :             :                                  * a ROW expression, whole-row Vars need special treatment, so
    9981                 :             :                                  * use get_rule_list_toplevel.
    9982                 :             :                                  */
    9983                 :           0 :                                 appendStringInfoString(buf, "(ROW(");
    9984                 :           0 :                                 get_rule_list_toplevel(rcexpr->largs, context, true);
    9985                 :             : 
    9986                 :             :                                 /*
    9987                 :             :                                  * We assume that the name of the first-column operator will
    9988                 :             :                                  * do for all the rest too.  This is definitely open to
    9989                 :             :                                  * failure, eg if some but not all operators were renamed
    9990                 :             :                                  * since the construct was parsed, but there seems no way to
    9991                 :             :                                  * be perfect.
    9992                 :             :                                  */
    9993                 :           0 :                                 appendStringInfo(buf, ") %s ROW(",
    9994                 :           0 :                                                                  generate_operator_name(linitial_oid(rcexpr->opnos),
    9995                 :           0 :                                                                                                                 exprType(linitial(rcexpr->largs)),
    9996                 :           0 :                                                                                                                 exprType(linitial(rcexpr->rargs))));
    9997                 :           0 :                                 get_rule_list_toplevel(rcexpr->rargs, context, true);
    9998                 :           0 :                                 appendStringInfoString(buf, "))");
    9999                 :           0 :                         }
   10000                 :           0 :                         break;
   10001                 :             : 
   10002                 :             :                 case T_CoalesceExpr:
   10003                 :             :                         {
   10004                 :           0 :                                 CoalesceExpr *coalesceexpr = (CoalesceExpr *) node;
   10005                 :             : 
   10006                 :           0 :                                 appendStringInfoString(buf, "COALESCE(");
   10007                 :           0 :                                 get_rule_expr((Node *) coalesceexpr->args, context, true);
   10008                 :           0 :                                 appendStringInfoChar(buf, ')');
   10009                 :           0 :                         }
   10010                 :           0 :                         break;
   10011                 :             : 
   10012                 :             :                 case T_MinMaxExpr:
   10013                 :             :                         {
   10014                 :           0 :                                 MinMaxExpr *minmaxexpr = (MinMaxExpr *) node;
   10015                 :             : 
   10016      [ #  #  # ]:           0 :                                 switch (minmaxexpr->op)
   10017                 :             :                                 {
   10018                 :             :                                         case IS_GREATEST:
   10019                 :           0 :                                                 appendStringInfoString(buf, "GREATEST(");
   10020                 :           0 :                                                 break;
   10021                 :             :                                         case IS_LEAST:
   10022                 :           0 :                                                 appendStringInfoString(buf, "LEAST(");
   10023                 :           0 :                                                 break;
   10024                 :             :                                 }
   10025                 :           0 :                                 get_rule_expr((Node *) minmaxexpr->args, context, true);
   10026                 :           0 :                                 appendStringInfoChar(buf, ')');
   10027                 :           0 :                         }
   10028                 :           0 :                         break;
   10029                 :             : 
   10030                 :             :                 case T_SQLValueFunction:
   10031                 :             :                         {
   10032                 :           0 :                                 SQLValueFunction *svf = (SQLValueFunction *) node;
   10033                 :             : 
   10034                 :             :                                 /*
   10035                 :             :                                  * Note: this code knows that typmod for time, timestamp, and
   10036                 :             :                                  * timestamptz just prints as integer.
   10037                 :             :                                  */
   10038   [ #  #  #  #  :           0 :                                 switch (svf->op)
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
   10039                 :             :                                 {
   10040                 :             :                                         case SVFOP_CURRENT_DATE:
   10041                 :           0 :                                                 appendStringInfoString(buf, "CURRENT_DATE");
   10042                 :           0 :                                                 break;
   10043                 :             :                                         case SVFOP_CURRENT_TIME:
   10044                 :           0 :                                                 appendStringInfoString(buf, "CURRENT_TIME");
   10045                 :           0 :                                                 break;
   10046                 :             :                                         case SVFOP_CURRENT_TIME_N:
   10047                 :           0 :                                                 appendStringInfo(buf, "CURRENT_TIME(%d)", svf->typmod);
   10048                 :           0 :                                                 break;
   10049                 :             :                                         case SVFOP_CURRENT_TIMESTAMP:
   10050                 :           0 :                                                 appendStringInfoString(buf, "CURRENT_TIMESTAMP");
   10051                 :           0 :                                                 break;
   10052                 :             :                                         case SVFOP_CURRENT_TIMESTAMP_N:
   10053                 :           0 :                                                 appendStringInfo(buf, "CURRENT_TIMESTAMP(%d)",
   10054                 :           0 :                                                                                  svf->typmod);
   10055                 :           0 :                                                 break;
   10056                 :             :                                         case SVFOP_LOCALTIME:
   10057                 :           0 :                                                 appendStringInfoString(buf, "LOCALTIME");
   10058                 :           0 :                                                 break;
   10059                 :             :                                         case SVFOP_LOCALTIME_N:
   10060                 :           0 :                                                 appendStringInfo(buf, "LOCALTIME(%d)", svf->typmod);
   10061                 :           0 :                                                 break;
   10062                 :             :                                         case SVFOP_LOCALTIMESTAMP:
   10063                 :           0 :                                                 appendStringInfoString(buf, "LOCALTIMESTAMP");
   10064                 :           0 :                                                 break;
   10065                 :             :                                         case SVFOP_LOCALTIMESTAMP_N:
   10066                 :           0 :                                                 appendStringInfo(buf, "LOCALTIMESTAMP(%d)",
   10067                 :           0 :                                                                                  svf->typmod);
   10068                 :           0 :                                                 break;
   10069                 :             :                                         case SVFOP_CURRENT_ROLE:
   10070                 :           0 :                                                 appendStringInfoString(buf, "CURRENT_ROLE");
   10071                 :           0 :                                                 break;
   10072                 :             :                                         case SVFOP_CURRENT_USER:
   10073                 :           0 :                                                 appendStringInfoString(buf, "CURRENT_USER");
   10074                 :           0 :                                                 break;
   10075                 :             :                                         case SVFOP_USER:
   10076                 :           0 :                                                 appendStringInfoString(buf, "USER");
   10077                 :           0 :                                                 break;
   10078                 :             :                                         case SVFOP_SESSION_USER:
   10079                 :           0 :                                                 appendStringInfoString(buf, "SESSION_USER");
   10080                 :           0 :                                                 break;
   10081                 :             :                                         case SVFOP_CURRENT_CATALOG:
   10082                 :           0 :                                                 appendStringInfoString(buf, "CURRENT_CATALOG");
   10083                 :           0 :                                                 break;
   10084                 :             :                                         case SVFOP_CURRENT_SCHEMA:
   10085                 :           0 :                                                 appendStringInfoString(buf, "CURRENT_SCHEMA");
   10086                 :           0 :                                                 break;
   10087                 :             :                                 }
   10088                 :           0 :                         }
   10089                 :           0 :                         break;
   10090                 :             : 
   10091                 :             :                 case T_XmlExpr:
   10092                 :             :                         {
   10093                 :           0 :                                 XmlExpr    *xexpr = (XmlExpr *) node;
   10094                 :           0 :                                 bool            needcomma = false;
   10095                 :           0 :                                 ListCell   *arg;
   10096                 :           0 :                                 ListCell   *narg;
   10097                 :           0 :                                 Const      *con;
   10098                 :             : 
   10099   [ #  #  #  #  :           0 :                                 switch (xexpr->op)
             #  #  #  # ]
   10100                 :             :                                 {
   10101                 :             :                                         case IS_XMLCONCAT:
   10102                 :           0 :                                                 appendStringInfoString(buf, "XMLCONCAT(");
   10103                 :           0 :                                                 break;
   10104                 :             :                                         case IS_XMLELEMENT:
   10105                 :           0 :                                                 appendStringInfoString(buf, "XMLELEMENT(");
   10106                 :           0 :                                                 break;
   10107                 :             :                                         case IS_XMLFOREST:
   10108                 :           0 :                                                 appendStringInfoString(buf, "XMLFOREST(");
   10109                 :           0 :                                                 break;
   10110                 :             :                                         case IS_XMLPARSE:
   10111                 :           0 :                                                 appendStringInfoString(buf, "XMLPARSE(");
   10112                 :           0 :                                                 break;
   10113                 :             :                                         case IS_XMLPI:
   10114                 :           0 :                                                 appendStringInfoString(buf, "XMLPI(");
   10115                 :           0 :                                                 break;
   10116                 :             :                                         case IS_XMLROOT:
   10117                 :           0 :                                                 appendStringInfoString(buf, "XMLROOT(");
   10118                 :           0 :                                                 break;
   10119                 :             :                                         case IS_XMLSERIALIZE:
   10120                 :           0 :                                                 appendStringInfoString(buf, "XMLSERIALIZE(");
   10121                 :           0 :                                                 break;
   10122                 :             :                                         case IS_DOCUMENT:
   10123                 :             :                                                 break;
   10124                 :             :                                 }
   10125   [ #  #  #  # ]:           0 :                                 if (xexpr->op == IS_XMLPARSE || xexpr->op == IS_XMLSERIALIZE)
   10126                 :             :                                 {
   10127         [ #  # ]:           0 :                                         if (xexpr->xmloption == XMLOPTION_DOCUMENT)
   10128                 :           0 :                                                 appendStringInfoString(buf, "DOCUMENT ");
   10129                 :             :                                         else
   10130                 :           0 :                                                 appendStringInfoString(buf, "CONTENT ");
   10131                 :           0 :                                 }
   10132         [ #  # ]:           0 :                                 if (xexpr->name)
   10133                 :             :                                 {
   10134                 :           0 :                                         appendStringInfo(buf, "NAME %s",
   10135                 :           0 :                                                                          quote_identifier(map_xml_name_to_sql_identifier(xexpr->name)));
   10136                 :           0 :                                         needcomma = true;
   10137                 :           0 :                                 }
   10138         [ #  # ]:           0 :                                 if (xexpr->named_args)
   10139                 :             :                                 {
   10140         [ #  # ]:           0 :                                         if (xexpr->op != IS_XMLFOREST)
   10141                 :             :                                         {
   10142         [ #  # ]:           0 :                                                 if (needcomma)
   10143                 :           0 :                                                         appendStringInfoString(buf, ", ");
   10144                 :           0 :                                                 appendStringInfoString(buf, "XMLATTRIBUTES(");
   10145                 :           0 :                                                 needcomma = false;
   10146                 :           0 :                                         }
   10147   [ #  #  #  #  :           0 :                                         forboth(arg, xexpr->named_args, narg, xexpr->arg_names)
          #  #  #  #  #  
                #  #  # ]
   10148                 :             :                                         {
   10149                 :           0 :                                                 Node       *e = (Node *) lfirst(arg);
   10150                 :           0 :                                                 char       *argname = strVal(lfirst(narg));
   10151                 :             : 
   10152         [ #  # ]:           0 :                                                 if (needcomma)
   10153                 :           0 :                                                         appendStringInfoString(buf, ", ");
   10154                 :           0 :                                                 get_rule_expr(e, context, true);
   10155                 :           0 :                                                 appendStringInfo(buf, " AS %s",
   10156                 :           0 :                                                                                  quote_identifier(map_xml_name_to_sql_identifier(argname)));
   10157                 :           0 :                                                 needcomma = true;
   10158                 :           0 :                                         }
   10159         [ #  # ]:           0 :                                         if (xexpr->op != IS_XMLFOREST)
   10160                 :           0 :                                                 appendStringInfoChar(buf, ')');
   10161                 :           0 :                                 }
   10162         [ #  # ]:           0 :                                 if (xexpr->args)
   10163                 :             :                                 {
   10164         [ #  # ]:           0 :                                         if (needcomma)
   10165                 :           0 :                                                 appendStringInfoString(buf, ", ");
   10166   [ #  #  #  #  :           0 :                                         switch (xexpr->op)
                      # ]
   10167                 :             :                                         {
   10168                 :             :                                                 case IS_XMLCONCAT:
   10169                 :             :                                                 case IS_XMLELEMENT:
   10170                 :             :                                                 case IS_XMLFOREST:
   10171                 :             :                                                 case IS_XMLPI:
   10172                 :             :                                                 case IS_XMLSERIALIZE:
   10173                 :             :                                                         /* no extra decoration needed */
   10174                 :           0 :                                                         get_rule_expr((Node *) xexpr->args, context, true);
   10175                 :           0 :                                                         break;
   10176                 :             :                                                 case IS_XMLPARSE:
   10177         [ #  # ]:           0 :                                                         Assert(list_length(xexpr->args) == 2);
   10178                 :             : 
   10179                 :           0 :                                                         get_rule_expr((Node *) linitial(xexpr->args),
   10180                 :           0 :                                                                                   context, true);
   10181                 :             : 
   10182                 :           0 :                                                         con = lsecond_node(Const, xexpr->args);
   10183         [ #  # ]:           0 :                                                         Assert(!con->constisnull);
   10184         [ #  # ]:           0 :                                                         if (DatumGetBool(con->constvalue))
   10185                 :           0 :                                                                 appendStringInfoString(buf,
   10186                 :             :                                                                                                            " PRESERVE WHITESPACE");
   10187                 :             :                                                         else
   10188                 :           0 :                                                                 appendStringInfoString(buf,
   10189                 :             :                                                                                                            " STRIP WHITESPACE");
   10190                 :           0 :                                                         break;
   10191                 :             :                                                 case IS_XMLROOT:
   10192         [ #  # ]:           0 :                                                         Assert(list_length(xexpr->args) == 3);
   10193                 :             : 
   10194                 :           0 :                                                         get_rule_expr((Node *) linitial(xexpr->args),
   10195                 :           0 :                                                                                   context, true);
   10196                 :             : 
   10197                 :           0 :                                                         appendStringInfoString(buf, ", VERSION ");
   10198                 :           0 :                                                         con = (Const *) lsecond(xexpr->args);
   10199   [ #  #  #  # ]:           0 :                                                         if (IsA(con, Const) &&
   10200                 :           0 :                                                                 con->constisnull)
   10201                 :           0 :                                                                 appendStringInfoString(buf, "NO VALUE");
   10202                 :             :                                                         else
   10203                 :           0 :                                                                 get_rule_expr((Node *) con, context, false);
   10204                 :             : 
   10205                 :           0 :                                                         con = lthird_node(Const, xexpr->args);
   10206         [ #  # ]:           0 :                                                         if (con->constisnull)
   10207                 :             :                                                                  /* suppress STANDALONE NO VALUE */ ;
   10208                 :             :                                                         else
   10209                 :             :                                                         {
   10210   [ #  #  #  # ]:           0 :                                                                 switch (DatumGetInt32(con->constvalue))
   10211                 :             :                                                                 {
   10212                 :             :                                                                         case XML_STANDALONE_YES:
   10213                 :           0 :                                                                                 appendStringInfoString(buf,
   10214                 :             :                                                                                                                            ", STANDALONE YES");
   10215                 :           0 :                                                                                 break;
   10216                 :             :                                                                         case XML_STANDALONE_NO:
   10217                 :           0 :                                                                                 appendStringInfoString(buf,
   10218                 :             :                                                                                                                            ", STANDALONE NO");
   10219                 :           0 :                                                                                 break;
   10220                 :             :                                                                         case XML_STANDALONE_NO_VALUE:
   10221                 :           0 :                                                                                 appendStringInfoString(buf,
   10222                 :             :                                                                                                                            ", STANDALONE NO VALUE");
   10223                 :           0 :                                                                                 break;
   10224                 :             :                                                                         default:
   10225                 :           0 :                                                                                 break;
   10226                 :             :                                                                 }
   10227                 :             :                                                         }
   10228                 :           0 :                                                         break;
   10229                 :             :                                                 case IS_DOCUMENT:
   10230                 :           0 :                                                         get_rule_expr_paren((Node *) xexpr->args, context, false, node);
   10231                 :           0 :                                                         break;
   10232                 :             :                                         }
   10233                 :           0 :                                 }
   10234         [ #  # ]:           0 :                                 if (xexpr->op == IS_XMLSERIALIZE)
   10235                 :             :                                 {
   10236                 :           0 :                                         appendStringInfo(buf, " AS %s",
   10237                 :           0 :                                                                          format_type_with_typemod(xexpr->type,
   10238                 :           0 :                                                                                                                           xexpr->typmod));
   10239         [ #  # ]:           0 :                                         if (xexpr->indent)
   10240                 :           0 :                                                 appendStringInfoString(buf, " INDENT");
   10241                 :             :                                         else
   10242                 :           0 :                                                 appendStringInfoString(buf, " NO INDENT");
   10243                 :           0 :                                 }
   10244                 :             : 
   10245         [ #  # ]:           0 :                                 if (xexpr->op == IS_DOCUMENT)
   10246                 :           0 :                                         appendStringInfoString(buf, " IS DOCUMENT");
   10247                 :             :                                 else
   10248                 :           0 :                                         appendStringInfoChar(buf, ')');
   10249                 :           0 :                         }
   10250                 :           0 :                         break;
   10251                 :             : 
   10252                 :             :                 case T_NullTest:
   10253                 :             :                         {
   10254                 :           0 :                                 NullTest   *ntest = (NullTest *) node;
   10255                 :             : 
   10256         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
   10257                 :           0 :                                         appendStringInfoChar(buf, '(');
   10258                 :           0 :                                 get_rule_expr_paren((Node *) ntest->arg, context, true, node);
   10259                 :             : 
   10260                 :             :                                 /*
   10261                 :             :                                  * For scalar inputs, we prefer to print as IS [NOT] NULL,
   10262                 :             :                                  * which is shorter and traditional.  If it's a rowtype input
   10263                 :             :                                  * but we're applying a scalar test, must print IS [NOT]
   10264                 :             :                                  * DISTINCT FROM NULL to be semantically correct.
   10265                 :             :                                  */
   10266   [ #  #  #  # ]:           0 :                                 if (ntest->argisrow ||
   10267                 :           0 :                                         !type_is_rowtype(exprType((Node *) ntest->arg)))
   10268                 :             :                                 {
   10269      [ #  #  # ]:           0 :                                         switch (ntest->nulltesttype)
   10270                 :             :                                         {
   10271                 :             :                                                 case IS_NULL:
   10272                 :           0 :                                                         appendStringInfoString(buf, " IS NULL");
   10273                 :           0 :                                                         break;
   10274                 :             :                                                 case IS_NOT_NULL:
   10275                 :           0 :                                                         appendStringInfoString(buf, " IS NOT NULL");
   10276                 :           0 :                                                         break;
   10277                 :             :                                                 default:
   10278   [ #  #  #  # ]:           0 :                                                         elog(ERROR, "unrecognized nulltesttype: %d",
   10279                 :             :                                                                  (int) ntest->nulltesttype);
   10280                 :           0 :                                         }
   10281                 :           0 :                                 }
   10282                 :             :                                 else
   10283                 :             :                                 {
   10284      [ #  #  # ]:           0 :                                         switch (ntest->nulltesttype)
   10285                 :             :                                         {
   10286                 :             :                                                 case IS_NULL:
   10287                 :           0 :                                                         appendStringInfoString(buf, " IS NOT DISTINCT FROM NULL");
   10288                 :           0 :                                                         break;
   10289                 :             :                                                 case IS_NOT_NULL:
   10290                 :           0 :                                                         appendStringInfoString(buf, " IS DISTINCT FROM NULL");
   10291                 :           0 :                                                         break;
   10292                 :             :                                                 default:
   10293   [ #  #  #  # ]:           0 :                                                         elog(ERROR, "unrecognized nulltesttype: %d",
   10294                 :             :                                                                  (int) ntest->nulltesttype);
   10295                 :           0 :                                         }
   10296                 :             :                                 }
   10297         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
   10298                 :           0 :                                         appendStringInfoChar(buf, ')');
   10299                 :           0 :                         }
   10300                 :           0 :                         break;
   10301                 :             : 
   10302                 :             :                 case T_BooleanTest:
   10303                 :             :                         {
   10304                 :           0 :                                 BooleanTest *btest = (BooleanTest *) node;
   10305                 :             : 
   10306         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
   10307                 :           0 :                                         appendStringInfoChar(buf, '(');
   10308                 :           0 :                                 get_rule_expr_paren((Node *) btest->arg, context, false, node);
   10309   [ #  #  #  #  :           0 :                                 switch (btest->booltesttype)
                #  #  # ]
   10310                 :             :                                 {
   10311                 :             :                                         case IS_TRUE:
   10312                 :           0 :                                                 appendStringInfoString(buf, " IS TRUE");
   10313                 :           0 :                                                 break;
   10314                 :             :                                         case IS_NOT_TRUE:
   10315                 :           0 :                                                 appendStringInfoString(buf, " IS NOT TRUE");
   10316                 :           0 :                                                 break;
   10317                 :             :                                         case IS_FALSE:
   10318                 :           0 :                                                 appendStringInfoString(buf, " IS FALSE");
   10319                 :           0 :                                                 break;
   10320                 :             :                                         case IS_NOT_FALSE:
   10321                 :           0 :                                                 appendStringInfoString(buf, " IS NOT FALSE");
   10322                 :           0 :                                                 break;
   10323                 :             :                                         case IS_UNKNOWN:
   10324                 :           0 :                                                 appendStringInfoString(buf, " IS UNKNOWN");
   10325                 :           0 :                                                 break;
   10326                 :             :                                         case IS_NOT_UNKNOWN:
   10327                 :           0 :                                                 appendStringInfoString(buf, " IS NOT UNKNOWN");
   10328                 :           0 :                                                 break;
   10329                 :             :                                         default:
   10330   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "unrecognized booltesttype: %d",
   10331                 :             :                                                          (int) btest->booltesttype);
   10332                 :           0 :                                 }
   10333         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
   10334                 :           0 :                                         appendStringInfoChar(buf, ')');
   10335                 :           0 :                         }
   10336                 :           0 :                         break;
   10337                 :             : 
   10338                 :             :                 case T_CoerceToDomain:
   10339                 :             :                         {
   10340                 :           0 :                                 CoerceToDomain *ctest = (CoerceToDomain *) node;
   10341                 :           0 :                                 Node       *arg = (Node *) ctest->arg;
   10342                 :             : 
   10343   [ #  #  #  # ]:           0 :                                 if (ctest->coercionformat == COERCE_IMPLICIT_CAST &&
   10344                 :           0 :                                         !showimplicit)
   10345                 :             :                                 {
   10346                 :             :                                         /* don't show the implicit cast */
   10347                 :           0 :                                         get_rule_expr(arg, context, false);
   10348                 :           0 :                                 }
   10349                 :             :                                 else
   10350                 :             :                                 {
   10351                 :           0 :                                         get_coercion_expr(arg, context,
   10352                 :           0 :                                                                           ctest->resulttype,
   10353                 :           0 :                                                                           ctest->resulttypmod,
   10354                 :           0 :                                                                           node);
   10355                 :             :                                 }
   10356                 :           0 :                         }
   10357                 :           0 :                         break;
   10358                 :             : 
   10359                 :             :                 case T_CoerceToDomainValue:
   10360                 :           0 :                         appendStringInfoString(buf, "VALUE");
   10361                 :           0 :                         break;
   10362                 :             : 
   10363                 :             :                 case T_SetToDefault:
   10364                 :           0 :                         appendStringInfoString(buf, "DEFAULT");
   10365                 :           0 :                         break;
   10366                 :             : 
   10367                 :             :                 case T_CurrentOfExpr:
   10368                 :             :                         {
   10369                 :           0 :                                 CurrentOfExpr *cexpr = (CurrentOfExpr *) node;
   10370                 :             : 
   10371         [ #  # ]:           0 :                                 if (cexpr->cursor_name)
   10372                 :           0 :                                         appendStringInfo(buf, "CURRENT OF %s",
   10373                 :           0 :                                                                          quote_identifier(cexpr->cursor_name));
   10374                 :             :                                 else
   10375                 :           0 :                                         appendStringInfo(buf, "CURRENT OF $%d",
   10376                 :           0 :                                                                          cexpr->cursor_param);
   10377                 :           0 :                         }
   10378                 :           0 :                         break;
   10379                 :             : 
   10380                 :             :                 case T_NextValueExpr:
   10381                 :             :                         {
   10382                 :           0 :                                 NextValueExpr *nvexpr = (NextValueExpr *) node;
   10383                 :             : 
   10384                 :             :                                 /*
   10385                 :             :                                  * This isn't exactly nextval(), but that seems close enough
   10386                 :             :                                  * for EXPLAIN's purposes.
   10387                 :             :                                  */
   10388                 :           0 :                                 appendStringInfoString(buf, "nextval(");
   10389                 :           0 :                                 simple_quote_literal(buf,
   10390                 :           0 :                                                                          generate_relation_name(nvexpr->seqid,
   10391                 :             :                                                                                                                         NIL));
   10392                 :           0 :                                 appendStringInfoChar(buf, ')');
   10393                 :           0 :                         }
   10394                 :           0 :                         break;
   10395                 :             : 
   10396                 :             :                 case T_InferenceElem:
   10397                 :             :                         {
   10398                 :           0 :                                 InferenceElem *iexpr = (InferenceElem *) node;
   10399                 :           0 :                                 bool            save_varprefix;
   10400                 :           0 :                                 bool            need_parens;
   10401                 :             : 
   10402                 :             :                                 /*
   10403                 :             :                                  * InferenceElem can only refer to target relation, so a
   10404                 :             :                                  * prefix is not useful, and indeed would cause parse errors.
   10405                 :             :                                  */
   10406                 :           0 :                                 save_varprefix = context->varprefix;
   10407                 :           0 :                                 context->varprefix = false;
   10408                 :             : 
   10409                 :             :                                 /*
   10410                 :             :                                  * Parenthesize the element unless it's a simple Var or a bare
   10411                 :             :                                  * function call.  Follows pg_get_indexdef_worker().
   10412                 :             :                                  */
   10413                 :           0 :                                 need_parens = !IsA(iexpr->expr, Var);
   10414   [ #  #  #  # ]:           0 :                                 if (IsA(iexpr->expr, FuncExpr) &&
   10415                 :           0 :                                         ((FuncExpr *) iexpr->expr)->funcformat ==
   10416                 :             :                                         COERCE_EXPLICIT_CALL)
   10417                 :           0 :                                         need_parens = false;
   10418                 :             : 
   10419         [ #  # ]:           0 :                                 if (need_parens)
   10420                 :           0 :                                         appendStringInfoChar(buf, '(');
   10421                 :           0 :                                 get_rule_expr((Node *) iexpr->expr,
   10422                 :           0 :                                                           context, false);
   10423         [ #  # ]:           0 :                                 if (need_parens)
   10424                 :           0 :                                         appendStringInfoChar(buf, ')');
   10425                 :             : 
   10426                 :           0 :                                 context->varprefix = save_varprefix;
   10427                 :             : 
   10428         [ #  # ]:           0 :                                 if (iexpr->infercollid)
   10429                 :           0 :                                         appendStringInfo(buf, " COLLATE %s",
   10430                 :           0 :                                                                          generate_collation_name(iexpr->infercollid));
   10431                 :             : 
   10432                 :             :                                 /* Add the operator class name, if not default */
   10433         [ #  # ]:           0 :                                 if (iexpr->inferopclass)
   10434                 :             :                                 {
   10435                 :           0 :                                         Oid                     inferopclass = iexpr->inferopclass;
   10436                 :           0 :                                         Oid                     inferopcinputtype = get_opclass_input_type(iexpr->inferopclass);
   10437                 :             : 
   10438                 :           0 :                                         get_opclass_name(inferopclass, inferopcinputtype, buf);
   10439                 :           0 :                                 }
   10440                 :           0 :                         }
   10441                 :           0 :                         break;
   10442                 :             : 
   10443                 :             :                 case T_ReturningExpr:
   10444                 :             :                         {
   10445                 :           0 :                                 ReturningExpr *retExpr = (ReturningExpr *) node;
   10446                 :             : 
   10447                 :             :                                 /*
   10448                 :             :                                  * We cannot see a ReturningExpr in rule deparsing, only while
   10449                 :             :                                  * EXPLAINing a query plan (ReturningExpr nodes are only ever
   10450                 :             :                                  * adding during query rewriting). Just display the expression
   10451                 :             :                                  * returned (an expanded view column).
   10452                 :             :                                  */
   10453                 :           0 :                                 get_rule_expr((Node *) retExpr->retexpr, context, showimplicit);
   10454                 :           0 :                         }
   10455                 :           0 :                         break;
   10456                 :             : 
   10457                 :             :                 case T_PartitionBoundSpec:
   10458                 :             :                         {
   10459                 :           0 :                                 PartitionBoundSpec *spec = (PartitionBoundSpec *) node;
   10460                 :           0 :                                 ListCell   *cell;
   10461                 :           0 :                                 char       *sep;
   10462                 :             : 
   10463         [ #  # ]:           0 :                                 if (spec->is_default)
   10464                 :             :                                 {
   10465                 :           0 :                                         appendStringInfoString(buf, "DEFAULT");
   10466                 :           0 :                                         break;
   10467                 :             :                                 }
   10468                 :             : 
   10469   [ #  #  #  # ]:           0 :                                 switch (spec->strategy)
   10470                 :             :                                 {
   10471                 :             :                                         case PARTITION_STRATEGY_HASH:
   10472         [ #  # ]:           0 :                                                 Assert(spec->modulus > 0 && spec->remainder >= 0);
   10473         [ #  # ]:           0 :                                                 Assert(spec->modulus > spec->remainder);
   10474                 :             : 
   10475                 :           0 :                                                 appendStringInfoString(buf, "FOR VALUES");
   10476                 :           0 :                                                 appendStringInfo(buf, " WITH (modulus %d, remainder %d)",
   10477                 :           0 :                                                                                  spec->modulus, spec->remainder);
   10478                 :           0 :                                                 break;
   10479                 :             : 
   10480                 :             :                                         case PARTITION_STRATEGY_LIST:
   10481         [ #  # ]:           0 :                                                 Assert(spec->listdatums != NIL);
   10482                 :             : 
   10483                 :           0 :                                                 appendStringInfoString(buf, "FOR VALUES IN (");
   10484                 :           0 :                                                 sep = "";
   10485   [ #  #  #  #  :           0 :                                                 foreach(cell, spec->listdatums)
                   #  # ]
   10486                 :             :                                                 {
   10487                 :           0 :                                                         Const      *val = lfirst_node(Const, cell);
   10488                 :             : 
   10489                 :           0 :                                                         appendStringInfoString(buf, sep);
   10490                 :           0 :                                                         get_const_expr(val, context, -1);
   10491                 :           0 :                                                         sep = ", ";
   10492                 :           0 :                                                 }
   10493                 :             : 
   10494                 :           0 :                                                 appendStringInfoChar(buf, ')');
   10495                 :           0 :                                                 break;
   10496                 :             : 
   10497                 :             :                                         case PARTITION_STRATEGY_RANGE:
   10498         [ #  # ]:           0 :                                                 Assert(spec->lowerdatums != NIL &&
   10499                 :             :                                                            spec->upperdatums != NIL &&
   10500                 :             :                                                            list_length(spec->lowerdatums) ==
   10501                 :             :                                                            list_length(spec->upperdatums));
   10502                 :             : 
   10503                 :           0 :                                                 appendStringInfo(buf, "FOR VALUES FROM %s TO %s",
   10504                 :           0 :                                                                                  get_range_partbound_string(spec->lowerdatums),
   10505                 :           0 :                                                                                  get_range_partbound_string(spec->upperdatums));
   10506                 :           0 :                                                 break;
   10507                 :             : 
   10508                 :             :                                         default:
   10509   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "unrecognized partition strategy: %d",
   10510                 :             :                                                          (int) spec->strategy);
   10511                 :           0 :                                                 break;
   10512                 :             :                                 }
   10513         [ #  # ]:           0 :                         }
   10514                 :           0 :                         break;
   10515                 :             : 
   10516                 :             :                 case T_JsonValueExpr:
   10517                 :             :                         {
   10518                 :           0 :                                 JsonValueExpr *jve = (JsonValueExpr *) node;
   10519                 :             : 
   10520                 :           0 :                                 get_rule_expr((Node *) jve->raw_expr, context, false);
   10521                 :           0 :                                 get_json_format(jve->format, context->buf);
   10522                 :           0 :                         }
   10523                 :           0 :                         break;
   10524                 :             : 
   10525                 :             :                 case T_JsonConstructorExpr:
   10526                 :           0 :                         get_json_constructor((JsonConstructorExpr *) node, context, false);
   10527                 :           0 :                         break;
   10528                 :             : 
   10529                 :             :                 case T_JsonIsPredicate:
   10530                 :             :                         {
   10531                 :           0 :                                 JsonIsPredicate *pred = (JsonIsPredicate *) node;
   10532                 :             : 
   10533         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
   10534                 :           0 :                                         appendStringInfoChar(context->buf, '(');
   10535                 :             : 
   10536                 :           0 :                                 get_rule_expr_paren(pred->expr, context, true, node);
   10537                 :             : 
   10538                 :           0 :                                 appendStringInfoString(context->buf, " IS JSON");
   10539                 :             : 
   10540                 :             :                                 /* TODO: handle FORMAT clause */
   10541                 :             : 
   10542   [ #  #  #  # ]:           0 :                                 switch (pred->item_type)
   10543                 :             :                                 {
   10544                 :             :                                         case JS_TYPE_SCALAR:
   10545                 :           0 :                                                 appendStringInfoString(context->buf, " SCALAR");
   10546                 :           0 :                                                 break;
   10547                 :             :                                         case JS_TYPE_ARRAY:
   10548                 :           0 :                                                 appendStringInfoString(context->buf, " ARRAY");
   10549                 :           0 :                                                 break;
   10550                 :             :                                         case JS_TYPE_OBJECT:
   10551                 :           0 :                                                 appendStringInfoString(context->buf, " OBJECT");
   10552                 :           0 :                                                 break;
   10553                 :             :                                         default:
   10554                 :           0 :                                                 break;
   10555                 :             :                                 }
   10556                 :             : 
   10557         [ #  # ]:           0 :                                 if (pred->unique_keys)
   10558                 :           0 :                                         appendStringInfoString(context->buf, " WITH UNIQUE KEYS");
   10559                 :             : 
   10560         [ #  # ]:           0 :                                 if (!PRETTY_PAREN(context))
   10561                 :           0 :                                         appendStringInfoChar(context->buf, ')');
   10562                 :           0 :                         }
   10563                 :           0 :                         break;
   10564                 :             : 
   10565                 :             :                 case T_JsonExpr:
   10566                 :             :                         {
   10567                 :           0 :                                 JsonExpr   *jexpr = (JsonExpr *) node;
   10568                 :             : 
   10569   [ #  #  #  # ]:           0 :                                 switch (jexpr->op)
   10570                 :             :                                 {
   10571                 :             :                                         case JSON_EXISTS_OP:
   10572                 :           0 :                                                 appendStringInfoString(buf, "JSON_EXISTS(");
   10573                 :           0 :                                                 break;
   10574                 :             :                                         case JSON_QUERY_OP:
   10575                 :           0 :                                                 appendStringInfoString(buf, "JSON_QUERY(");
   10576                 :           0 :                                                 break;
   10577                 :             :                                         case JSON_VALUE_OP:
   10578                 :           0 :                                                 appendStringInfoString(buf, "JSON_VALUE(");
   10579                 :           0 :                                                 break;
   10580                 :             :                                         default:
   10581   [ #  #  #  # ]:           0 :                                                 elog(ERROR, "unrecognized JsonExpr op: %d",
   10582                 :             :                                                          (int) jexpr->op);
   10583                 :           0 :                                 }
   10584                 :             : 
   10585                 :           0 :                                 get_rule_expr(jexpr->formatted_expr, context, showimplicit);
   10586                 :             : 
   10587                 :           0 :                                 appendStringInfoString(buf, ", ");
   10588                 :             : 
   10589                 :           0 :                                 get_json_path_spec(jexpr->path_spec, context, showimplicit);
   10590                 :             : 
   10591         [ #  # ]:           0 :                                 if (jexpr->passing_values)
   10592                 :             :                                 {
   10593                 :           0 :                                         ListCell   *lc1,
   10594                 :             :                                                            *lc2;
   10595                 :           0 :                                         bool            needcomma = false;
   10596                 :             : 
   10597                 :           0 :                                         appendStringInfoString(buf, " PASSING ");
   10598                 :             : 
   10599   [ #  #  #  #  :           0 :                                         forboth(lc1, jexpr->passing_names,
          #  #  #  #  #  
                #  #  # ]
   10600                 :             :                                                         lc2, jexpr->passing_values)
   10601                 :             :                                         {
   10602         [ #  # ]:           0 :                                                 if (needcomma)
   10603                 :           0 :                                                         appendStringInfoString(buf, ", ");
   10604                 :           0 :                                                 needcomma = true;
   10605                 :             : 
   10606                 :           0 :                                                 get_rule_expr((Node *) lfirst(lc2), context, showimplicit);
   10607                 :           0 :                                                 appendStringInfo(buf, " AS %s",
   10608                 :           0 :                                                                                  quote_identifier(lfirst_node(String, lc1)->sval));
   10609                 :           0 :                                         }
   10610                 :           0 :                                 }
   10611                 :             : 
   10612   [ #  #  #  # ]:           0 :                                 if (jexpr->op != JSON_EXISTS_OP ||
   10613                 :           0 :                                         jexpr->returning->typid != BOOLOID)
   10614                 :           0 :                                         get_json_returning(jexpr->returning, context->buf,
   10615                 :           0 :                                                                            jexpr->op == JSON_QUERY_OP);
   10616                 :             : 
   10617                 :           0 :                                 get_json_expr_options(jexpr, context,
   10618                 :           0 :                                                                           jexpr->op != JSON_EXISTS_OP ?
   10619                 :             :                                                                           JSON_BEHAVIOR_NULL :
   10620                 :             :                                                                           JSON_BEHAVIOR_FALSE);
   10621                 :             : 
   10622                 :           0 :                                 appendStringInfoChar(buf, ')');
   10623                 :           0 :                         }
   10624                 :           0 :                         break;
   10625                 :             : 
   10626                 :             :                 case T_List:
   10627                 :             :                         {
   10628                 :           0 :                                 char       *sep;
   10629                 :           0 :                                 ListCell   *l;
   10630                 :             : 
   10631                 :           0 :                                 sep = "";
   10632   [ #  #  #  #  :           0 :                                 foreach(l, (List *) node)
                   #  # ]
   10633                 :             :                                 {
   10634                 :           0 :                                         appendStringInfoString(buf, sep);
   10635                 :           0 :                                         get_rule_expr((Node *) lfirst(l), context, showimplicit);
   10636                 :           0 :                                         sep = ", ";
   10637                 :           0 :                                 }
   10638                 :           0 :                         }
   10639                 :           0 :                         break;
   10640                 :             : 
   10641                 :             :                 case T_TableFunc:
   10642                 :           0 :                         get_tablefunc((TableFunc *) node, context, showimplicit);
   10643                 :           0 :                         break;
   10644                 :             : 
   10645                 :             :                 default:
   10646   [ #  #  #  # ]:           0 :                         elog(ERROR, "unrecognized node type: %d", (int) nodeTag(node));
   10647                 :           0 :                         break;
   10648                 :             :         }
   10649                 :           0 : }
   10650                 :             : 
   10651                 :             : /*
   10652                 :             :  * get_rule_expr_toplevel               - Parse back a toplevel expression
   10653                 :             :  *
   10654                 :             :  * Same as get_rule_expr(), except that if the expr is just a Var, we pass
   10655                 :             :  * istoplevel = true not false to get_variable().  This causes whole-row Vars
   10656                 :             :  * to get printed with decoration that will prevent expansion of "*".
   10657                 :             :  * We need to use this in contexts such as ROW() and VALUES(), where the
   10658                 :             :  * parser would expand "foo.*" appearing at top level.  (In principle we'd
   10659                 :             :  * use this in get_target_list() too, but that has additional worries about
   10660                 :             :  * whether to print AS, so it needs to invoke get_variable() directly anyway.)
   10661                 :             :  */
   10662                 :             : static void
   10663                 :           0 : get_rule_expr_toplevel(Node *node, deparse_context *context,
   10664                 :             :                                            bool showimplicit)
   10665                 :             : {
   10666   [ #  #  #  # ]:           0 :         if (node && IsA(node, Var))
   10667                 :           0 :                 (void) get_variable((Var *) node, 0, true, context);
   10668                 :             :         else
   10669                 :           0 :                 get_rule_expr(node, context, showimplicit);
   10670                 :           0 : }
   10671                 :             : 
   10672                 :             : /*
   10673                 :             :  * get_rule_list_toplevel               - Parse back a list of toplevel expressions
   10674                 :             :  *
   10675                 :             :  * Apply get_rule_expr_toplevel() to each element of a List.
   10676                 :             :  *
   10677                 :             :  * This adds commas between the expressions, but caller is responsible
   10678                 :             :  * for printing surrounding decoration.
   10679                 :             :  */
   10680                 :             : static void
   10681                 :           0 : get_rule_list_toplevel(List *lst, deparse_context *context,
   10682                 :             :                                            bool showimplicit)
   10683                 :             : {
   10684                 :           0 :         const char *sep;
   10685                 :           0 :         ListCell   *lc;
   10686                 :             : 
   10687                 :           0 :         sep = "";
   10688   [ #  #  #  #  :           0 :         foreach(lc, lst)
                   #  # ]
   10689                 :             :         {
   10690                 :           0 :                 Node       *e = (Node *) lfirst(lc);
   10691                 :             : 
   10692                 :           0 :                 appendStringInfoString(context->buf, sep);
   10693                 :           0 :                 get_rule_expr_toplevel(e, context, showimplicit);
   10694                 :           0 :                 sep = ", ";
   10695                 :           0 :         }
   10696                 :           0 : }
   10697                 :             : 
   10698                 :             : /*
   10699                 :             :  * get_rule_expr_funccall               - Parse back a function-call expression
   10700                 :             :  *
   10701                 :             :  * Same as get_rule_expr(), except that we guarantee that the output will
   10702                 :             :  * look like a function call, or like one of the things the grammar treats as
   10703                 :             :  * equivalent to a function call (see the func_expr_windowless production).
   10704                 :             :  * This is needed in places where the grammar uses func_expr_windowless and
   10705                 :             :  * you can't substitute a parenthesized a_expr.  If what we have isn't going
   10706                 :             :  * to look like a function call, wrap it in a dummy CAST() expression, which
   10707                 :             :  * will satisfy the grammar --- and, indeed, is likely what the user wrote to
   10708                 :             :  * produce such a thing.
   10709                 :             :  */
   10710                 :             : static void
   10711                 :           0 : get_rule_expr_funccall(Node *node, deparse_context *context,
   10712                 :             :                                            bool showimplicit)
   10713                 :             : {
   10714         [ #  # ]:           0 :         if (looks_like_function(node))
   10715                 :           0 :                 get_rule_expr(node, context, showimplicit);
   10716                 :             :         else
   10717                 :             :         {
   10718                 :           0 :                 StringInfo      buf = context->buf;
   10719                 :             : 
   10720                 :           0 :                 appendStringInfoString(buf, "CAST(");
   10721                 :             :                 /* no point in showing any top-level implicit cast */
   10722                 :           0 :                 get_rule_expr(node, context, false);
   10723                 :           0 :                 appendStringInfo(buf, " AS %s)",
   10724                 :           0 :                                                  format_type_with_typemod(exprType(node),
   10725                 :           0 :                                                                                                   exprTypmod(node)));
   10726                 :           0 :         }
   10727                 :           0 : }
   10728                 :             : 
   10729                 :             : /*
   10730                 :             :  * Helper function to identify node types that satisfy func_expr_windowless.
   10731                 :             :  * If in doubt, "false" is always a safe answer.
   10732                 :             :  */
   10733                 :             : static bool
   10734                 :           0 : looks_like_function(Node *node)
   10735                 :             : {
   10736         [ #  # ]:           0 :         if (node == NULL)
   10737                 :           0 :                 return false;                   /* probably shouldn't happen */
   10738      [ #  #  # ]:           0 :         switch (nodeTag(node))
   10739                 :             :         {
   10740                 :             :                 case T_FuncExpr:
   10741                 :             :                         /* OK, unless it's going to deparse as a cast */
   10742         [ #  # ]:           0 :                         return (((FuncExpr *) node)->funcformat == COERCE_EXPLICIT_CALL ||
   10743                 :           0 :                                         ((FuncExpr *) node)->funcformat == COERCE_SQL_SYNTAX);
   10744                 :             :                 case T_NullIfExpr:
   10745                 :             :                 case T_CoalesceExpr:
   10746                 :             :                 case T_MinMaxExpr:
   10747                 :             :                 case T_SQLValueFunction:
   10748                 :             :                 case T_XmlExpr:
   10749                 :             :                 case T_JsonExpr:
   10750                 :             :                         /* these are all accepted by func_expr_common_subexpr */
   10751                 :           0 :                         return true;
   10752                 :             :                 default:
   10753                 :           0 :                         break;
   10754                 :             :         }
   10755                 :           0 :         return false;
   10756                 :           0 : }
   10757                 :             : 
   10758                 :             : 
   10759                 :             : /*
   10760                 :             :  * get_oper_expr                        - Parse back an OpExpr node
   10761                 :             :  */
   10762                 :             : static void
   10763                 :           0 : get_oper_expr(OpExpr *expr, deparse_context *context)
   10764                 :             : {
   10765                 :           0 :         StringInfo      buf = context->buf;
   10766                 :           0 :         Oid                     opno = expr->opno;
   10767                 :           0 :         List       *args = expr->args;
   10768                 :             : 
   10769         [ #  # ]:           0 :         if (!PRETTY_PAREN(context))
   10770                 :           0 :                 appendStringInfoChar(buf, '(');
   10771         [ #  # ]:           0 :         if (list_length(args) == 2)
   10772                 :             :         {
   10773                 :             :                 /* binary operator */
   10774                 :           0 :                 Node       *arg1 = (Node *) linitial(args);
   10775                 :           0 :                 Node       *arg2 = (Node *) lsecond(args);
   10776                 :             : 
   10777                 :           0 :                 get_rule_expr_paren(arg1, context, true, (Node *) expr);
   10778                 :           0 :                 appendStringInfo(buf, " %s ",
   10779                 :           0 :                                                  generate_operator_name(opno,
   10780                 :           0 :                                                                                                 exprType(arg1),
   10781                 :           0 :                                                                                                 exprType(arg2)));
   10782                 :           0 :                 get_rule_expr_paren(arg2, context, true, (Node *) expr);
   10783                 :           0 :         }
   10784                 :             :         else
   10785                 :             :         {
   10786                 :             :                 /* prefix operator */
   10787                 :           0 :                 Node       *arg = (Node *) linitial(args);
   10788                 :             : 
   10789                 :           0 :                 appendStringInfo(buf, "%s ",
   10790                 :           0 :                                                  generate_operator_name(opno,
   10791                 :             :                                                                                                 InvalidOid,
   10792                 :           0 :                                                                                                 exprType(arg)));
   10793                 :           0 :                 get_rule_expr_paren(arg, context, true, (Node *) expr);
   10794                 :           0 :         }
   10795         [ #  # ]:           0 :         if (!PRETTY_PAREN(context))
   10796                 :           0 :                 appendStringInfoChar(buf, ')');
   10797                 :           0 : }
   10798                 :             : 
   10799                 :             : /*
   10800                 :             :  * get_func_expr                        - Parse back a FuncExpr node
   10801                 :             :  */
   10802                 :             : static void
   10803                 :           0 : get_func_expr(FuncExpr *expr, deparse_context *context,
   10804                 :             :                           bool showimplicit)
   10805                 :             : {
   10806                 :           0 :         StringInfo      buf = context->buf;
   10807                 :           0 :         Oid                     funcoid = expr->funcid;
   10808                 :           0 :         Oid                     argtypes[FUNC_MAX_ARGS];
   10809                 :           0 :         int                     nargs;
   10810                 :           0 :         List       *argnames;
   10811                 :           0 :         bool            use_variadic;
   10812                 :           0 :         ListCell   *l;
   10813                 :             : 
   10814                 :             :         /*
   10815                 :             :          * If the function call came from an implicit coercion, then just show the
   10816                 :             :          * first argument --- unless caller wants to see implicit coercions.
   10817                 :             :          */
   10818   [ #  #  #  # ]:           0 :         if (expr->funcformat == COERCE_IMPLICIT_CAST && !showimplicit)
   10819                 :             :         {
   10820                 :           0 :                 get_rule_expr_paren((Node *) linitial(expr->args), context,
   10821                 :           0 :                                                         false, (Node *) expr);
   10822                 :           0 :                 return;
   10823                 :             :         }
   10824                 :             : 
   10825                 :             :         /*
   10826                 :             :          * If the function call came from a cast, then show the first argument
   10827                 :             :          * plus an explicit cast operation.
   10828                 :             :          */
   10829   [ #  #  #  # ]:           0 :         if (expr->funcformat == COERCE_EXPLICIT_CAST ||
   10830                 :           0 :                 expr->funcformat == COERCE_IMPLICIT_CAST)
   10831                 :             :         {
   10832                 :           0 :                 Node       *arg = linitial(expr->args);
   10833                 :           0 :                 Oid                     rettype = expr->funcresulttype;
   10834                 :           0 :                 int32           coercedTypmod;
   10835                 :             : 
   10836                 :             :                 /* Get the typmod if this is a length-coercion function */
   10837                 :           0 :                 (void) exprIsLengthCoercion((Node *) expr, &coercedTypmod);
   10838                 :             : 
   10839                 :           0 :                 get_coercion_expr(arg, context,
   10840                 :           0 :                                                   rettype, coercedTypmod,
   10841                 :           0 :                                                   (Node *) expr);
   10842                 :             : 
   10843                 :             :                 return;
   10844                 :           0 :         }
   10845                 :             : 
   10846                 :             :         /*
   10847                 :             :          * If the function was called using one of the SQL spec's random special
   10848                 :             :          * syntaxes, try to reproduce that.  If we don't recognize the function,
   10849                 :             :          * fall through.
   10850                 :             :          */
   10851         [ #  # ]:           0 :         if (expr->funcformat == COERCE_SQL_SYNTAX)
   10852                 :             :         {
   10853         [ #  # ]:           0 :                 if (get_func_sql_syntax(expr, context))
   10854                 :           0 :                         return;
   10855                 :           0 :         }
   10856                 :             : 
   10857                 :             :         /*
   10858                 :             :          * Normal function: display as proname(args).  First we need to extract
   10859                 :             :          * the argument datatypes.
   10860                 :             :          */
   10861         [ #  # ]:           0 :         if (list_length(expr->args) > FUNC_MAX_ARGS)
   10862   [ #  #  #  # ]:           0 :                 ereport(ERROR,
   10863                 :             :                                 (errcode(ERRCODE_TOO_MANY_ARGUMENTS),
   10864                 :             :                                  errmsg("too many arguments")));
   10865                 :           0 :         nargs = 0;
   10866                 :           0 :         argnames = NIL;
   10867   [ #  #  #  #  :           0 :         foreach(l, expr->args)
                   #  # ]
   10868                 :             :         {
   10869                 :           0 :                 Node       *arg = (Node *) lfirst(l);
   10870                 :             : 
   10871         [ #  # ]:           0 :                 if (IsA(arg, NamedArgExpr))
   10872                 :           0 :                         argnames = lappend(argnames, ((NamedArgExpr *) arg)->name);
   10873                 :           0 :                 argtypes[nargs] = exprType(arg);
   10874                 :           0 :                 nargs++;
   10875                 :           0 :         }
   10876                 :             : 
   10877                 :           0 :         appendStringInfo(buf, "%s(",
   10878                 :           0 :                                          generate_function_name(funcoid, nargs,
   10879                 :           0 :                                                                                         argnames, argtypes,
   10880                 :           0 :                                                                                         expr->funcvariadic,
   10881                 :             :                                                                                         &use_variadic,
   10882                 :           0 :                                                                                         context->inGroupBy));
   10883                 :           0 :         nargs = 0;
   10884   [ #  #  #  #  :           0 :         foreach(l, expr->args)
                   #  # ]
   10885                 :             :         {
   10886         [ #  # ]:           0 :                 if (nargs++ > 0)
   10887                 :           0 :                         appendStringInfoString(buf, ", ");
   10888   [ #  #  #  # ]:           0 :                 if (use_variadic && lnext(expr->args, l) == NULL)
   10889                 :           0 :                         appendStringInfoString(buf, "VARIADIC ");
   10890                 :           0 :                 get_rule_expr((Node *) lfirst(l), context, true);
   10891                 :           0 :         }
   10892                 :           0 :         appendStringInfoChar(buf, ')');
   10893         [ #  # ]:           0 : }
   10894                 :             : 
   10895                 :             : /*
   10896                 :             :  * get_agg_expr                 - Parse back an Aggref node
   10897                 :             :  */
   10898                 :             : static void
   10899                 :           0 : get_agg_expr(Aggref *aggref, deparse_context *context,
   10900                 :             :                          Aggref *original_aggref)
   10901                 :             : {
   10902                 :           0 :         get_agg_expr_helper(aggref, context, original_aggref, NULL, NULL,
   10903                 :             :                                                 false);
   10904                 :           0 : }
   10905                 :             : 
   10906                 :             : /*
   10907                 :             :  * get_agg_expr_helper          - subroutine for get_agg_expr and
   10908                 :             :  *                                                      get_json_agg_constructor
   10909                 :             :  */
   10910                 :             : static void
   10911                 :           0 : get_agg_expr_helper(Aggref *aggref, deparse_context *context,
   10912                 :             :                                         Aggref *original_aggref, const char *funcname,
   10913                 :             :                                         const char *options, bool is_json_objectagg)
   10914                 :             : {
   10915                 :           0 :         StringInfo      buf = context->buf;
   10916                 :           0 :         Oid                     argtypes[FUNC_MAX_ARGS];
   10917                 :           0 :         int                     nargs;
   10918                 :           0 :         bool            use_variadic = false;
   10919                 :             : 
   10920                 :             :         /*
   10921                 :             :          * For a combining aggregate, we look up and deparse the corresponding
   10922                 :             :          * partial aggregate instead.  This is necessary because our input
   10923                 :             :          * argument list has been replaced; the new argument list always has just
   10924                 :             :          * one element, which will point to a partial Aggref that supplies us with
   10925                 :             :          * transition states to combine.
   10926                 :             :          */
   10927         [ #  # ]:           0 :         if (DO_AGGSPLIT_COMBINE(aggref->aggsplit))
   10928                 :             :         {
   10929                 :           0 :                 TargetEntry *tle;
   10930                 :             : 
   10931         [ #  # ]:           0 :                 Assert(list_length(aggref->args) == 1);
   10932                 :           0 :                 tle = linitial_node(TargetEntry, aggref->args);
   10933                 :           0 :                 resolve_special_varno((Node *) tle->expr, context,
   10934                 :           0 :                                                           get_agg_combine_expr, original_aggref);
   10935                 :             :                 return;
   10936                 :           0 :         }
   10937                 :             : 
   10938                 :             :         /*
   10939                 :             :          * Mark as PARTIAL, if appropriate.  We look to the original aggref so as
   10940                 :             :          * to avoid printing this when recursing from the code just above.
   10941                 :             :          */
   10942         [ #  # ]:           0 :         if (DO_AGGSPLIT_SKIPFINAL(original_aggref->aggsplit))
   10943                 :           0 :                 appendStringInfoString(buf, "PARTIAL ");
   10944                 :             : 
   10945                 :             :         /* Extract the argument types as seen by the parser */
   10946                 :           0 :         nargs = get_aggregate_argtypes(aggref, argtypes);
   10947                 :             : 
   10948         [ #  # ]:           0 :         if (!funcname)
   10949                 :           0 :                 funcname = generate_function_name(aggref->aggfnoid, nargs, NIL,
   10950                 :           0 :                                                                                   argtypes, aggref->aggvariadic,
   10951                 :             :                                                                                   &use_variadic,
   10952                 :           0 :                                                                                   context->inGroupBy);
   10953                 :             : 
   10954                 :             :         /* Print the aggregate name, schema-qualified if needed */
   10955                 :           0 :         appendStringInfo(buf, "%s(%s", funcname,
   10956                 :           0 :                                          (aggref->aggdistinct != NIL) ? "DISTINCT " : "");
   10957                 :             : 
   10958         [ #  # ]:           0 :         if (AGGKIND_IS_ORDERED_SET(aggref->aggkind))
   10959                 :             :         {
   10960                 :             :                 /*
   10961                 :             :                  * Ordered-set aggregates do not use "*" syntax.  Also, we needn't
   10962                 :             :                  * worry about inserting VARIADIC.  So we can just dump the direct
   10963                 :             :                  * args as-is.
   10964                 :             :                  */
   10965         [ #  # ]:           0 :                 Assert(!aggref->aggvariadic);
   10966                 :           0 :                 get_rule_expr((Node *) aggref->aggdirectargs, context, true);
   10967         [ #  # ]:           0 :                 Assert(aggref->aggorder != NIL);
   10968                 :           0 :                 appendStringInfoString(buf, ") WITHIN GROUP (ORDER BY ");
   10969                 :           0 :                 get_rule_orderby(aggref->aggorder, aggref->args, false, context);
   10970                 :           0 :         }
   10971                 :             :         else
   10972                 :             :         {
   10973                 :             :                 /* aggstar can be set only in zero-argument aggregates */
   10974         [ #  # ]:           0 :                 if (aggref->aggstar)
   10975                 :           0 :                         appendStringInfoChar(buf, '*');
   10976                 :             :                 else
   10977                 :             :                 {
   10978                 :           0 :                         ListCell   *l;
   10979                 :           0 :                         int                     i;
   10980                 :             : 
   10981                 :           0 :                         i = 0;
   10982   [ #  #  #  #  :           0 :                         foreach(l, aggref->args)
                   #  # ]
   10983                 :             :                         {
   10984                 :           0 :                                 TargetEntry *tle = (TargetEntry *) lfirst(l);
   10985                 :           0 :                                 Node       *arg = (Node *) tle->expr;
   10986                 :             : 
   10987         [ #  # ]:           0 :                                 Assert(!IsA(arg, NamedArgExpr));
   10988         [ #  # ]:           0 :                                 if (tle->resjunk)
   10989                 :           0 :                                         continue;
   10990         [ #  # ]:           0 :                                 if (i++ > 0)
   10991                 :             :                                 {
   10992         [ #  # ]:           0 :                                         if (is_json_objectagg)
   10993                 :             :                                         {
   10994                 :             :                                                 /*
   10995                 :             :                                                  * the ABSENT ON NULL and WITH UNIQUE args are printed
   10996                 :             :                                                  * separately, so ignore them here
   10997                 :             :                                                  */
   10998         [ #  # ]:           0 :                                                 if (i > 2)
   10999                 :           0 :                                                         break;
   11000                 :             : 
   11001                 :           0 :                                                 appendStringInfoString(buf, " : ");
   11002                 :           0 :                                         }
   11003                 :             :                                         else
   11004                 :           0 :                                                 appendStringInfoString(buf, ", ");
   11005                 :           0 :                                 }
   11006   [ #  #  #  # ]:           0 :                                 if (use_variadic && i == nargs)
   11007                 :           0 :                                         appendStringInfoString(buf, "VARIADIC ");
   11008                 :           0 :                                 get_rule_expr(arg, context, true);
   11009      [ #  #  # ]:           0 :                         }
   11010                 :           0 :                 }
   11011                 :             : 
   11012         [ #  # ]:           0 :                 if (aggref->aggorder != NIL)
   11013                 :             :                 {
   11014                 :           0 :                         appendStringInfoString(buf, " ORDER BY ");
   11015                 :           0 :                         get_rule_orderby(aggref->aggorder, aggref->args, false, context);
   11016                 :           0 :                 }
   11017                 :             :         }
   11018                 :             : 
   11019         [ #  # ]:           0 :         if (options)
   11020                 :           0 :                 appendStringInfoString(buf, options);
   11021                 :             : 
   11022         [ #  # ]:           0 :         if (aggref->aggfilter != NULL)
   11023                 :             :         {
   11024                 :           0 :                 appendStringInfoString(buf, ") FILTER (WHERE ");
   11025                 :           0 :                 get_rule_expr((Node *) aggref->aggfilter, context, false);
   11026                 :           0 :         }
   11027                 :             : 
   11028                 :           0 :         appendStringInfoChar(buf, ')');
   11029         [ #  # ]:           0 : }
   11030                 :             : 
   11031                 :             : /*
   11032                 :             :  * This is a helper function for get_agg_expr().  It's used when we deparse
   11033                 :             :  * a combining Aggref; resolve_special_varno locates the corresponding partial
   11034                 :             :  * Aggref and then calls this.
   11035                 :             :  */
   11036                 :             : static void
   11037                 :           0 : get_agg_combine_expr(Node *node, deparse_context *context, void *callback_arg)
   11038                 :             : {
   11039                 :           0 :         Aggref     *aggref;
   11040                 :           0 :         Aggref     *original_aggref = callback_arg;
   11041                 :             : 
   11042         [ #  # ]:           0 :         if (!IsA(node, Aggref))
   11043   [ #  #  #  # ]:           0 :                 elog(ERROR, "combining Aggref does not point to an Aggref");
   11044                 :             : 
   11045                 :           0 :         aggref = (Aggref *) node;
   11046                 :           0 :         get_agg_expr(aggref, context, original_aggref);
   11047                 :           0 : }
   11048                 :             : 
   11049                 :             : /*
   11050                 :             :  * get_windowfunc_expr  - Parse back a WindowFunc node
   11051                 :             :  */
   11052                 :             : static void
   11053                 :           0 : get_windowfunc_expr(WindowFunc *wfunc, deparse_context *context)
   11054                 :             : {
   11055                 :           0 :         get_windowfunc_expr_helper(wfunc, context, NULL, NULL, false);
   11056                 :           0 : }
   11057                 :             : 
   11058                 :             : 
   11059                 :             : /*
   11060                 :             :  * get_windowfunc_expr_helper   - subroutine for get_windowfunc_expr and
   11061                 :             :  *                                                              get_json_agg_constructor
   11062                 :             :  */
   11063                 :             : static void
   11064                 :           0 : get_windowfunc_expr_helper(WindowFunc *wfunc, deparse_context *context,
   11065                 :             :                                                    const char *funcname, const char *options,
   11066                 :             :                                                    bool is_json_objectagg)
   11067                 :             : {
   11068                 :           0 :         StringInfo      buf = context->buf;
   11069                 :           0 :         Oid                     argtypes[FUNC_MAX_ARGS];
   11070                 :           0 :         int                     nargs;
   11071                 :           0 :         List       *argnames;
   11072                 :           0 :         ListCell   *l;
   11073                 :             : 
   11074         [ #  # ]:           0 :         if (list_length(wfunc->args) > FUNC_MAX_ARGS)
   11075   [ #  #  #  # ]:           0 :                 ereport(ERROR,
   11076                 :             :                                 (errcode(ERRCODE_TOO_MANY_ARGUMENTS),
   11077                 :             :                                  errmsg("too many arguments")));
   11078                 :           0 :         nargs = 0;
   11079                 :           0 :         argnames = NIL;
   11080   [ #  #  #  #  :           0 :         foreach(l, wfunc->args)
                   #  # ]
   11081                 :             :         {
   11082                 :           0 :                 Node       *arg = (Node *) lfirst(l);
   11083                 :             : 
   11084         [ #  # ]:           0 :                 if (IsA(arg, NamedArgExpr))
   11085                 :           0 :                         argnames = lappend(argnames, ((NamedArgExpr *) arg)->name);
   11086                 :           0 :                 argtypes[nargs] = exprType(arg);
   11087                 :           0 :                 nargs++;
   11088                 :           0 :         }
   11089                 :             : 
   11090         [ #  # ]:           0 :         if (!funcname)
   11091                 :           0 :                 funcname = generate_function_name(wfunc->winfnoid, nargs, argnames,
   11092                 :           0 :                                                                                   argtypes, false, NULL,
   11093                 :           0 :                                                                                   context->inGroupBy);
   11094                 :             : 
   11095                 :           0 :         appendStringInfo(buf, "%s(", funcname);
   11096                 :             : 
   11097                 :             :         /* winstar can be set only in zero-argument aggregates */
   11098         [ #  # ]:           0 :         if (wfunc->winstar)
   11099                 :           0 :                 appendStringInfoChar(buf, '*');
   11100                 :             :         else
   11101                 :             :         {
   11102         [ #  # ]:           0 :                 if (is_json_objectagg)
   11103                 :             :                 {
   11104                 :           0 :                         get_rule_expr((Node *) linitial(wfunc->args), context, false);
   11105                 :           0 :                         appendStringInfoString(buf, " : ");
   11106                 :           0 :                         get_rule_expr((Node *) lsecond(wfunc->args), context, false);
   11107                 :           0 :                 }
   11108                 :             :                 else
   11109                 :           0 :                         get_rule_expr((Node *) wfunc->args, context, true);
   11110                 :             :         }
   11111                 :             : 
   11112         [ #  # ]:           0 :         if (options)
   11113                 :           0 :                 appendStringInfoString(buf, options);
   11114                 :             : 
   11115         [ #  # ]:           0 :         if (wfunc->aggfilter != NULL)
   11116                 :             :         {
   11117                 :           0 :                 appendStringInfoString(buf, ") FILTER (WHERE ");
   11118                 :           0 :                 get_rule_expr((Node *) wfunc->aggfilter, context, false);
   11119                 :           0 :         }
   11120                 :             : 
   11121                 :           0 :         appendStringInfoString(buf, ") ");
   11122                 :             : 
   11123         [ #  # ]:           0 :         if (wfunc->ignore_nulls == PARSER_IGNORE_NULLS)
   11124                 :           0 :                 appendStringInfoString(buf, "IGNORE NULLS ");
   11125                 :             : 
   11126                 :           0 :         appendStringInfoString(buf, "OVER ");
   11127                 :             : 
   11128         [ #  # ]:           0 :         if (context->windowClause)
   11129                 :             :         {
   11130                 :             :                 /* Query-decompilation case: search the windowClause list */
   11131   [ #  #  #  #  :           0 :                 foreach(l, context->windowClause)
                   #  # ]
   11132                 :             :                 {
   11133                 :           0 :                         WindowClause *wc = (WindowClause *) lfirst(l);
   11134                 :             : 
   11135         [ #  # ]:           0 :                         if (wc->winref == wfunc->winref)
   11136                 :             :                         {
   11137         [ #  # ]:           0 :                                 if (wc->name)
   11138                 :           0 :                                         appendStringInfoString(buf, quote_identifier(wc->name));
   11139                 :             :                                 else
   11140                 :           0 :                                         get_rule_windowspec(wc, context->targetList, context);
   11141                 :           0 :                                 break;
   11142                 :             :                         }
   11143         [ #  # ]:           0 :                 }
   11144         [ #  # ]:           0 :                 if (l == NULL)
   11145   [ #  #  #  # ]:           0 :                         elog(ERROR, "could not find window clause for winref %u",
   11146                 :             :                                  wfunc->winref);
   11147                 :           0 :         }
   11148                 :             :         else
   11149                 :             :         {
   11150                 :             :                 /*
   11151                 :             :                  * In EXPLAIN, search the namespace stack for a matching WindowAgg
   11152                 :             :                  * node (probably it's always the first entry), and print winname.
   11153                 :             :                  */
   11154   [ #  #  #  #  :           0 :                 foreach(l, context->namespaces)
                   #  # ]
   11155                 :             :                 {
   11156                 :           0 :                         deparse_namespace *dpns = (deparse_namespace *) lfirst(l);
   11157                 :             : 
   11158   [ #  #  #  # ]:           0 :                         if (dpns->plan && IsA(dpns->plan, WindowAgg))
   11159                 :             :                         {
   11160                 :           0 :                                 WindowAgg  *wagg = (WindowAgg *) dpns->plan;
   11161                 :             : 
   11162         [ #  # ]:           0 :                                 if (wagg->winref == wfunc->winref)
   11163                 :             :                                 {
   11164                 :           0 :                                         appendStringInfoString(buf, quote_identifier(wagg->winname));
   11165                 :           0 :                                         break;
   11166                 :             :                                 }
   11167         [ #  # ]:           0 :                         }
   11168         [ #  # ]:           0 :                 }
   11169         [ #  # ]:           0 :                 if (l == NULL)
   11170   [ #  #  #  # ]:           0 :                         elog(ERROR, "could not find window clause for winref %u",
   11171                 :             :                                  wfunc->winref);
   11172                 :             :         }
   11173                 :           0 : }
   11174                 :             : 
   11175                 :             : /*
   11176                 :             :  * get_func_sql_syntax          - Parse back a SQL-syntax function call
   11177                 :             :  *
   11178                 :             :  * Returns true if we successfully deparsed, false if we did not
   11179                 :             :  * recognize the function.
   11180                 :             :  */
   11181                 :             : static bool
   11182                 :           0 : get_func_sql_syntax(FuncExpr *expr, deparse_context *context)
   11183                 :             : {
   11184                 :           0 :         StringInfo      buf = context->buf;
   11185                 :           0 :         Oid                     funcoid = expr->funcid;
   11186                 :             : 
   11187   [ #  #  #  #  :           0 :         switch (funcoid)
          #  #  #  #  #  
          #  #  #  #  #  
                #  #  # ]
   11188                 :             :         {
   11189                 :             :                 case F_TIMEZONE_INTERVAL_TIMESTAMP:
   11190                 :             :                 case F_TIMEZONE_INTERVAL_TIMESTAMPTZ:
   11191                 :             :                 case F_TIMEZONE_INTERVAL_TIMETZ:
   11192                 :             :                 case F_TIMEZONE_TEXT_TIMESTAMP:
   11193                 :             :                 case F_TIMEZONE_TEXT_TIMESTAMPTZ:
   11194                 :             :                 case F_TIMEZONE_TEXT_TIMETZ:
   11195                 :             :                         /* AT TIME ZONE ... note reversed argument order */
   11196                 :           0 :                         appendStringInfoChar(buf, '(');
   11197                 :           0 :                         get_rule_expr_paren((Node *) lsecond(expr->args), context, false,
   11198                 :           0 :                                                                 (Node *) expr);
   11199                 :           0 :                         appendStringInfoString(buf, " AT TIME ZONE ");
   11200                 :           0 :                         get_rule_expr_paren((Node *) linitial(expr->args), context, false,
   11201                 :           0 :                                                                 (Node *) expr);
   11202                 :           0 :                         appendStringInfoChar(buf, ')');
   11203                 :           0 :                         return true;
   11204                 :             : 
   11205                 :             :                 case F_TIMEZONE_TIMESTAMP:
   11206                 :             :                 case F_TIMEZONE_TIMESTAMPTZ:
   11207                 :             :                 case F_TIMEZONE_TIMETZ:
   11208                 :             :                         /* AT LOCAL */
   11209                 :           0 :                         appendStringInfoChar(buf, '(');
   11210                 :           0 :                         get_rule_expr_paren((Node *) linitial(expr->args), context, false,
   11211                 :           0 :                                                                 (Node *) expr);
   11212                 :           0 :                         appendStringInfoString(buf, " AT LOCAL)");
   11213                 :           0 :                         return true;
   11214                 :             : 
   11215                 :             :                 case F_OVERLAPS_TIMESTAMPTZ_INTERVAL_TIMESTAMPTZ_INTERVAL:
   11216                 :             :                 case F_OVERLAPS_TIMESTAMPTZ_INTERVAL_TIMESTAMPTZ_TIMESTAMPTZ:
   11217                 :             :                 case F_OVERLAPS_TIMESTAMPTZ_TIMESTAMPTZ_TIMESTAMPTZ_INTERVAL:
   11218                 :             :                 case F_OVERLAPS_TIMESTAMPTZ_TIMESTAMPTZ_TIMESTAMPTZ_TIMESTAMPTZ:
   11219                 :             :                 case F_OVERLAPS_TIMESTAMP_INTERVAL_TIMESTAMP_INTERVAL:
   11220                 :             :                 case F_OVERLAPS_TIMESTAMP_INTERVAL_TIMESTAMP_TIMESTAMP:
   11221                 :             :                 case F_OVERLAPS_TIMESTAMP_TIMESTAMP_TIMESTAMP_INTERVAL:
   11222                 :             :                 case F_OVERLAPS_TIMESTAMP_TIMESTAMP_TIMESTAMP_TIMESTAMP:
   11223                 :             :                 case F_OVERLAPS_TIMETZ_TIMETZ_TIMETZ_TIMETZ:
   11224                 :             :                 case F_OVERLAPS_TIME_INTERVAL_TIME_INTERVAL:
   11225                 :             :                 case F_OVERLAPS_TIME_INTERVAL_TIME_TIME:
   11226                 :             :                 case F_OVERLAPS_TIME_TIME_TIME_INTERVAL:
   11227                 :             :                 case F_OVERLAPS_TIME_TIME_TIME_TIME:
   11228                 :             :                         /* (x1, x2) OVERLAPS (y1, y2) */
   11229                 :           0 :                         appendStringInfoString(buf, "((");
   11230                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11231                 :           0 :                         appendStringInfoString(buf, ", ");
   11232                 :           0 :                         get_rule_expr((Node *) lsecond(expr->args), context, false);
   11233                 :           0 :                         appendStringInfoString(buf, ") OVERLAPS (");
   11234                 :           0 :                         get_rule_expr((Node *) lthird(expr->args), context, false);
   11235                 :           0 :                         appendStringInfoString(buf, ", ");
   11236                 :           0 :                         get_rule_expr((Node *) lfourth(expr->args), context, false);
   11237                 :           0 :                         appendStringInfoString(buf, "))");
   11238                 :           0 :                         return true;
   11239                 :             : 
   11240                 :             :                 case F_EXTRACT_TEXT_DATE:
   11241                 :             :                 case F_EXTRACT_TEXT_TIME:
   11242                 :             :                 case F_EXTRACT_TEXT_TIMETZ:
   11243                 :             :                 case F_EXTRACT_TEXT_TIMESTAMP:
   11244                 :             :                 case F_EXTRACT_TEXT_TIMESTAMPTZ:
   11245                 :             :                 case F_EXTRACT_TEXT_INTERVAL:
   11246                 :             :                         /* EXTRACT (x FROM y) */
   11247                 :           0 :                         appendStringInfoString(buf, "EXTRACT(");
   11248                 :             :                         {
   11249                 :           0 :                                 Const      *con = (Const *) linitial(expr->args);
   11250                 :             : 
   11251         [ #  # ]:           0 :                                 Assert(IsA(con, Const) &&
   11252                 :             :                                            con->consttype == TEXTOID &&
   11253                 :             :                                            !con->constisnull);
   11254                 :           0 :                                 appendStringInfoString(buf, TextDatumGetCString(con->constvalue));
   11255                 :           0 :                         }
   11256                 :           0 :                         appendStringInfoString(buf, " FROM ");
   11257                 :           0 :                         get_rule_expr((Node *) lsecond(expr->args), context, false);
   11258                 :           0 :                         appendStringInfoChar(buf, ')');
   11259                 :           0 :                         return true;
   11260                 :             : 
   11261                 :             :                 case F_IS_NORMALIZED:
   11262                 :             :                         /* IS xxx NORMALIZED */
   11263                 :           0 :                         appendStringInfoChar(buf, '(');
   11264                 :           0 :                         get_rule_expr_paren((Node *) linitial(expr->args), context, false,
   11265                 :           0 :                                                                 (Node *) expr);
   11266                 :           0 :                         appendStringInfoString(buf, " IS");
   11267         [ #  # ]:           0 :                         if (list_length(expr->args) == 2)
   11268                 :             :                         {
   11269                 :           0 :                                 Const      *con = (Const *) lsecond(expr->args);
   11270                 :             : 
   11271         [ #  # ]:           0 :                                 Assert(IsA(con, Const) &&
   11272                 :             :                                            con->consttype == TEXTOID &&
   11273                 :             :                                            !con->constisnull);
   11274                 :           0 :                                 appendStringInfo(buf, " %s",
   11275                 :           0 :                                                                  TextDatumGetCString(con->constvalue));
   11276                 :           0 :                         }
   11277                 :           0 :                         appendStringInfoString(buf, " NORMALIZED)");
   11278                 :           0 :                         return true;
   11279                 :             : 
   11280                 :             :                 case F_PG_COLLATION_FOR:
   11281                 :             :                         /* COLLATION FOR */
   11282                 :           0 :                         appendStringInfoString(buf, "COLLATION FOR (");
   11283                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11284                 :           0 :                         appendStringInfoChar(buf, ')');
   11285                 :           0 :                         return true;
   11286                 :             : 
   11287                 :             :                 case F_NORMALIZE:
   11288                 :             :                         /* NORMALIZE() */
   11289                 :           0 :                         appendStringInfoString(buf, "NORMALIZE(");
   11290                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11291         [ #  # ]:           0 :                         if (list_length(expr->args) == 2)
   11292                 :             :                         {
   11293                 :           0 :                                 Const      *con = (Const *) lsecond(expr->args);
   11294                 :             : 
   11295         [ #  # ]:           0 :                                 Assert(IsA(con, Const) &&
   11296                 :             :                                            con->consttype == TEXTOID &&
   11297                 :             :                                            !con->constisnull);
   11298                 :           0 :                                 appendStringInfo(buf, ", %s",
   11299                 :           0 :                                                                  TextDatumGetCString(con->constvalue));
   11300                 :           0 :                         }
   11301                 :           0 :                         appendStringInfoChar(buf, ')');
   11302                 :           0 :                         return true;
   11303                 :             : 
   11304                 :             :                 case F_OVERLAY_BIT_BIT_INT4:
   11305                 :             :                 case F_OVERLAY_BIT_BIT_INT4_INT4:
   11306                 :             :                 case F_OVERLAY_BYTEA_BYTEA_INT4:
   11307                 :             :                 case F_OVERLAY_BYTEA_BYTEA_INT4_INT4:
   11308                 :             :                 case F_OVERLAY_TEXT_TEXT_INT4:
   11309                 :             :                 case F_OVERLAY_TEXT_TEXT_INT4_INT4:
   11310                 :             :                         /* OVERLAY() */
   11311                 :           0 :                         appendStringInfoString(buf, "OVERLAY(");
   11312                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11313                 :           0 :                         appendStringInfoString(buf, " PLACING ");
   11314                 :           0 :                         get_rule_expr((Node *) lsecond(expr->args), context, false);
   11315                 :           0 :                         appendStringInfoString(buf, " FROM ");
   11316                 :           0 :                         get_rule_expr((Node *) lthird(expr->args), context, false);
   11317         [ #  # ]:           0 :                         if (list_length(expr->args) == 4)
   11318                 :             :                         {
   11319                 :           0 :                                 appendStringInfoString(buf, " FOR ");
   11320                 :           0 :                                 get_rule_expr((Node *) lfourth(expr->args), context, false);
   11321                 :           0 :                         }
   11322                 :           0 :                         appendStringInfoChar(buf, ')');
   11323                 :           0 :                         return true;
   11324                 :             : 
   11325                 :             :                 case F_POSITION_BIT_BIT:
   11326                 :             :                 case F_POSITION_BYTEA_BYTEA:
   11327                 :             :                 case F_POSITION_TEXT_TEXT:
   11328                 :             :                         /* POSITION() ... extra parens since args are b_expr not a_expr */
   11329                 :           0 :                         appendStringInfoString(buf, "POSITION((");
   11330                 :           0 :                         get_rule_expr((Node *) lsecond(expr->args), context, false);
   11331                 :           0 :                         appendStringInfoString(buf, ") IN (");
   11332                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11333                 :           0 :                         appendStringInfoString(buf, "))");
   11334                 :           0 :                         return true;
   11335                 :             : 
   11336                 :             :                 case F_SUBSTRING_BIT_INT4:
   11337                 :             :                 case F_SUBSTRING_BIT_INT4_INT4:
   11338                 :             :                 case F_SUBSTRING_BYTEA_INT4:
   11339                 :             :                 case F_SUBSTRING_BYTEA_INT4_INT4:
   11340                 :             :                 case F_SUBSTRING_TEXT_INT4:
   11341                 :             :                 case F_SUBSTRING_TEXT_INT4_INT4:
   11342                 :             :                         /* SUBSTRING FROM/FOR (i.e., integer-position variants) */
   11343                 :           0 :                         appendStringInfoString(buf, "SUBSTRING(");
   11344                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11345                 :           0 :                         appendStringInfoString(buf, " FROM ");
   11346                 :           0 :                         get_rule_expr((Node *) lsecond(expr->args), context, false);
   11347         [ #  # ]:           0 :                         if (list_length(expr->args) == 3)
   11348                 :             :                         {
   11349                 :           0 :                                 appendStringInfoString(buf, " FOR ");
   11350                 :           0 :                                 get_rule_expr((Node *) lthird(expr->args), context, false);
   11351                 :           0 :                         }
   11352                 :           0 :                         appendStringInfoChar(buf, ')');
   11353                 :           0 :                         return true;
   11354                 :             : 
   11355                 :             :                 case F_SUBSTRING_TEXT_TEXT_TEXT:
   11356                 :             :                         /* SUBSTRING SIMILAR/ESCAPE */
   11357                 :           0 :                         appendStringInfoString(buf, "SUBSTRING(");
   11358                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11359                 :           0 :                         appendStringInfoString(buf, " SIMILAR ");
   11360                 :           0 :                         get_rule_expr((Node *) lsecond(expr->args), context, false);
   11361                 :           0 :                         appendStringInfoString(buf, " ESCAPE ");
   11362                 :           0 :                         get_rule_expr((Node *) lthird(expr->args), context, false);
   11363                 :           0 :                         appendStringInfoChar(buf, ')');
   11364                 :           0 :                         return true;
   11365                 :             : 
   11366                 :             :                 case F_BTRIM_BYTEA_BYTEA:
   11367                 :             :                 case F_BTRIM_TEXT:
   11368                 :             :                 case F_BTRIM_TEXT_TEXT:
   11369                 :             :                         /* TRIM() */
   11370                 :           0 :                         appendStringInfoString(buf, "TRIM(BOTH");
   11371         [ #  # ]:           0 :                         if (list_length(expr->args) == 2)
   11372                 :             :                         {
   11373                 :           0 :                                 appendStringInfoChar(buf, ' ');
   11374                 :           0 :                                 get_rule_expr((Node *) lsecond(expr->args), context, false);
   11375                 :           0 :                         }
   11376                 :           0 :                         appendStringInfoString(buf, " FROM ");
   11377                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11378                 :           0 :                         appendStringInfoChar(buf, ')');
   11379                 :           0 :                         return true;
   11380                 :             : 
   11381                 :             :                 case F_LTRIM_BYTEA_BYTEA:
   11382                 :             :                 case F_LTRIM_TEXT:
   11383                 :             :                 case F_LTRIM_TEXT_TEXT:
   11384                 :             :                         /* TRIM() */
   11385                 :           0 :                         appendStringInfoString(buf, "TRIM(LEADING");
   11386         [ #  # ]:           0 :                         if (list_length(expr->args) == 2)
   11387                 :             :                         {
   11388                 :           0 :                                 appendStringInfoChar(buf, ' ');
   11389                 :           0 :                                 get_rule_expr((Node *) lsecond(expr->args), context, false);
   11390                 :           0 :                         }
   11391                 :           0 :                         appendStringInfoString(buf, " FROM ");
   11392                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11393                 :           0 :                         appendStringInfoChar(buf, ')');
   11394                 :           0 :                         return true;
   11395                 :             : 
   11396                 :             :                 case F_RTRIM_BYTEA_BYTEA:
   11397                 :             :                 case F_RTRIM_TEXT:
   11398                 :             :                 case F_RTRIM_TEXT_TEXT:
   11399                 :             :                         /* TRIM() */
   11400                 :           0 :                         appendStringInfoString(buf, "TRIM(TRAILING");
   11401         [ #  # ]:           0 :                         if (list_length(expr->args) == 2)
   11402                 :             :                         {
   11403                 :           0 :                                 appendStringInfoChar(buf, ' ');
   11404                 :           0 :                                 get_rule_expr((Node *) lsecond(expr->args), context, false);
   11405                 :           0 :                         }
   11406                 :           0 :                         appendStringInfoString(buf, " FROM ");
   11407                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11408                 :           0 :                         appendStringInfoChar(buf, ')');
   11409                 :           0 :                         return true;
   11410                 :             : 
   11411                 :             :                 case F_SYSTEM_USER:
   11412                 :           0 :                         appendStringInfoString(buf, "SYSTEM_USER");
   11413                 :           0 :                         return true;
   11414                 :             : 
   11415                 :             :                 case F_XMLEXISTS:
   11416                 :             :                         /* XMLEXISTS ... extra parens because args are c_expr */
   11417                 :           0 :                         appendStringInfoString(buf, "XMLEXISTS((");
   11418                 :           0 :                         get_rule_expr((Node *) linitial(expr->args), context, false);
   11419                 :           0 :                         appendStringInfoString(buf, ") PASSING (");
   11420                 :           0 :                         get_rule_expr((Node *) lsecond(expr->args), context, false);
   11421                 :           0 :                         appendStringInfoString(buf, "))");
   11422                 :           0 :                         return true;
   11423                 :             :         }
   11424                 :           0 :         return false;
   11425                 :           0 : }
   11426                 :             : 
   11427                 :             : /* ----------
   11428                 :             :  * get_coercion_expr
   11429                 :             :  *
   11430                 :             :  *      Make a string representation of a value coerced to a specific type
   11431                 :             :  * ----------
   11432                 :             :  */
   11433                 :             : static void
   11434                 :           0 : get_coercion_expr(Node *arg, deparse_context *context,
   11435                 :             :                                   Oid resulttype, int32 resulttypmod,
   11436                 :             :                                   Node *parentNode)
   11437                 :             : {
   11438                 :           0 :         StringInfo      buf = context->buf;
   11439                 :             : 
   11440                 :             :         /*
   11441                 :             :          * Since parse_coerce.c doesn't immediately collapse application of
   11442                 :             :          * length-coercion functions to constants, what we'll typically see in
   11443                 :             :          * such cases is a Const with typmod -1 and a length-coercion function
   11444                 :             :          * right above it.  Avoid generating redundant output. However, beware of
   11445                 :             :          * suppressing casts when the user actually wrote something like
   11446                 :             :          * 'foo'::text::char(3).
   11447                 :             :          *
   11448                 :             :          * Note: it might seem that we are missing the possibility of needing to
   11449                 :             :          * print a COLLATE clause for such a Const.  However, a Const could only
   11450                 :             :          * have nondefault collation in a post-constant-folding tree, in which the
   11451                 :             :          * length coercion would have been folded too.  See also the special
   11452                 :             :          * handling of CollateExpr in coerce_to_target_type(): any collation
   11453                 :             :          * marking will be above the coercion node, not below it.
   11454                 :             :          */
   11455   [ #  #  #  # ]:           0 :         if (arg && IsA(arg, Const) &&
   11456   [ #  #  #  # ]:           0 :                 ((Const *) arg)->consttype == resulttype &&
   11457                 :           0 :                 ((Const *) arg)->consttypmod == -1)
   11458                 :             :         {
   11459                 :             :                 /* Show the constant without normal ::typename decoration */
   11460                 :           0 :                 get_const_expr((Const *) arg, context, -1);
   11461                 :           0 :         }
   11462                 :             :         else
   11463                 :             :         {
   11464         [ #  # ]:           0 :                 if (!PRETTY_PAREN(context))
   11465                 :           0 :                         appendStringInfoChar(buf, '(');
   11466                 :           0 :                 get_rule_expr_paren(arg, context, false, parentNode);
   11467         [ #  # ]:           0 :                 if (!PRETTY_PAREN(context))
   11468                 :           0 :                         appendStringInfoChar(buf, ')');
   11469                 :             :         }
   11470                 :             : 
   11471                 :             :         /*
   11472                 :             :          * Never emit resulttype(arg) functional notation. A pg_proc entry could
   11473                 :             :          * take precedence, and a resulttype in pg_temp would require schema
   11474                 :             :          * qualification that format_type_with_typemod() would usually omit. We've
   11475                 :             :          * standardized on arg::resulttype, but CAST(arg AS resulttype) notation
   11476                 :             :          * would work fine.
   11477                 :             :          */
   11478                 :           0 :         appendStringInfo(buf, "::%s",
   11479                 :           0 :                                          format_type_with_typemod(resulttype, resulttypmod));
   11480                 :           0 : }
   11481                 :             : 
   11482                 :             : /* ----------
   11483                 :             :  * get_const_expr
   11484                 :             :  *
   11485                 :             :  *      Make a string representation of a Const
   11486                 :             :  *
   11487                 :             :  * showtype can be -1 to never show "::typename" decoration, or +1 to always
   11488                 :             :  * show it, or 0 to show it only if the constant wouldn't be assumed to be
   11489                 :             :  * the right type by default.
   11490                 :             :  *
   11491                 :             :  * If the Const's collation isn't default for its type, show that too.
   11492                 :             :  * We mustn't do this when showtype is -1 (since that means the caller will
   11493                 :             :  * print "::typename", and we can't put a COLLATE clause in between).  It's
   11494                 :             :  * caller's responsibility that collation isn't missed in such cases.
   11495                 :             :  * ----------
   11496                 :             :  */
   11497                 :             : static void
   11498                 :           0 : get_const_expr(Const *constval, deparse_context *context, int showtype)
   11499                 :             : {
   11500                 :           0 :         StringInfo      buf = context->buf;
   11501                 :           0 :         Oid                     typoutput;
   11502                 :           0 :         bool            typIsVarlena;
   11503                 :           0 :         char       *extval;
   11504                 :           0 :         bool            needlabel = false;
   11505                 :             : 
   11506         [ #  # ]:           0 :         if (constval->constisnull)
   11507                 :             :         {
   11508                 :             :                 /*
   11509                 :             :                  * Always label the type of a NULL constant to prevent misdecisions
   11510                 :             :                  * about type when reparsing.
   11511                 :             :                  */
   11512                 :           0 :                 appendStringInfoString(buf, "NULL");
   11513         [ #  # ]:           0 :                 if (showtype >= 0)
   11514                 :             :                 {
   11515                 :           0 :                         appendStringInfo(buf, "::%s",
   11516                 :           0 :                                                          format_type_with_typemod(constval->consttype,
   11517                 :           0 :                                                                                                           constval->consttypmod));
   11518                 :           0 :                         get_const_collation(constval, context);
   11519                 :           0 :                 }
   11520                 :           0 :                 return;
   11521                 :             :         }
   11522                 :             : 
   11523                 :           0 :         getTypeOutputInfo(constval->consttype,
   11524                 :             :                                           &typoutput, &typIsVarlena);
   11525                 :             : 
   11526                 :           0 :         extval = OidOutputFunctionCall(typoutput, constval->constvalue);
   11527                 :             : 
   11528   [ #  #  #  # ]:           0 :         switch (constval->consttype)
   11529                 :             :         {
   11530                 :             :                 case INT4OID:
   11531                 :             : 
   11532                 :             :                         /*
   11533                 :             :                          * INT4 can be printed without any decoration, unless it is
   11534                 :             :                          * negative; in that case print it as '-nnn'::integer to ensure
   11535                 :             :                          * that the output will re-parse as a constant, not as a constant
   11536                 :             :                          * plus operator.  In most cases we could get away with printing
   11537                 :             :                          * (-nnn) instead, because of the way that gram.y handles negative
   11538                 :             :                          * literals; but that doesn't work for INT_MIN, and it doesn't
   11539                 :             :                          * seem that much prettier anyway.
   11540                 :             :                          */
   11541         [ #  # ]:           0 :                         if (extval[0] != '-')
   11542                 :           0 :                                 appendStringInfoString(buf, extval);
   11543                 :             :                         else
   11544                 :             :                         {
   11545                 :           0 :                                 appendStringInfo(buf, "'%s'", extval);
   11546                 :           0 :                                 needlabel = true;       /* we must attach a cast */
   11547                 :             :                         }
   11548                 :           0 :                         break;
   11549                 :             : 
   11550                 :             :                 case NUMERICOID:
   11551                 :             : 
   11552                 :             :                         /*
   11553                 :             :                          * NUMERIC can be printed without quotes if it looks like a float
   11554                 :             :                          * constant (not an integer, and not Infinity or NaN) and doesn't
   11555                 :             :                          * have a leading sign (for the same reason as for INT4).
   11556                 :             :                          */
   11557   [ #  #  #  # ]:           0 :                         if (isdigit((unsigned char) extval[0]) &&
   11558                 :           0 :                                 strcspn(extval, "eE.") != strlen(extval))
   11559                 :             :                         {
   11560                 :           0 :                                 appendStringInfoString(buf, extval);
   11561                 :           0 :                         }
   11562                 :             :                         else
   11563                 :             :                         {
   11564                 :           0 :                                 appendStringInfo(buf, "'%s'", extval);
   11565                 :           0 :                                 needlabel = true;       /* we must attach a cast */
   11566                 :             :                         }
   11567                 :           0 :                         break;
   11568                 :             : 
   11569                 :             :                 case BOOLOID:
   11570         [ #  # ]:           0 :                         if (strcmp(extval, "t") == 0)
   11571                 :           0 :                                 appendStringInfoString(buf, "true");
   11572                 :             :                         else
   11573                 :           0 :                                 appendStringInfoString(buf, "false");
   11574                 :           0 :                         break;
   11575                 :             : 
   11576                 :             :                 default:
   11577                 :           0 :                         simple_quote_literal(buf, extval);
   11578                 :           0 :                         break;
   11579                 :             :         }
   11580                 :             : 
   11581                 :           0 :         pfree(extval);
   11582                 :             : 
   11583         [ #  # ]:           0 :         if (showtype < 0)
   11584                 :           0 :                 return;
   11585                 :             : 
   11586                 :             :         /*
   11587                 :             :          * For showtype == 0, append ::typename unless the constant will be
   11588                 :             :          * implicitly typed as the right type when it is read in.
   11589                 :             :          *
   11590                 :             :          * XXX this code has to be kept in sync with the behavior of the parser,
   11591                 :             :          * especially make_const.
   11592                 :             :          */
   11593   [ #  #  #  # ]:           0 :         switch (constval->consttype)
   11594                 :             :         {
   11595                 :             :                 case BOOLOID:
   11596                 :             :                 case UNKNOWNOID:
   11597                 :             :                         /* These types can be left unlabeled */
   11598                 :           0 :                         needlabel = false;
   11599                 :           0 :                         break;
   11600                 :             :                 case INT4OID:
   11601                 :             :                         /* We determined above whether a label is needed */
   11602                 :             :                         break;
   11603                 :             :                 case NUMERICOID:
   11604                 :             : 
   11605                 :             :                         /*
   11606                 :             :                          * Float-looking constants will be typed as numeric, which we
   11607                 :             :                          * checked above; but if there's a nondefault typmod we need to
   11608                 :             :                          * show it.
   11609                 :             :                          */
   11610                 :           0 :                         needlabel |= (constval->consttypmod >= 0);
   11611                 :           0 :                         break;
   11612                 :             :                 default:
   11613                 :           0 :                         needlabel = true;
   11614                 :           0 :                         break;
   11615                 :             :         }
   11616   [ #  #  #  # ]:           0 :         if (needlabel || showtype > 0)
   11617                 :           0 :                 appendStringInfo(buf, "::%s",
   11618                 :           0 :                                                  format_type_with_typemod(constval->consttype,
   11619                 :           0 :                                                                                                   constval->consttypmod));
   11620                 :             : 
   11621                 :           0 :         get_const_collation(constval, context);
   11622         [ #  # ]:           0 : }
   11623                 :             : 
   11624                 :             : /*
   11625                 :             :  * helper for get_const_expr: append COLLATE if needed
   11626                 :             :  */
   11627                 :             : static void
   11628                 :           0 : get_const_collation(Const *constval, deparse_context *context)
   11629                 :             : {
   11630                 :           0 :         StringInfo      buf = context->buf;
   11631                 :             : 
   11632         [ #  # ]:           0 :         if (OidIsValid(constval->constcollid))
   11633                 :             :         {
   11634                 :           0 :                 Oid                     typcollation = get_typcollation(constval->consttype);
   11635                 :             : 
   11636         [ #  # ]:           0 :                 if (constval->constcollid != typcollation)
   11637                 :             :                 {
   11638                 :           0 :                         appendStringInfo(buf, " COLLATE %s",
   11639                 :           0 :                                                          generate_collation_name(constval->constcollid));
   11640                 :           0 :                 }
   11641                 :           0 :         }
   11642                 :           0 : }
   11643                 :             : 
   11644                 :             : /*
   11645                 :             :  * get_json_path_spec           - Parse back a JSON path specification
   11646                 :             :  */
   11647                 :             : static void
   11648                 :           0 : get_json_path_spec(Node *path_spec, deparse_context *context, bool showimplicit)
   11649                 :             : {
   11650         [ #  # ]:           0 :         if (IsA(path_spec, Const))
   11651                 :           0 :                 get_const_expr((Const *) path_spec, context, -1);
   11652                 :             :         else
   11653                 :           0 :                 get_rule_expr(path_spec, context, showimplicit);
   11654                 :           0 : }
   11655                 :             : 
   11656                 :             : /*
   11657                 :             :  * get_json_format                      - Parse back a JsonFormat node
   11658                 :             :  */
   11659                 :             : static void
   11660                 :           0 : get_json_format(JsonFormat *format, StringInfo buf)
   11661                 :             : {
   11662         [ #  # ]:           0 :         if (format->format_type == JS_FORMAT_DEFAULT)
   11663                 :           0 :                 return;
   11664                 :             : 
   11665                 :           0 :         appendStringInfoString(buf,
   11666                 :           0 :                                                    format->format_type == JS_FORMAT_JSONB ?
   11667                 :             :                                                    " FORMAT JSONB" : " FORMAT JSON");
   11668                 :             : 
   11669         [ #  # ]:           0 :         if (format->encoding != JS_ENC_DEFAULT)
   11670                 :             :         {
   11671                 :           0 :                 const char *encoding;
   11672                 :             : 
   11673                 :           0 :                 encoding =
   11674         [ #  # ]:           0 :                         format->encoding == JS_ENC_UTF16 ? "UTF16" :
   11675                 :           0 :                         format->encoding == JS_ENC_UTF32 ? "UTF32" : "UTF8";
   11676                 :             : 
   11677                 :           0 :                 appendStringInfo(buf, " ENCODING %s", encoding);
   11678                 :           0 :         }
   11679                 :           0 : }
   11680                 :             : 
   11681                 :             : /*
   11682                 :             :  * get_json_returning           - Parse back a JsonReturning structure
   11683                 :             :  */
   11684                 :             : static void
   11685                 :           0 : get_json_returning(JsonReturning *returning, StringInfo buf,
   11686                 :             :                                    bool json_format_by_default)
   11687                 :             : {
   11688         [ #  # ]:           0 :         if (!OidIsValid(returning->typid))
   11689                 :           0 :                 return;
   11690                 :             : 
   11691                 :           0 :         appendStringInfo(buf, " RETURNING %s",
   11692                 :           0 :                                          format_type_with_typemod(returning->typid,
   11693                 :           0 :                                                                                           returning->typmod));
   11694                 :             : 
   11695   [ #  #  #  # ]:           0 :         if (!json_format_by_default ||
   11696                 :           0 :                 returning->format->format_type !=
   11697                 :           0 :                 (returning->typid == JSONBOID ? JS_FORMAT_JSONB : JS_FORMAT_JSON))
   11698                 :           0 :                 get_json_format(returning->format, buf);
   11699                 :           0 : }
   11700                 :             : 
   11701                 :             : /*
   11702                 :             :  * get_json_constructor         - Parse back a JsonConstructorExpr node
   11703                 :             :  */
   11704                 :             : static void
   11705                 :           0 : get_json_constructor(JsonConstructorExpr *ctor, deparse_context *context,
   11706                 :             :                                          bool showimplicit)
   11707                 :             : {
   11708                 :           0 :         StringInfo      buf = context->buf;
   11709                 :           0 :         const char *funcname;
   11710                 :           0 :         bool            is_json_object;
   11711                 :           0 :         int                     curridx;
   11712                 :           0 :         ListCell   *lc;
   11713                 :             : 
   11714         [ #  # ]:           0 :         if (ctor->type == JSCTOR_JSON_OBJECTAGG)
   11715                 :             :         {
   11716                 :           0 :                 get_json_agg_constructor(ctor, context, "JSON_OBJECTAGG", true);
   11717                 :           0 :                 return;
   11718                 :             :         }
   11719         [ #  # ]:           0 :         else if (ctor->type == JSCTOR_JSON_ARRAYAGG)
   11720                 :             :         {
   11721                 :           0 :                 get_json_agg_constructor(ctor, context, "JSON_ARRAYAGG", false);
   11722                 :           0 :                 return;
   11723                 :             :         }
   11724                 :             : 
   11725   [ #  #  #  #  :           0 :         switch (ctor->type)
                   #  # ]
   11726                 :             :         {
   11727                 :             :                 case JSCTOR_JSON_OBJECT:
   11728                 :           0 :                         funcname = "JSON_OBJECT";
   11729                 :           0 :                         break;
   11730                 :             :                 case JSCTOR_JSON_ARRAY:
   11731                 :           0 :                         funcname = "JSON_ARRAY";
   11732                 :           0 :                         break;
   11733                 :             :                 case JSCTOR_JSON_PARSE:
   11734                 :           0 :                         funcname = "JSON";
   11735                 :           0 :                         break;
   11736                 :             :                 case JSCTOR_JSON_SCALAR:
   11737                 :           0 :                         funcname = "JSON_SCALAR";
   11738                 :           0 :                         break;
   11739                 :             :                 case JSCTOR_JSON_SERIALIZE:
   11740                 :           0 :                         funcname = "JSON_SERIALIZE";
   11741                 :           0 :                         break;
   11742                 :             :                 default:
   11743   [ #  #  #  # ]:           0 :                         elog(ERROR, "invalid JsonConstructorType %d", ctor->type);
   11744                 :           0 :         }
   11745                 :             : 
   11746                 :           0 :         appendStringInfo(buf, "%s(", funcname);
   11747                 :             : 
   11748                 :           0 :         is_json_object = ctor->type == JSCTOR_JSON_OBJECT;
   11749   [ #  #  #  #  :           0 :         foreach(lc, ctor->args)
                   #  # ]
   11750                 :             :         {
   11751                 :           0 :                 curridx = foreach_current_index(lc);
   11752         [ #  # ]:           0 :                 if (curridx > 0)
   11753                 :             :                 {
   11754                 :           0 :                         const char *sep;
   11755                 :             : 
   11756         [ #  # ]:           0 :                         sep = (is_json_object && (curridx % 2) != 0) ? " : " : ", ";
   11757                 :           0 :                         appendStringInfoString(buf, sep);
   11758                 :           0 :                 }
   11759                 :             : 
   11760                 :           0 :                 get_rule_expr((Node *) lfirst(lc), context, true);
   11761                 :           0 :         }
   11762                 :             : 
   11763                 :           0 :         get_json_constructor_options(ctor, buf);
   11764                 :           0 :         appendStringInfoChar(buf, ')');
   11765         [ #  # ]:           0 : }
   11766                 :             : 
   11767                 :             : /*
   11768                 :             :  * Append options, if any, to the JSON constructor being deparsed
   11769                 :             :  */
   11770                 :             : static void
   11771                 :           0 : get_json_constructor_options(JsonConstructorExpr *ctor, StringInfo buf)
   11772                 :             : {
   11773         [ #  # ]:           0 :         if (ctor->absent_on_null)
   11774                 :             :         {
   11775   [ #  #  #  # ]:           0 :                 if (ctor->type == JSCTOR_JSON_OBJECT ||
   11776                 :           0 :                         ctor->type == JSCTOR_JSON_OBJECTAGG)
   11777                 :           0 :                         appendStringInfoString(buf, " ABSENT ON NULL");
   11778                 :           0 :         }
   11779                 :             :         else
   11780                 :             :         {
   11781   [ #  #  #  # ]:           0 :                 if (ctor->type == JSCTOR_JSON_ARRAY ||
   11782                 :           0 :                         ctor->type == JSCTOR_JSON_ARRAYAGG)
   11783                 :           0 :                         appendStringInfoString(buf, " NULL ON NULL");
   11784                 :             :         }
   11785                 :             : 
   11786         [ #  # ]:           0 :         if (ctor->unique)
   11787                 :           0 :                 appendStringInfoString(buf, " WITH UNIQUE KEYS");
   11788                 :             : 
   11789                 :             :         /*
   11790                 :             :          * Append RETURNING clause if needed; JSON() and JSON_SCALAR() don't
   11791                 :             :          * support one.
   11792                 :             :          */
   11793   [ #  #  #  # ]:           0 :         if (ctor->type != JSCTOR_JSON_PARSE && ctor->type != JSCTOR_JSON_SCALAR)
   11794                 :           0 :                 get_json_returning(ctor->returning, buf, true);
   11795                 :           0 : }
   11796                 :             : 
   11797                 :             : /*
   11798                 :             :  * get_json_agg_constructor - Parse back an aggregate JsonConstructorExpr node
   11799                 :             :  */
   11800                 :             : static void
   11801                 :           0 : get_json_agg_constructor(JsonConstructorExpr *ctor, deparse_context *context,
   11802                 :             :                                                  const char *funcname, bool is_json_objectagg)
   11803                 :             : {
   11804                 :           0 :         StringInfoData options;
   11805                 :             : 
   11806                 :           0 :         initStringInfo(&options);
   11807                 :           0 :         get_json_constructor_options(ctor, &options);
   11808                 :             : 
   11809         [ #  # ]:           0 :         if (IsA(ctor->func, Aggref))
   11810                 :           0 :                 get_agg_expr_helper((Aggref *) ctor->func, context,
   11811                 :           0 :                                                         (Aggref *) ctor->func,
   11812                 :           0 :                                                         funcname, options.data, is_json_objectagg);
   11813         [ #  # ]:           0 :         else if (IsA(ctor->func, WindowFunc))
   11814                 :           0 :                 get_windowfunc_expr_helper((WindowFunc *) ctor->func, context,
   11815                 :           0 :                                                                    funcname, options.data,
   11816                 :           0 :                                                                    is_json_objectagg);
   11817                 :             :         else
   11818   [ #  #  #  # ]:           0 :                 elog(ERROR, "invalid JsonConstructorExpr underlying node type: %d",
   11819                 :             :                          nodeTag(ctor->func));
   11820                 :           0 : }
   11821                 :             : 
   11822                 :             : /*
   11823                 :             :  * simple_quote_literal - Format a string as a SQL literal, append to buf
   11824                 :             :  */
   11825                 :             : static void
   11826                 :           0 : simple_quote_literal(StringInfo buf, const char *val)
   11827                 :             : {
   11828                 :           0 :         const char *valptr;
   11829                 :             : 
   11830                 :             :         /*
   11831                 :             :          * We always form the string literal according to standard SQL rules.
   11832                 :             :          */
   11833                 :           0 :         appendStringInfoChar(buf, '\'');
   11834         [ #  # ]:           0 :         for (valptr = val; *valptr; valptr++)
   11835                 :             :         {
   11836                 :           0 :                 char            ch = *valptr;
   11837                 :             : 
   11838   [ #  #  #  #  :           0 :                 if (SQL_STR_DOUBLE(ch, false))
                   #  # ]
   11839                 :           0 :                         appendStringInfoChar(buf, ch);
   11840                 :           0 :                 appendStringInfoChar(buf, ch);
   11841                 :           0 :         }
   11842                 :           0 :         appendStringInfoChar(buf, '\'');
   11843                 :           0 : }
   11844                 :             : 
   11845                 :             : 
   11846                 :             : /* ----------
   11847                 :             :  * get_sublink_expr                     - Parse back a sublink
   11848                 :             :  * ----------
   11849                 :             :  */
   11850                 :             : static void
   11851                 :           0 : get_sublink_expr(SubLink *sublink, deparse_context *context)
   11852                 :             : {
   11853                 :           0 :         StringInfo      buf = context->buf;
   11854                 :           0 :         Query      *query = (Query *) (sublink->subselect);
   11855                 :           0 :         char       *opname = NULL;
   11856                 :           0 :         bool            need_paren;
   11857                 :             : 
   11858         [ #  # ]:           0 :         if (sublink->subLinkType == ARRAY_SUBLINK)
   11859                 :           0 :                 appendStringInfoString(buf, "ARRAY(");
   11860                 :             :         else
   11861                 :           0 :                 appendStringInfoChar(buf, '(');
   11862                 :             : 
   11863                 :             :         /*
   11864                 :             :          * Note that we print the name of only the first operator, when there are
   11865                 :             :          * multiple combining operators.  This is an approximation that could go
   11866                 :             :          * wrong in various scenarios (operators in different schemas, renamed
   11867                 :             :          * operators, etc) but there is not a whole lot we can do about it, since
   11868                 :             :          * the syntax allows only one operator to be shown.
   11869                 :             :          */
   11870         [ #  # ]:           0 :         if (sublink->testexpr)
   11871                 :             :         {
   11872         [ #  # ]:           0 :                 if (IsA(sublink->testexpr, OpExpr))
   11873                 :             :                 {
   11874                 :             :                         /* single combining operator */
   11875                 :           0 :                         OpExpr     *opexpr = (OpExpr *) sublink->testexpr;
   11876                 :             : 
   11877                 :           0 :                         get_rule_expr(linitial(opexpr->args), context, true);
   11878                 :           0 :                         opname = generate_operator_name(opexpr->opno,
   11879                 :           0 :                                                                                         exprType(linitial(opexpr->args)),
   11880                 :           0 :                                                                                         exprType(lsecond(opexpr->args)));
   11881                 :           0 :                 }
   11882         [ #  # ]:           0 :                 else if (IsA(sublink->testexpr, BoolExpr))
   11883                 :             :                 {
   11884                 :             :                         /* multiple combining operators, = or <> cases */
   11885                 :           0 :                         char       *sep;
   11886                 :           0 :                         ListCell   *l;
   11887                 :             : 
   11888                 :           0 :                         appendStringInfoChar(buf, '(');
   11889                 :           0 :                         sep = "";
   11890   [ #  #  #  #  :           0 :                         foreach(l, ((BoolExpr *) sublink->testexpr)->args)
                   #  # ]
   11891                 :             :                         {
   11892                 :           0 :                                 OpExpr     *opexpr = lfirst_node(OpExpr, l);
   11893                 :             : 
   11894                 :           0 :                                 appendStringInfoString(buf, sep);
   11895                 :           0 :                                 get_rule_expr(linitial(opexpr->args), context, true);
   11896         [ #  # ]:           0 :                                 if (!opname)
   11897                 :           0 :                                         opname = generate_operator_name(opexpr->opno,
   11898                 :           0 :                                                                                                         exprType(linitial(opexpr->args)),
   11899                 :           0 :                                                                                                         exprType(lsecond(opexpr->args)));
   11900                 :           0 :                                 sep = ", ";
   11901                 :           0 :                         }
   11902                 :           0 :                         appendStringInfoChar(buf, ')');
   11903                 :           0 :                 }
   11904         [ #  # ]:           0 :                 else if (IsA(sublink->testexpr, RowCompareExpr))
   11905                 :             :                 {
   11906                 :             :                         /* multiple combining operators, < <= > >= cases */
   11907                 :           0 :                         RowCompareExpr *rcexpr = (RowCompareExpr *) sublink->testexpr;
   11908                 :             : 
   11909                 :           0 :                         appendStringInfoChar(buf, '(');
   11910                 :           0 :                         get_rule_expr((Node *) rcexpr->largs, context, true);
   11911                 :           0 :                         opname = generate_operator_name(linitial_oid(rcexpr->opnos),
   11912                 :           0 :                                                                                         exprType(linitial(rcexpr->largs)),
   11913                 :           0 :                                                                                         exprType(linitial(rcexpr->rargs)));
   11914                 :           0 :                         appendStringInfoChar(buf, ')');
   11915                 :           0 :                 }
   11916                 :             :                 else
   11917   [ #  #  #  # ]:           0 :                         elog(ERROR, "unrecognized testexpr type: %d",
   11918                 :             :                                  (int) nodeTag(sublink->testexpr));
   11919                 :           0 :         }
   11920                 :             : 
   11921                 :           0 :         need_paren = true;
   11922                 :             : 
   11923   [ #  #  #  #  :           0 :         switch (sublink->subLinkType)
                #  #  # ]
   11924                 :             :         {
   11925                 :             :                 case EXISTS_SUBLINK:
   11926                 :           0 :                         appendStringInfoString(buf, "EXISTS ");
   11927                 :           0 :                         break;
   11928                 :             : 
   11929                 :             :                 case ANY_SUBLINK:
   11930         [ #  # ]:           0 :                         if (strcmp(opname, "=") == 0) /* Represent = ANY as IN */
   11931                 :           0 :                                 appendStringInfoString(buf, " IN ");
   11932                 :             :                         else
   11933                 :           0 :                                 appendStringInfo(buf, " %s ANY ", opname);
   11934                 :           0 :                         break;
   11935                 :             : 
   11936                 :             :                 case ALL_SUBLINK:
   11937                 :           0 :                         appendStringInfo(buf, " %s ALL ", opname);
   11938                 :           0 :                         break;
   11939                 :             : 
   11940                 :             :                 case ROWCOMPARE_SUBLINK:
   11941                 :           0 :                         appendStringInfo(buf, " %s ", opname);
   11942                 :           0 :                         break;
   11943                 :             : 
   11944                 :             :                 case EXPR_SUBLINK:
   11945                 :             :                 case MULTIEXPR_SUBLINK:
   11946                 :             :                 case ARRAY_SUBLINK:
   11947                 :           0 :                         need_paren = false;
   11948                 :           0 :                         break;
   11949                 :             : 
   11950                 :           0 :                 case CTE_SUBLINK:               /* shouldn't occur in a SubLink */
   11951                 :             :                 default:
   11952   [ #  #  #  # ]:           0 :                         elog(ERROR, "unrecognized sublink type: %d",
   11953                 :             :                                  (int) sublink->subLinkType);
   11954                 :           0 :                         break;
   11955                 :             :         }
   11956                 :             : 
   11957         [ #  # ]:           0 :         if (need_paren)
   11958                 :           0 :                 appendStringInfoChar(buf, '(');
   11959                 :             : 
   11960                 :           0 :         get_query_def(query, buf, context->namespaces, NULL, false,
   11961                 :           0 :                                   context->prettyFlags, context->wrapColumn,
   11962                 :           0 :                                   context->indentLevel);
   11963                 :             : 
   11964         [ #  # ]:           0 :         if (need_paren)
   11965                 :           0 :                 appendStringInfoString(buf, "))");
   11966                 :             :         else
   11967                 :           0 :                 appendStringInfoChar(buf, ')');
   11968                 :           0 : }
   11969                 :             : 
   11970                 :             : 
   11971                 :             : /* ----------
   11972                 :             :  * get_xmltable                 - Parse back a XMLTABLE function
   11973                 :             :  * ----------
   11974                 :             :  */
   11975                 :             : static void
   11976                 :           0 : get_xmltable(TableFunc *tf, deparse_context *context, bool showimplicit)
   11977                 :             : {
   11978                 :           0 :         StringInfo      buf = context->buf;
   11979                 :             : 
   11980                 :           0 :         appendStringInfoString(buf, "XMLTABLE(");
   11981                 :             : 
   11982         [ #  # ]:           0 :         if (tf->ns_uris != NIL)
   11983                 :             :         {
   11984                 :           0 :                 ListCell   *lc1,
   11985                 :             :                                    *lc2;
   11986                 :           0 :                 bool            first = true;
   11987                 :             : 
   11988                 :           0 :                 appendStringInfoString(buf, "XMLNAMESPACES (");
   11989   [ #  #  #  #  :           0 :                 forboth(lc1, tf->ns_uris, lc2, tf->ns_names)
          #  #  #  #  #  
                #  #  # ]
   11990                 :             :                 {
   11991                 :           0 :                         Node       *expr = (Node *) lfirst(lc1);
   11992                 :           0 :                         String     *ns_node = lfirst_node(String, lc2);
   11993                 :             : 
   11994         [ #  # ]:           0 :                         if (!first)
   11995                 :           0 :                                 appendStringInfoString(buf, ", ");
   11996                 :             :                         else
   11997                 :           0 :                                 first = false;
   11998                 :             : 
   11999         [ #  # ]:           0 :                         if (ns_node != NULL)
   12000                 :             :                         {
   12001                 :           0 :                                 get_rule_expr(expr, context, showimplicit);
   12002                 :           0 :                                 appendStringInfo(buf, " AS %s",
   12003                 :           0 :                                                                  quote_identifier(strVal(ns_node)));
   12004                 :           0 :                         }
   12005                 :             :                         else
   12006                 :             :                         {
   12007                 :           0 :                                 appendStringInfoString(buf, "DEFAULT ");
   12008                 :           0 :                                 get_rule_expr(expr, context, showimplicit);
   12009                 :             :                         }
   12010                 :           0 :                 }
   12011                 :           0 :                 appendStringInfoString(buf, "), ");
   12012                 :           0 :         }
   12013                 :             : 
   12014                 :           0 :         appendStringInfoChar(buf, '(');
   12015                 :           0 :         get_rule_expr((Node *) tf->rowexpr, context, showimplicit);
   12016                 :           0 :         appendStringInfoString(buf, ") PASSING (");
   12017                 :           0 :         get_rule_expr((Node *) tf->docexpr, context, showimplicit);
   12018                 :           0 :         appendStringInfoChar(buf, ')');
   12019                 :             : 
   12020         [ #  # ]:           0 :         if (tf->colexprs != NIL)
   12021                 :             :         {
   12022                 :           0 :                 ListCell   *l1;
   12023                 :           0 :                 ListCell   *l2;
   12024                 :           0 :                 ListCell   *l3;
   12025                 :           0 :                 ListCell   *l4;
   12026                 :           0 :                 ListCell   *l5;
   12027                 :           0 :                 int                     colnum = 0;
   12028                 :             : 
   12029                 :           0 :                 appendStringInfoString(buf, " COLUMNS ");
   12030   [ #  #  #  #  :           0 :                 forfive(l1, tf->colnames, l2, tf->coltypes, l3, tf->coltypmods,
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                      # ]
   12031                 :             :                                 l4, tf->colexprs, l5, tf->coldefexprs)
   12032                 :             :                 {
   12033                 :           0 :                         char       *colname = strVal(lfirst(l1));
   12034                 :           0 :                         Oid                     typid = lfirst_oid(l2);
   12035                 :           0 :                         int32           typmod = lfirst_int(l3);
   12036                 :           0 :                         Node       *colexpr = (Node *) lfirst(l4);
   12037                 :           0 :                         Node       *coldefexpr = (Node *) lfirst(l5);
   12038                 :           0 :                         bool            ordinality = (tf->ordinalitycol == colnum);
   12039                 :           0 :                         bool            notnull = bms_is_member(colnum, tf->notnulls);
   12040                 :             : 
   12041         [ #  # ]:           0 :                         if (colnum > 0)
   12042                 :           0 :                                 appendStringInfoString(buf, ", ");
   12043                 :           0 :                         colnum++;
   12044                 :             : 
   12045                 :           0 :                         appendStringInfo(buf, "%s %s", quote_identifier(colname),
   12046         [ #  # ]:           0 :                                                          ordinality ? "FOR ORDINALITY" :
   12047                 :           0 :                                                          format_type_with_typemod(typid, typmod));
   12048         [ #  # ]:           0 :                         if (ordinality)
   12049                 :           0 :                                 continue;
   12050                 :             : 
   12051         [ #  # ]:           0 :                         if (coldefexpr != NULL)
   12052                 :             :                         {
   12053                 :           0 :                                 appendStringInfoString(buf, " DEFAULT (");
   12054                 :           0 :                                 get_rule_expr((Node *) coldefexpr, context, showimplicit);
   12055                 :           0 :                                 appendStringInfoChar(buf, ')');
   12056                 :           0 :                         }
   12057         [ #  # ]:           0 :                         if (colexpr != NULL)
   12058                 :             :                         {
   12059                 :           0 :                                 appendStringInfoString(buf, " PATH (");
   12060                 :           0 :                                 get_rule_expr((Node *) colexpr, context, showimplicit);
   12061                 :           0 :                                 appendStringInfoChar(buf, ')');
   12062                 :           0 :                         }
   12063         [ #  # ]:           0 :                         if (notnull)
   12064                 :           0 :                                 appendStringInfoString(buf, " NOT NULL");
   12065      [ #  #  # ]:           0 :                 }
   12066                 :           0 :         }
   12067                 :             : 
   12068                 :           0 :         appendStringInfoChar(buf, ')');
   12069                 :           0 : }
   12070                 :             : 
   12071                 :             : /*
   12072                 :             :  * get_json_table_nested_columns - Parse back nested JSON_TABLE columns
   12073                 :             :  */
   12074                 :             : static void
   12075                 :           0 : get_json_table_nested_columns(TableFunc *tf, JsonTablePlan *plan,
   12076                 :             :                                                           deparse_context *context, bool showimplicit,
   12077                 :             :                                                           bool needcomma)
   12078                 :             : {
   12079         [ #  # ]:           0 :         if (IsA(plan, JsonTablePathScan))
   12080                 :             :         {
   12081                 :           0 :                 JsonTablePathScan *scan = castNode(JsonTablePathScan, plan);
   12082                 :             : 
   12083         [ #  # ]:           0 :                 if (needcomma)
   12084                 :           0 :                         appendStringInfoChar(context->buf, ',');
   12085                 :             : 
   12086                 :           0 :                 appendStringInfoChar(context->buf, ' ');
   12087                 :           0 :                 appendContextKeyword(context, "NESTED PATH ", 0, 0, 0);
   12088                 :           0 :                 get_const_expr(scan->path->value, context, -1);
   12089                 :           0 :                 appendStringInfo(context->buf, " AS %s", quote_identifier(scan->path->name));
   12090                 :           0 :                 get_json_table_columns(tf, scan, context, showimplicit);
   12091                 :           0 :         }
   12092         [ #  # ]:           0 :         else if (IsA(plan, JsonTableSiblingJoin))
   12093                 :             :         {
   12094                 :           0 :                 JsonTableSiblingJoin *join = (JsonTableSiblingJoin *) plan;
   12095                 :             : 
   12096                 :           0 :                 get_json_table_nested_columns(tf, join->lplan, context, showimplicit,
   12097                 :           0 :                                                                           needcomma);
   12098                 :           0 :                 get_json_table_nested_columns(tf, join->rplan, context, showimplicit,
   12099                 :             :                                                                           true);
   12100                 :           0 :         }
   12101                 :           0 : }
   12102                 :             : 
   12103                 :             : /*
   12104                 :             :  * get_json_table_columns - Parse back JSON_TABLE columns
   12105                 :             :  */
   12106                 :             : static void
   12107                 :           0 : get_json_table_columns(TableFunc *tf, JsonTablePathScan *scan,
   12108                 :             :                                            deparse_context *context,
   12109                 :             :                                            bool showimplicit)
   12110                 :             : {
   12111                 :           0 :         StringInfo      buf = context->buf;
   12112                 :           0 :         ListCell   *lc_colname;
   12113                 :           0 :         ListCell   *lc_coltype;
   12114                 :           0 :         ListCell   *lc_coltypmod;
   12115                 :           0 :         ListCell   *lc_colvalexpr;
   12116                 :           0 :         int                     colnum = 0;
   12117                 :             : 
   12118                 :           0 :         appendStringInfoChar(buf, ' ');
   12119                 :           0 :         appendContextKeyword(context, "COLUMNS (", 0, 0, 0);
   12120                 :             : 
   12121         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
   12122                 :           0 :                 context->indentLevel += PRETTYINDENT_VAR;
   12123                 :             : 
   12124   [ #  #  #  #  :           0 :         forfour(lc_colname, tf->colnames,
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  #  
                      # ]
   12125                 :             :                         lc_coltype, tf->coltypes,
   12126                 :             :                         lc_coltypmod, tf->coltypmods,
   12127                 :             :                         lc_colvalexpr, tf->colvalexprs)
   12128                 :             :         {
   12129                 :           0 :                 char       *colname = strVal(lfirst(lc_colname));
   12130                 :           0 :                 JsonExpr   *colexpr;
   12131                 :           0 :                 Oid                     typid;
   12132                 :           0 :                 int32           typmod;
   12133                 :           0 :                 bool            ordinality;
   12134                 :           0 :                 JsonBehaviorType default_behavior;
   12135                 :             : 
   12136                 :           0 :                 typid = lfirst_oid(lc_coltype);
   12137                 :           0 :                 typmod = lfirst_int(lc_coltypmod);
   12138                 :           0 :                 colexpr = castNode(JsonExpr, lfirst(lc_colvalexpr));
   12139                 :             : 
   12140                 :             :                 /* Skip columns that don't belong to this scan. */
   12141   [ #  #  #  # ]:           0 :                 if (scan->colMin < 0 || colnum < scan->colMin)
   12142                 :             :                 {
   12143                 :           0 :                         colnum++;
   12144                 :           0 :                         continue;
   12145                 :             :                 }
   12146         [ #  # ]:           0 :                 if (colnum > scan->colMax)
   12147                 :           0 :                         break;
   12148                 :             : 
   12149         [ #  # ]:           0 :                 if (colnum > scan->colMin)
   12150                 :           0 :                         appendStringInfoString(buf, ", ");
   12151                 :             : 
   12152                 :           0 :                 colnum++;
   12153                 :             : 
   12154                 :           0 :                 ordinality = !colexpr;
   12155                 :             : 
   12156                 :           0 :                 appendContextKeyword(context, "", 0, 0, 0);
   12157                 :             : 
   12158                 :           0 :                 appendStringInfo(buf, "%s %s", quote_identifier(colname),
   12159         [ #  # ]:           0 :                                                  ordinality ? "FOR ORDINALITY" :
   12160                 :           0 :                                                  format_type_with_typemod(typid, typmod));
   12161         [ #  # ]:           0 :                 if (ordinality)
   12162                 :           0 :                         continue;
   12163                 :             : 
   12164                 :             :                 /*
   12165                 :             :                  * Set default_behavior to guide get_json_expr_options() on whether to
   12166                 :             :                  * emit the ON ERROR / EMPTY clauses.
   12167                 :             :                  */
   12168         [ #  # ]:           0 :                 if (colexpr->op == JSON_EXISTS_OP)
   12169                 :             :                 {
   12170                 :           0 :                         appendStringInfoString(buf, " EXISTS");
   12171                 :           0 :                         default_behavior = JSON_BEHAVIOR_FALSE;
   12172                 :           0 :                 }
   12173                 :             :                 else
   12174                 :             :                 {
   12175         [ #  # ]:           0 :                         if (colexpr->op == JSON_QUERY_OP)
   12176                 :             :                         {
   12177                 :           0 :                                 char            typcategory;
   12178                 :           0 :                                 bool            typispreferred;
   12179                 :             : 
   12180                 :           0 :                                 get_type_category_preferred(typid, &typcategory, &typispreferred);
   12181                 :             : 
   12182         [ #  # ]:           0 :                                 if (typcategory == TYPCATEGORY_STRING)
   12183                 :           0 :                                         appendStringInfoString(buf,
   12184                 :           0 :                                                                                    colexpr->format->format_type == JS_FORMAT_JSONB ?
   12185                 :             :                                                                                    " FORMAT JSONB" : " FORMAT JSON");
   12186                 :           0 :                         }
   12187                 :             : 
   12188                 :           0 :                         default_behavior = JSON_BEHAVIOR_NULL;
   12189                 :             :                 }
   12190                 :             : 
   12191                 :           0 :                 appendStringInfoString(buf, " PATH ");
   12192                 :             : 
   12193                 :           0 :                 get_json_path_spec(colexpr->path_spec, context, showimplicit);
   12194                 :             : 
   12195                 :           0 :                 get_json_expr_options(colexpr, context, default_behavior);
   12196      [ #  #  # ]:           0 :         }
   12197                 :             : 
   12198         [ #  # ]:           0 :         if (scan->child)
   12199                 :           0 :                 get_json_table_nested_columns(tf, scan->child, context, showimplicit,
   12200                 :           0 :                                                                           scan->colMin >= 0);
   12201                 :             : 
   12202         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
   12203                 :           0 :                 context->indentLevel -= PRETTYINDENT_VAR;
   12204                 :             : 
   12205                 :           0 :         appendContextKeyword(context, ")", 0, 0, 0);
   12206                 :           0 : }
   12207                 :             : 
   12208                 :             : /* ----------
   12209                 :             :  * get_json_table                       - Parse back a JSON_TABLE function
   12210                 :             :  * ----------
   12211                 :             :  */
   12212                 :             : static void
   12213                 :           0 : get_json_table(TableFunc *tf, deparse_context *context, bool showimplicit)
   12214                 :             : {
   12215                 :           0 :         StringInfo      buf = context->buf;
   12216                 :           0 :         JsonExpr   *jexpr = castNode(JsonExpr, tf->docexpr);
   12217                 :           0 :         JsonTablePathScan *root = castNode(JsonTablePathScan, tf->plan);
   12218                 :             : 
   12219                 :           0 :         appendStringInfoString(buf, "JSON_TABLE(");
   12220                 :             : 
   12221         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
   12222                 :           0 :                 context->indentLevel += PRETTYINDENT_VAR;
   12223                 :             : 
   12224                 :           0 :         appendContextKeyword(context, "", 0, 0, 0);
   12225                 :             : 
   12226                 :           0 :         get_rule_expr(jexpr->formatted_expr, context, showimplicit);
   12227                 :             : 
   12228                 :           0 :         appendStringInfoString(buf, ", ");
   12229                 :             : 
   12230                 :           0 :         get_const_expr(root->path->value, context, -1);
   12231                 :             : 
   12232                 :           0 :         appendStringInfo(buf, " AS %s", quote_identifier(root->path->name));
   12233                 :             : 
   12234         [ #  # ]:           0 :         if (jexpr->passing_values)
   12235                 :             :         {
   12236                 :           0 :                 ListCell   *lc1,
   12237                 :             :                                    *lc2;
   12238                 :           0 :                 bool            needcomma = false;
   12239                 :             : 
   12240                 :           0 :                 appendStringInfoChar(buf, ' ');
   12241                 :           0 :                 appendContextKeyword(context, "PASSING ", 0, 0, 0);
   12242                 :             : 
   12243         [ #  # ]:           0 :                 if (PRETTY_INDENT(context))
   12244                 :           0 :                         context->indentLevel += PRETTYINDENT_VAR;
   12245                 :             : 
   12246   [ #  #  #  #  :           0 :                 forboth(lc1, jexpr->passing_names,
          #  #  #  #  #  
                #  #  # ]
   12247                 :             :                                 lc2, jexpr->passing_values)
   12248                 :             :                 {
   12249         [ #  # ]:           0 :                         if (needcomma)
   12250                 :           0 :                                 appendStringInfoString(buf, ", ");
   12251                 :           0 :                         needcomma = true;
   12252                 :             : 
   12253                 :           0 :                         appendContextKeyword(context, "", 0, 0, 0);
   12254                 :             : 
   12255                 :           0 :                         get_rule_expr((Node *) lfirst(lc2), context, false);
   12256                 :           0 :                         appendStringInfo(buf, " AS %s",
   12257                 :           0 :                                                          quote_identifier((lfirst_node(String, lc1))->sval)
   12258                 :             :                                 );
   12259                 :           0 :                 }
   12260                 :             : 
   12261         [ #  # ]:           0 :                 if (PRETTY_INDENT(context))
   12262                 :           0 :                         context->indentLevel -= PRETTYINDENT_VAR;
   12263                 :           0 :         }
   12264                 :             : 
   12265                 :           0 :         get_json_table_columns(tf, castNode(JsonTablePathScan, tf->plan), context,
   12266                 :           0 :                                                    showimplicit);
   12267                 :             : 
   12268         [ #  # ]:           0 :         if (jexpr->on_error->btype != JSON_BEHAVIOR_EMPTY_ARRAY)
   12269                 :           0 :                 get_json_behavior(jexpr->on_error, context, "ERROR");
   12270                 :             : 
   12271         [ #  # ]:           0 :         if (PRETTY_INDENT(context))
   12272                 :           0 :                 context->indentLevel -= PRETTYINDENT_VAR;
   12273                 :             : 
   12274                 :           0 :         appendContextKeyword(context, ")", 0, 0, 0);
   12275                 :           0 : }
   12276                 :             : 
   12277                 :             : /* ----------
   12278                 :             :  * get_tablefunc                        - Parse back a table function
   12279                 :             :  * ----------
   12280                 :             :  */
   12281                 :             : static void
   12282                 :           0 : get_tablefunc(TableFunc *tf, deparse_context *context, bool showimplicit)
   12283                 :             : {
   12284                 :             :         /* XMLTABLE and JSON_TABLE are the only existing implementations.  */
   12285                 :             : 
   12286         [ #  # ]:           0 :         if (tf->functype == TFT_XMLTABLE)
   12287                 :           0 :                 get_xmltable(tf, context, showimplicit);
   12288         [ #  # ]:           0 :         else if (tf->functype == TFT_JSON_TABLE)
   12289                 :           0 :                 get_json_table(tf, context, showimplicit);
   12290                 :           0 : }
   12291                 :             : 
   12292                 :             : /* ----------
   12293                 :             :  * get_from_clause                      - Parse back a FROM clause
   12294                 :             :  *
   12295                 :             :  * "prefix" is the keyword that denotes the start of the list of FROM
   12296                 :             :  * elements. It is FROM when used to parse back SELECT and UPDATE, but
   12297                 :             :  * is USING when parsing back DELETE.
   12298                 :             :  * ----------
   12299                 :             :  */
   12300                 :             : static void
   12301                 :           0 : get_from_clause(Query *query, const char *prefix, deparse_context *context)
   12302                 :             : {
   12303                 :           0 :         StringInfo      buf = context->buf;
   12304                 :           0 :         bool            first = true;
   12305                 :           0 :         ListCell   *l;
   12306                 :             : 
   12307                 :             :         /*
   12308                 :             :          * We use the query's jointree as a guide to what to print.  However, we
   12309                 :             :          * must ignore auto-added RTEs that are marked not inFromCl. (These can
   12310                 :             :          * only appear at the top level of the jointree, so it's sufficient to
   12311                 :             :          * check here.)  This check also ensures we ignore the rule pseudo-RTEs
   12312                 :             :          * for NEW and OLD.
   12313                 :             :          */
   12314   [ #  #  #  #  :           0 :         foreach(l, query->jointree->fromlist)
                   #  # ]
   12315                 :             :         {
   12316                 :           0 :                 Node       *jtnode = (Node *) lfirst(l);
   12317                 :             : 
   12318         [ #  # ]:           0 :                 if (IsA(jtnode, RangeTblRef))
   12319                 :             :                 {
   12320                 :           0 :                         int                     varno = ((RangeTblRef *) jtnode)->rtindex;
   12321                 :           0 :                         RangeTblEntry *rte = rt_fetch(varno, query->rtable);
   12322                 :             : 
   12323         [ #  # ]:           0 :                         if (!rte->inFromCl)
   12324                 :           0 :                                 continue;
   12325         [ #  # ]:           0 :                 }
   12326                 :             : 
   12327         [ #  # ]:           0 :                 if (first)
   12328                 :             :                 {
   12329                 :           0 :                         appendContextKeyword(context, prefix,
   12330                 :             :                                                                  -PRETTYINDENT_STD, PRETTYINDENT_STD, 2);
   12331                 :           0 :                         first = false;
   12332                 :             : 
   12333                 :           0 :                         get_from_clause_item(jtnode, query, context);
   12334                 :           0 :                 }
   12335                 :             :                 else
   12336                 :             :                 {
   12337                 :           0 :                         StringInfoData itembuf;
   12338                 :             : 
   12339                 :           0 :                         appendStringInfoString(buf, ", ");
   12340                 :             : 
   12341                 :             :                         /*
   12342                 :             :                          * Put the new FROM item's text into itembuf so we can decide
   12343                 :             :                          * after we've got it whether or not it needs to go on a new line.
   12344                 :             :                          */
   12345                 :           0 :                         initStringInfo(&itembuf);
   12346                 :           0 :                         context->buf = &itembuf;
   12347                 :             : 
   12348                 :           0 :                         get_from_clause_item(jtnode, query, context);
   12349                 :             : 
   12350                 :             :                         /* Restore context's output buffer */
   12351                 :           0 :                         context->buf = buf;
   12352                 :             : 
   12353                 :             :                         /* Consider line-wrapping if enabled */
   12354   [ #  #  #  # ]:           0 :                         if (PRETTY_INDENT(context) && context->wrapColumn >= 0)
   12355                 :             :                         {
   12356                 :             :                                 /* Does the new item start with a new line? */
   12357   [ #  #  #  # ]:           0 :                                 if (itembuf.len > 0 && itembuf.data[0] == '\n')
   12358                 :             :                                 {
   12359                 :             :                                         /* If so, we shouldn't add anything */
   12360                 :             :                                         /* instead, remove any trailing spaces currently in buf */
   12361                 :           0 :                                         removeStringInfoSpaces(buf);
   12362                 :           0 :                                 }
   12363                 :             :                                 else
   12364                 :             :                                 {
   12365                 :           0 :                                         char       *trailing_nl;
   12366                 :             : 
   12367                 :             :                                         /* Locate the start of the current line in the buffer */
   12368                 :           0 :                                         trailing_nl = strrchr(buf->data, '\n');
   12369         [ #  # ]:           0 :                                         if (trailing_nl == NULL)
   12370                 :           0 :                                                 trailing_nl = buf->data;
   12371                 :             :                                         else
   12372                 :           0 :                                                 trailing_nl++;
   12373                 :             : 
   12374                 :             :                                         /*
   12375                 :             :                                          * Add a newline, plus some indentation, if the new item
   12376                 :             :                                          * would cause an overflow.
   12377                 :             :                                          */
   12378         [ #  # ]:           0 :                                         if (strlen(trailing_nl) + itembuf.len > context->wrapColumn)
   12379                 :           0 :                                                 appendContextKeyword(context, "", -PRETTYINDENT_STD,
   12380                 :             :                                                                                          PRETTYINDENT_STD,
   12381                 :             :                                                                                          PRETTYINDENT_VAR);
   12382                 :           0 :                                 }
   12383                 :           0 :                         }
   12384                 :             : 
   12385                 :             :                         /* Add the new item */
   12386                 :           0 :                         appendBinaryStringInfo(buf, itembuf.data, itembuf.len);
   12387                 :             : 
   12388                 :             :                         /* clean up */
   12389                 :           0 :                         pfree(itembuf.data);
   12390                 :           0 :                 }
   12391      [ #  #  # ]:           0 :         }
   12392                 :           0 : }
   12393                 :             : 
   12394                 :             : static void
   12395                 :           0 : get_from_clause_item(Node *jtnode, Query *query, deparse_context *context)
   12396                 :             : {
   12397                 :           0 :         StringInfo      buf = context->buf;
   12398                 :           0 :         deparse_namespace *dpns = (deparse_namespace *) linitial(context->namespaces);
   12399                 :             : 
   12400         [ #  # ]:           0 :         if (IsA(jtnode, RangeTblRef))
   12401                 :             :         {
   12402                 :           0 :                 int                     varno = ((RangeTblRef *) jtnode)->rtindex;
   12403                 :           0 :                 RangeTblEntry *rte = rt_fetch(varno, query->rtable);
   12404                 :           0 :                 deparse_columns *colinfo = deparse_columns_fetch(varno, dpns);
   12405                 :           0 :                 RangeTblFunction *rtfunc1 = NULL;
   12406                 :             : 
   12407         [ #  # ]:           0 :                 if (rte->lateral)
   12408                 :           0 :                         appendStringInfoString(buf, "LATERAL ");
   12409                 :             : 
   12410                 :             :                 /* Print the FROM item proper */
   12411   [ #  #  #  #  :           0 :                 switch (rte->rtekind)
                #  #  # ]
   12412                 :             :                 {
   12413                 :             :                         case RTE_RELATION:
   12414                 :             :                                 /* Normal relation RTE */
   12415                 :           0 :                                 appendStringInfo(buf, "%s%s",
   12416                 :           0 :                                                                  only_marker(rte),
   12417                 :           0 :                                                                  generate_relation_name(rte->relid,
   12418                 :           0 :                                                                                                                 context->namespaces));
   12419                 :           0 :                                 break;
   12420                 :             :                         case RTE_SUBQUERY:
   12421                 :             :                                 /* Subquery RTE */
   12422                 :           0 :                                 appendStringInfoChar(buf, '(');
   12423                 :           0 :                                 get_query_def(rte->subquery, buf, context->namespaces, NULL,
   12424                 :             :                                                           true,
   12425                 :           0 :                                                           context->prettyFlags, context->wrapColumn,
   12426                 :           0 :                                                           context->indentLevel);
   12427                 :           0 :                                 appendStringInfoChar(buf, ')');
   12428                 :           0 :                                 break;
   12429                 :             :                         case RTE_FUNCTION:
   12430                 :             :                                 /* Function RTE */
   12431                 :           0 :                                 rtfunc1 = (RangeTblFunction *) linitial(rte->functions);
   12432                 :             : 
   12433                 :             :                                 /*
   12434                 :             :                                  * Omit ROWS FROM() syntax for just one function, unless it
   12435                 :             :                                  * has both a coldeflist and WITH ORDINALITY. If it has both,
   12436                 :             :                                  * we must use ROWS FROM() syntax to avoid ambiguity about
   12437                 :             :                                  * whether the coldeflist includes the ordinality column.
   12438                 :             :                                  */
   12439   [ #  #  #  # ]:           0 :                                 if (list_length(rte->functions) == 1 &&
   12440         [ #  # ]:           0 :                                         (rtfunc1->funccolnames == NIL || !rte->funcordinality))
   12441                 :             :                                 {
   12442                 :           0 :                                         get_rule_expr_funccall(rtfunc1->funcexpr, context, true);
   12443                 :             :                                         /* we'll print the coldeflist below, if it has one */
   12444                 :           0 :                                 }
   12445                 :             :                                 else
   12446                 :             :                                 {
   12447                 :           0 :                                         bool            all_unnest;
   12448                 :           0 :                                         ListCell   *lc;
   12449                 :             : 
   12450                 :             :                                         /*
   12451                 :             :                                          * If all the function calls in the list are to unnest,
   12452                 :             :                                          * and none need a coldeflist, then collapse the list back
   12453                 :             :                                          * down to UNNEST(args).  (If we had more than one
   12454                 :             :                                          * built-in unnest function, this would get more
   12455                 :             :                                          * difficult.)
   12456                 :             :                                          *
   12457                 :             :                                          * XXX This is pretty ugly, since it makes not-terribly-
   12458                 :             :                                          * future-proof assumptions about what the parser would do
   12459                 :             :                                          * with the output; but the alternative is to emit our
   12460                 :             :                                          * nonstandard ROWS FROM() notation for what might have
   12461                 :             :                                          * been a perfectly spec-compliant multi-argument
   12462                 :             :                                          * UNNEST().
   12463                 :             :                                          */
   12464                 :           0 :                                         all_unnest = true;
   12465   [ #  #  #  #  :           0 :                                         foreach(lc, rte->functions)
                   #  # ]
   12466                 :             :                                         {
   12467                 :           0 :                                                 RangeTblFunction *rtfunc = (RangeTblFunction *) lfirst(lc);
   12468                 :             : 
   12469         [ #  # ]:           0 :                                                 if (!IsA(rtfunc->funcexpr, FuncExpr) ||
   12470   [ #  #  #  # ]:           0 :                                                         ((FuncExpr *) rtfunc->funcexpr)->funcid != F_UNNEST_ANYARRAY ||
   12471                 :           0 :                                                         rtfunc->funccolnames != NIL)
   12472                 :             :                                                 {
   12473                 :           0 :                                                         all_unnest = false;
   12474                 :           0 :                                                         break;
   12475                 :             :                                                 }
   12476         [ #  # ]:           0 :                                         }
   12477                 :             : 
   12478         [ #  # ]:           0 :                                         if (all_unnest)
   12479                 :             :                                         {
   12480                 :           0 :                                                 List       *allargs = NIL;
   12481                 :             : 
   12482   [ #  #  #  #  :           0 :                                                 foreach(lc, rte->functions)
                   #  # ]
   12483                 :             :                                                 {
   12484                 :           0 :                                                         RangeTblFunction *rtfunc = (RangeTblFunction *) lfirst(lc);
   12485                 :           0 :                                                         List       *args = ((FuncExpr *) rtfunc->funcexpr)->args;
   12486                 :             : 
   12487                 :           0 :                                                         allargs = list_concat(allargs, args);
   12488                 :           0 :                                                 }
   12489                 :             : 
   12490                 :           0 :                                                 appendStringInfoString(buf, "UNNEST(");
   12491                 :           0 :                                                 get_rule_expr((Node *) allargs, context, true);
   12492                 :           0 :                                                 appendStringInfoChar(buf, ')');
   12493                 :           0 :                                         }
   12494                 :             :                                         else
   12495                 :             :                                         {
   12496                 :           0 :                                                 int                     funcno = 0;
   12497                 :             : 
   12498                 :           0 :                                                 appendStringInfoString(buf, "ROWS FROM(");
   12499   [ #  #  #  #  :           0 :                                                 foreach(lc, rte->functions)
                   #  # ]
   12500                 :             :                                                 {
   12501                 :           0 :                                                         RangeTblFunction *rtfunc = (RangeTblFunction *) lfirst(lc);
   12502                 :             : 
   12503         [ #  # ]:           0 :                                                         if (funcno > 0)
   12504                 :           0 :                                                                 appendStringInfoString(buf, ", ");
   12505                 :           0 :                                                         get_rule_expr_funccall(rtfunc->funcexpr, context, true);
   12506         [ #  # ]:           0 :                                                         if (rtfunc->funccolnames != NIL)
   12507                 :             :                                                         {
   12508                 :             :                                                                 /* Reconstruct the column definition list */
   12509                 :           0 :                                                                 appendStringInfoString(buf, " AS ");
   12510                 :           0 :                                                                 get_from_clause_coldeflist(rtfunc,
   12511                 :             :                                                                                                                    NULL,
   12512                 :           0 :                                                                                                                    context);
   12513                 :           0 :                                                         }
   12514                 :           0 :                                                         funcno++;
   12515                 :           0 :                                                 }
   12516                 :           0 :                                                 appendStringInfoChar(buf, ')');
   12517                 :           0 :                                         }
   12518                 :             :                                         /* prevent printing duplicate coldeflist below */
   12519                 :           0 :                                         rtfunc1 = NULL;
   12520                 :           0 :                                 }
   12521         [ #  # ]:           0 :                                 if (rte->funcordinality)
   12522                 :           0 :                                         appendStringInfoString(buf, " WITH ORDINALITY");
   12523                 :           0 :                                 break;
   12524                 :             :                         case RTE_TABLEFUNC:
   12525                 :           0 :                                 get_tablefunc(rte->tablefunc, context, true);
   12526                 :           0 :                                 break;
   12527                 :             :                         case RTE_VALUES:
   12528                 :             :                                 /* Values list RTE */
   12529                 :           0 :                                 appendStringInfoChar(buf, '(');
   12530                 :           0 :                                 get_values_def(rte->values_lists, context);
   12531                 :           0 :                                 appendStringInfoChar(buf, ')');
   12532                 :           0 :                                 break;
   12533                 :             :                         case RTE_CTE:
   12534                 :           0 :                                 appendStringInfoString(buf, quote_identifier(rte->ctename));
   12535                 :           0 :                                 break;
   12536                 :             :                         default:
   12537   [ #  #  #  # ]:           0 :                                 elog(ERROR, "unrecognized RTE kind: %d", (int) rte->rtekind);
   12538                 :           0 :                                 break;
   12539                 :             :                 }
   12540                 :             : 
   12541                 :             :                 /* Print the relation alias, if needed */
   12542                 :           0 :                 get_rte_alias(rte, varno, false, context);
   12543                 :             : 
   12544                 :             :                 /* Print the column definitions or aliases, if needed */
   12545   [ #  #  #  # ]:           0 :                 if (rtfunc1 && rtfunc1->funccolnames != NIL)
   12546                 :             :                 {
   12547                 :             :                         /* Reconstruct the columndef list, which is also the aliases */
   12548                 :           0 :                         get_from_clause_coldeflist(rtfunc1, colinfo, context);
   12549                 :           0 :                 }
   12550                 :             :                 else
   12551                 :             :                 {
   12552                 :             :                         /* Else print column aliases as needed */
   12553                 :           0 :                         get_column_alias_list(colinfo, context);
   12554                 :             :                 }
   12555                 :             : 
   12556                 :             :                 /* Tablesample clause must go after any alias */
   12557   [ #  #  #  # ]:           0 :                 if (rte->rtekind == RTE_RELATION && rte->tablesample)
   12558                 :           0 :                         get_tablesample_def(rte->tablesample, context);
   12559                 :           0 :         }
   12560         [ #  # ]:           0 :         else if (IsA(jtnode, JoinExpr))
   12561                 :             :         {
   12562                 :           0 :                 JoinExpr   *j = (JoinExpr *) jtnode;
   12563                 :           0 :                 deparse_columns *colinfo = deparse_columns_fetch(j->rtindex, dpns);
   12564                 :           0 :                 bool            need_paren_on_right;
   12565                 :             : 
   12566         [ #  # ]:           0 :                 need_paren_on_right = PRETTY_PAREN(context) &&
   12567         [ #  # ]:           0 :                         !IsA(j->rarg, RangeTblRef) &&
   12568         [ #  # ]:           0 :                         !(IsA(j->rarg, JoinExpr) && ((JoinExpr *) j->rarg)->alias != NULL);
   12569                 :             : 
   12570   [ #  #  #  # ]:           0 :                 if (!PRETTY_PAREN(context) || j->alias != NULL)
   12571                 :           0 :                         appendStringInfoChar(buf, '(');
   12572                 :             : 
   12573                 :           0 :                 get_from_clause_item(j->larg, query, context);
   12574                 :             : 
   12575   [ #  #  #  #  :           0 :                 switch (j->jointype)
                      # ]
   12576                 :             :                 {
   12577                 :             :                         case JOIN_INNER:
   12578         [ #  # ]:           0 :                                 if (j->quals)
   12579                 :           0 :                                         appendContextKeyword(context, " JOIN ",
   12580                 :             :                                                                                  -PRETTYINDENT_STD,
   12581                 :             :                                                                                  PRETTYINDENT_STD,
   12582                 :             :                                                                                  PRETTYINDENT_JOIN);
   12583                 :             :                                 else
   12584                 :           0 :                                         appendContextKeyword(context, " CROSS JOIN ",
   12585                 :             :                                                                                  -PRETTYINDENT_STD,
   12586                 :             :                                                                                  PRETTYINDENT_STD,
   12587                 :             :                                                                                  PRETTYINDENT_JOIN);
   12588                 :           0 :                                 break;
   12589                 :             :                         case JOIN_LEFT:
   12590                 :           0 :                                 appendContextKeyword(context, " LEFT JOIN ",
   12591                 :             :                                                                          -PRETTYINDENT_STD,
   12592                 :             :                                                                          PRETTYINDENT_STD,
   12593                 :             :                                                                          PRETTYINDENT_JOIN);
   12594                 :           0 :                                 break;
   12595                 :             :                         case JOIN_FULL:
   12596                 :           0 :                                 appendContextKeyword(context, " FULL JOIN ",
   12597                 :             :                                                                          -PRETTYINDENT_STD,
   12598                 :             :                                                                          PRETTYINDENT_STD,
   12599                 :             :                                                                          PRETTYINDENT_JOIN);
   12600                 :           0 :                                 break;
   12601                 :             :                         case JOIN_RIGHT:
   12602                 :           0 :                                 appendContextKeyword(context, " RIGHT JOIN ",
   12603                 :             :                                                                          -PRETTYINDENT_STD,
   12604                 :             :                                                                          PRETTYINDENT_STD,
   12605                 :             :                                                                          PRETTYINDENT_JOIN);
   12606                 :           0 :                                 break;
   12607                 :             :                         default:
   12608   [ #  #  #  # ]:           0 :                                 elog(ERROR, "unrecognized join type: %d",
   12609                 :             :                                          (int) j->jointype);
   12610                 :           0 :                 }
   12611                 :             : 
   12612         [ #  # ]:           0 :                 if (need_paren_on_right)
   12613                 :           0 :                         appendStringInfoChar(buf, '(');
   12614                 :           0 :                 get_from_clause_item(j->rarg, query, context);
   12615         [ #  # ]:           0 :                 if (need_paren_on_right)
   12616                 :           0 :                         appendStringInfoChar(buf, ')');
   12617                 :             : 
   12618         [ #  # ]:           0 :                 if (j->usingClause)
   12619                 :             :                 {
   12620                 :           0 :                         ListCell   *lc;
   12621                 :           0 :                         bool            first = true;
   12622                 :             : 
   12623                 :           0 :                         appendStringInfoString(buf, " USING (");
   12624                 :             :                         /* Use the assigned names, not what's in usingClause */
   12625   [ #  #  #  #  :           0 :                         foreach(lc, colinfo->usingNames)
                   #  # ]
   12626                 :             :                         {
   12627                 :           0 :                                 char       *colname = (char *) lfirst(lc);
   12628                 :             : 
   12629         [ #  # ]:           0 :                                 if (first)
   12630                 :           0 :                                         first = false;
   12631                 :             :                                 else
   12632                 :           0 :                                         appendStringInfoString(buf, ", ");
   12633                 :           0 :                                 appendStringInfoString(buf, quote_identifier(colname));
   12634                 :           0 :                         }
   12635                 :           0 :                         appendStringInfoChar(buf, ')');
   12636                 :             : 
   12637         [ #  # ]:           0 :                         if (j->join_using_alias)
   12638                 :           0 :                                 appendStringInfo(buf, " AS %s",
   12639                 :           0 :                                                                  quote_identifier(j->join_using_alias->aliasname));
   12640                 :           0 :                 }
   12641         [ #  # ]:           0 :                 else if (j->quals)
   12642                 :             :                 {
   12643                 :           0 :                         appendStringInfoString(buf, " ON ");
   12644         [ #  # ]:           0 :                         if (!PRETTY_PAREN(context))
   12645                 :           0 :                                 appendStringInfoChar(buf, '(');
   12646                 :           0 :                         get_rule_expr(j->quals, context, false);
   12647         [ #  # ]:           0 :                         if (!PRETTY_PAREN(context))
   12648                 :           0 :                                 appendStringInfoChar(buf, ')');
   12649                 :           0 :                 }
   12650         [ #  # ]:           0 :                 else if (j->jointype != JOIN_INNER)
   12651                 :             :                 {
   12652                 :             :                         /* If we didn't say CROSS JOIN above, we must provide an ON */
   12653                 :           0 :                         appendStringInfoString(buf, " ON TRUE");
   12654                 :           0 :                 }
   12655                 :             : 
   12656   [ #  #  #  # ]:           0 :                 if (!PRETTY_PAREN(context) || j->alias != NULL)
   12657                 :           0 :                         appendStringInfoChar(buf, ')');
   12658                 :             : 
   12659                 :             :                 /* Yes, it's correct to put alias after the right paren ... */
   12660         [ #  # ]:           0 :                 if (j->alias != NULL)
   12661                 :             :                 {
   12662                 :             :                         /*
   12663                 :             :                          * Note that it's correct to emit an alias clause if and only if
   12664                 :             :                          * there was one originally.  Otherwise we'd be converting a named
   12665                 :             :                          * join to unnamed or vice versa, which creates semantic
   12666                 :             :                          * subtleties we don't want.  However, we might print a different
   12667                 :             :                          * alias name than was there originally.
   12668                 :             :                          */
   12669                 :           0 :                         appendStringInfo(buf, " %s",
   12670                 :           0 :                                                          quote_identifier(get_rtable_name(j->rtindex,
   12671                 :           0 :                                                                                                                           context)));
   12672                 :           0 :                         get_column_alias_list(colinfo, context);
   12673                 :           0 :                 }
   12674                 :           0 :         }
   12675                 :             :         else
   12676   [ #  #  #  # ]:           0 :                 elog(ERROR, "unrecognized node type: %d",
   12677                 :             :                          (int) nodeTag(jtnode));
   12678                 :           0 : }
   12679                 :             : 
   12680                 :             : /*
   12681                 :             :  * get_rte_alias - print the relation's alias, if needed
   12682                 :             :  *
   12683                 :             :  * If printed, the alias is preceded by a space, or by " AS " if use_as is true.
   12684                 :             :  */
   12685                 :             : static void
   12686                 :           0 : get_rte_alias(RangeTblEntry *rte, int varno, bool use_as,
   12687                 :             :                           deparse_context *context)
   12688                 :             : {
   12689                 :           0 :         deparse_namespace *dpns = (deparse_namespace *) linitial(context->namespaces);
   12690                 :           0 :         char       *refname = get_rtable_name(varno, context);
   12691                 :           0 :         deparse_columns *colinfo = deparse_columns_fetch(varno, dpns);
   12692                 :           0 :         bool            printalias = false;
   12693                 :             : 
   12694         [ #  # ]:           0 :         if (rte->alias != NULL)
   12695                 :             :         {
   12696                 :             :                 /* Always print alias if user provided one */
   12697                 :           0 :                 printalias = true;
   12698                 :           0 :         }
   12699         [ #  # ]:           0 :         else if (colinfo->printaliases)
   12700                 :             :         {
   12701                 :             :                 /* Always print alias if we need to print column aliases */
   12702                 :           0 :                 printalias = true;
   12703                 :           0 :         }
   12704         [ #  # ]:           0 :         else if (rte->rtekind == RTE_RELATION)
   12705                 :             :         {
   12706                 :             :                 /*
   12707                 :             :                  * No need to print alias if it's same as relation name (this would
   12708                 :             :                  * normally be the case, but not if set_rtable_names had to resolve a
   12709                 :             :                  * conflict).
   12710                 :             :                  */
   12711         [ #  # ]:           0 :                 if (strcmp(refname, get_relation_name(rte->relid)) != 0)
   12712                 :           0 :                         printalias = true;
   12713                 :           0 :         }
   12714         [ #  # ]:           0 :         else if (rte->rtekind == RTE_FUNCTION)
   12715                 :             :         {
   12716                 :             :                 /*
   12717                 :             :                  * For a function RTE, always print alias.  This covers possible
   12718                 :             :                  * renaming of the function and/or instability of the FigureColname
   12719                 :             :                  * rules for things that aren't simple functions.  Note we'd need to
   12720                 :             :                  * force it anyway for the columndef list case.
   12721                 :             :                  */
   12722                 :           0 :                 printalias = true;
   12723                 :           0 :         }
   12724   [ #  #  #  # ]:           0 :         else if (rte->rtekind == RTE_SUBQUERY ||
   12725                 :           0 :                          rte->rtekind == RTE_VALUES)
   12726                 :             :         {
   12727                 :             :                 /*
   12728                 :             :                  * For a subquery, always print alias.  This makes the output
   12729                 :             :                  * SQL-spec-compliant, even though we allow such aliases to be omitted
   12730                 :             :                  * on input.
   12731                 :             :                  */
   12732                 :           0 :                 printalias = true;
   12733                 :           0 :         }
   12734         [ #  # ]:           0 :         else if (rte->rtekind == RTE_CTE)
   12735                 :             :         {
   12736                 :             :                 /*
   12737                 :             :                  * No need to print alias if it's same as CTE name (this would
   12738                 :             :                  * normally be the case, but not if set_rtable_names had to resolve a
   12739                 :             :                  * conflict).
   12740                 :             :                  */
   12741         [ #  # ]:           0 :                 if (strcmp(refname, rte->ctename) != 0)
   12742                 :           0 :                         printalias = true;
   12743                 :           0 :         }
   12744                 :             : 
   12745         [ #  # ]:           0 :         if (printalias)
   12746                 :           0 :                 appendStringInfo(context->buf, "%s%s",
   12747                 :           0 :                                                  use_as ? " AS " : " ",
   12748                 :           0 :                                                  quote_identifier(refname));
   12749                 :           0 : }
   12750                 :             : 
   12751                 :             : /*
   12752                 :             :  * get_column_alias_list - print column alias list for an RTE
   12753                 :             :  *
   12754                 :             :  * Caller must already have printed the relation's alias name.
   12755                 :             :  */
   12756                 :             : static void
   12757                 :           0 : get_column_alias_list(deparse_columns *colinfo, deparse_context *context)
   12758                 :             : {
   12759                 :           0 :         StringInfo      buf = context->buf;
   12760                 :           0 :         int                     i;
   12761                 :           0 :         bool            first = true;
   12762                 :             : 
   12763                 :             :         /* Don't print aliases if not needed */
   12764         [ #  # ]:           0 :         if (!colinfo->printaliases)
   12765                 :           0 :                 return;
   12766                 :             : 
   12767         [ #  # ]:           0 :         for (i = 0; i < colinfo->num_new_cols; i++)
   12768                 :             :         {
   12769                 :           0 :                 char       *colname = colinfo->new_colnames[i];
   12770                 :             : 
   12771         [ #  # ]:           0 :                 if (first)
   12772                 :             :                 {
   12773                 :           0 :                         appendStringInfoChar(buf, '(');
   12774                 :           0 :                         first = false;
   12775                 :           0 :                 }
   12776                 :             :                 else
   12777                 :           0 :                         appendStringInfoString(buf, ", ");
   12778                 :           0 :                 appendStringInfoString(buf, quote_identifier(colname));
   12779                 :           0 :         }
   12780         [ #  # ]:           0 :         if (!first)
   12781                 :           0 :                 appendStringInfoChar(buf, ')');
   12782         [ #  # ]:           0 : }
   12783                 :             : 
   12784                 :             : /*
   12785                 :             :  * get_from_clause_coldeflist - reproduce FROM clause coldeflist
   12786                 :             :  *
   12787                 :             :  * When printing a top-level coldeflist (which is syntactically also the
   12788                 :             :  * relation's column alias list), use column names from colinfo.  But when
   12789                 :             :  * printing a coldeflist embedded inside ROWS FROM(), we prefer to use the
   12790                 :             :  * original coldeflist's names, which are available in rtfunc->funccolnames.
   12791                 :             :  * Pass NULL for colinfo to select the latter behavior.
   12792                 :             :  *
   12793                 :             :  * The coldeflist is appended immediately (no space) to buf.  Caller is
   12794                 :             :  * responsible for ensuring that an alias or AS is present before it.
   12795                 :             :  */
   12796                 :             : static void
   12797                 :           0 : get_from_clause_coldeflist(RangeTblFunction *rtfunc,
   12798                 :             :                                                    deparse_columns *colinfo,
   12799                 :             :                                                    deparse_context *context)
   12800                 :             : {
   12801                 :           0 :         StringInfo      buf = context->buf;
   12802                 :           0 :         ListCell   *l1;
   12803                 :           0 :         ListCell   *l2;
   12804                 :           0 :         ListCell   *l3;
   12805                 :           0 :         ListCell   *l4;
   12806                 :           0 :         int                     i;
   12807                 :             : 
   12808                 :           0 :         appendStringInfoChar(buf, '(');
   12809                 :             : 
   12810                 :           0 :         i = 0;
   12811   [ #  #  #  #  :           0 :         forfour(l1, rtfunc->funccoltypes,
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  #  
                      # ]
   12812                 :             :                         l2, rtfunc->funccoltypmods,
   12813                 :             :                         l3, rtfunc->funccolcollations,
   12814                 :             :                         l4, rtfunc->funccolnames)
   12815                 :             :         {
   12816                 :           0 :                 Oid                     atttypid = lfirst_oid(l1);
   12817                 :           0 :                 int32           atttypmod = lfirst_int(l2);
   12818                 :           0 :                 Oid                     attcollation = lfirst_oid(l3);
   12819                 :           0 :                 char       *attname;
   12820                 :             : 
   12821         [ #  # ]:           0 :                 if (colinfo)
   12822                 :           0 :                         attname = colinfo->colnames[i];
   12823                 :             :                 else
   12824                 :           0 :                         attname = strVal(lfirst(l4));
   12825                 :             : 
   12826         [ #  # ]:           0 :                 Assert(attname);                /* shouldn't be any dropped columns here */
   12827                 :             : 
   12828         [ #  # ]:           0 :                 if (i > 0)
   12829                 :           0 :                         appendStringInfoString(buf, ", ");
   12830                 :           0 :                 appendStringInfo(buf, "%s %s",
   12831                 :           0 :                                                  quote_identifier(attname),
   12832                 :           0 :                                                  format_type_with_typemod(atttypid, atttypmod));
   12833   [ #  #  #  # ]:           0 :                 if (OidIsValid(attcollation) &&
   12834                 :           0 :                         attcollation != get_typcollation(atttypid))
   12835                 :           0 :                         appendStringInfo(buf, " COLLATE %s",
   12836                 :           0 :                                                          generate_collation_name(attcollation));
   12837                 :             : 
   12838                 :           0 :                 i++;
   12839                 :           0 :         }
   12840                 :             : 
   12841                 :           0 :         appendStringInfoChar(buf, ')');
   12842                 :           0 : }
   12843                 :             : 
   12844                 :             : /*
   12845                 :             :  * get_tablesample_def                  - print a TableSampleClause
   12846                 :             :  */
   12847                 :             : static void
   12848                 :           0 : get_tablesample_def(TableSampleClause *tablesample, deparse_context *context)
   12849                 :             : {
   12850                 :           0 :         StringInfo      buf = context->buf;
   12851                 :           0 :         Oid                     argtypes[1];
   12852                 :           0 :         int                     nargs;
   12853                 :           0 :         ListCell   *l;
   12854                 :             : 
   12855                 :             :         /*
   12856                 :             :          * We should qualify the handler's function name if it wouldn't be
   12857                 :             :          * resolved by lookup in the current search path.
   12858                 :             :          */
   12859                 :           0 :         argtypes[0] = INTERNALOID;
   12860                 :           0 :         appendStringInfo(buf, " TABLESAMPLE %s (",
   12861                 :           0 :                                          generate_function_name(tablesample->tsmhandler, 1,
   12862                 :           0 :                                                                                         NIL, argtypes,
   12863                 :             :                                                                                         false, NULL, false));
   12864                 :             : 
   12865                 :           0 :         nargs = 0;
   12866   [ #  #  #  #  :           0 :         foreach(l, tablesample->args)
                   #  # ]
   12867                 :             :         {
   12868         [ #  # ]:           0 :                 if (nargs++ > 0)
   12869                 :           0 :                         appendStringInfoString(buf, ", ");
   12870                 :           0 :                 get_rule_expr((Node *) lfirst(l), context, false);
   12871                 :           0 :         }
   12872                 :           0 :         appendStringInfoChar(buf, ')');
   12873                 :             : 
   12874         [ #  # ]:           0 :         if (tablesample->repeatable != NULL)
   12875                 :             :         {
   12876                 :           0 :                 appendStringInfoString(buf, " REPEATABLE (");
   12877                 :           0 :                 get_rule_expr((Node *) tablesample->repeatable, context, false);
   12878                 :           0 :                 appendStringInfoChar(buf, ')');
   12879                 :           0 :         }
   12880                 :           0 : }
   12881                 :             : 
   12882                 :             : /*
   12883                 :             :  * get_opclass_name                     - fetch name of an index operator class
   12884                 :             :  *
   12885                 :             :  * The opclass name is appended (after a space) to buf.
   12886                 :             :  *
   12887                 :             :  * Output is suppressed if the opclass is the default for the given
   12888                 :             :  * actual_datatype.  (If you don't want this behavior, just pass
   12889                 :             :  * InvalidOid for actual_datatype.)
   12890                 :             :  */
   12891                 :             : static void
   12892                 :           0 : get_opclass_name(Oid opclass, Oid actual_datatype,
   12893                 :             :                                  StringInfo buf)
   12894                 :             : {
   12895                 :           0 :         HeapTuple       ht_opc;
   12896                 :           0 :         Form_pg_opclass opcrec;
   12897                 :           0 :         char       *opcname;
   12898                 :           0 :         char       *nspname;
   12899                 :             : 
   12900                 :           0 :         ht_opc = SearchSysCache1(CLAOID, ObjectIdGetDatum(opclass));
   12901         [ #  # ]:           0 :         if (!HeapTupleIsValid(ht_opc))
   12902   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for opclass %u", opclass);
   12903                 :           0 :         opcrec = (Form_pg_opclass) GETSTRUCT(ht_opc);
   12904                 :             : 
   12905   [ #  #  #  # ]:           0 :         if (!OidIsValid(actual_datatype) ||
   12906                 :           0 :                 GetDefaultOpClass(actual_datatype, opcrec->opcmethod) != opclass)
   12907                 :             :         {
   12908                 :             :                 /* Okay, we need the opclass name.  Do we need to qualify it? */
   12909                 :           0 :                 opcname = NameStr(opcrec->opcname);
   12910         [ #  # ]:           0 :                 if (OpclassIsVisible(opclass))
   12911                 :           0 :                         appendStringInfo(buf, " %s", quote_identifier(opcname));
   12912                 :             :                 else
   12913                 :             :                 {
   12914                 :           0 :                         nspname = get_namespace_name_or_temp(opcrec->opcnamespace);
   12915                 :           0 :                         appendStringInfo(buf, " %s.%s",
   12916                 :           0 :                                                          quote_identifier(nspname),
   12917                 :           0 :                                                          quote_identifier(opcname));
   12918                 :             :                 }
   12919                 :           0 :         }
   12920                 :           0 :         ReleaseSysCache(ht_opc);
   12921                 :           0 : }
   12922                 :             : 
   12923                 :             : /*
   12924                 :             :  * generate_opclass_name
   12925                 :             :  *              Compute the name to display for an opclass specified by OID
   12926                 :             :  *
   12927                 :             :  * The result includes all necessary quoting and schema-prefixing.
   12928                 :             :  */
   12929                 :             : char *
   12930                 :           0 : generate_opclass_name(Oid opclass)
   12931                 :             : {
   12932                 :           0 :         StringInfoData buf;
   12933                 :             : 
   12934                 :           0 :         initStringInfo(&buf);
   12935                 :           0 :         get_opclass_name(opclass, InvalidOid, &buf);
   12936                 :             : 
   12937                 :           0 :         return &buf.data[1];                /* get_opclass_name() prepends space */
   12938                 :           0 : }
   12939                 :             : 
   12940                 :             : /*
   12941                 :             :  * processIndirection - take care of array and subfield assignment
   12942                 :             :  *
   12943                 :             :  * We strip any top-level FieldStore or assignment SubscriptingRef nodes that
   12944                 :             :  * appear in the input, printing them as decoration for the base column
   12945                 :             :  * name (which we assume the caller just printed).  We might also need to
   12946                 :             :  * strip CoerceToDomain nodes, but only ones that appear above assignment
   12947                 :             :  * nodes.
   12948                 :             :  *
   12949                 :             :  * Returns the subexpression that's to be assigned.
   12950                 :             :  */
   12951                 :             : static Node *
   12952                 :           0 : processIndirection(Node *node, deparse_context *context)
   12953                 :             : {
   12954                 :           0 :         StringInfo      buf = context->buf;
   12955                 :           0 :         CoerceToDomain *cdomain = NULL;
   12956                 :             : 
   12957                 :           0 :         for (;;)
   12958                 :             :         {
   12959         [ #  # ]:           0 :                 if (node == NULL)
   12960                 :           0 :                         break;
   12961         [ #  # ]:           0 :                 if (IsA(node, FieldStore))
   12962                 :             :                 {
   12963                 :           0 :                         FieldStore *fstore = (FieldStore *) node;
   12964                 :           0 :                         Oid                     typrelid;
   12965                 :           0 :                         char       *fieldname;
   12966                 :             : 
   12967                 :             :                         /* lookup tuple type */
   12968                 :           0 :                         typrelid = get_typ_typrelid(fstore->resulttype);
   12969         [ #  # ]:           0 :                         if (!OidIsValid(typrelid))
   12970   [ #  #  #  # ]:           0 :                                 elog(ERROR, "argument type %s of FieldStore is not a tuple type",
   12971                 :             :                                          format_type_be(fstore->resulttype));
   12972                 :             : 
   12973                 :             :                         /*
   12974                 :             :                          * Print the field name.  There should only be one target field in
   12975                 :             :                          * stored rules.  There could be more than that in executable
   12976                 :             :                          * target lists, but this function cannot be used for that case.
   12977                 :             :                          */
   12978         [ #  # ]:           0 :                         Assert(list_length(fstore->fieldnums) == 1);
   12979                 :           0 :                         fieldname = get_attname(typrelid,
   12980                 :           0 :                                                                         linitial_int(fstore->fieldnums), false);
   12981                 :           0 :                         appendStringInfo(buf, ".%s", quote_identifier(fieldname));
   12982                 :             : 
   12983                 :             :                         /*
   12984                 :             :                          * We ignore arg since it should be an uninteresting reference to
   12985                 :             :                          * the target column or subcolumn.
   12986                 :             :                          */
   12987                 :           0 :                         node = (Node *) linitial(fstore->newvals);
   12988                 :           0 :                 }
   12989         [ #  # ]:           0 :                 else if (IsA(node, SubscriptingRef))
   12990                 :             :                 {
   12991                 :           0 :                         SubscriptingRef *sbsref = (SubscriptingRef *) node;
   12992                 :             : 
   12993         [ #  # ]:           0 :                         if (sbsref->refassgnexpr == NULL)
   12994                 :           0 :                                 break;
   12995                 :             : 
   12996                 :           0 :                         printSubscripts(sbsref, context);
   12997                 :             : 
   12998                 :             :                         /*
   12999                 :             :                          * We ignore refexpr since it should be an uninteresting reference
   13000                 :             :                          * to the target column or subcolumn.
   13001                 :             :                          */
   13002                 :           0 :                         node = (Node *) sbsref->refassgnexpr;
   13003      [ #  #  # ]:           0 :                 }
   13004         [ #  # ]:           0 :                 else if (IsA(node, CoerceToDomain))
   13005                 :             :                 {
   13006                 :           0 :                         cdomain = (CoerceToDomain *) node;
   13007                 :             :                         /* If it's an explicit domain coercion, we're done */
   13008         [ #  # ]:           0 :                         if (cdomain->coercionformat != COERCE_IMPLICIT_CAST)
   13009                 :           0 :                                 break;
   13010                 :             :                         /* Tentatively descend past the CoerceToDomain */
   13011                 :           0 :                         node = (Node *) cdomain->arg;
   13012                 :           0 :                 }
   13013                 :             :                 else
   13014                 :           0 :                         break;
   13015                 :             :         }
   13016                 :             : 
   13017                 :             :         /*
   13018                 :             :          * If we descended past a CoerceToDomain whose argument turned out not to
   13019                 :             :          * be a FieldStore or array assignment, back up to the CoerceToDomain.
   13020                 :             :          * (This is not enough to be fully correct if there are nested implicit
   13021                 :             :          * CoerceToDomains, but such cases shouldn't ever occur.)
   13022                 :             :          */
   13023   [ #  #  #  # ]:           0 :         if (cdomain && node == (Node *) cdomain->arg)
   13024                 :           0 :                 node = (Node *) cdomain;
   13025                 :             : 
   13026                 :           0 :         return node;
   13027                 :           0 : }
   13028                 :             : 
   13029                 :             : static void
   13030                 :           0 : printSubscripts(SubscriptingRef *sbsref, deparse_context *context)
   13031                 :             : {
   13032                 :           0 :         StringInfo      buf = context->buf;
   13033                 :           0 :         ListCell   *lowlist_item;
   13034                 :           0 :         ListCell   *uplist_item;
   13035                 :             : 
   13036                 :           0 :         lowlist_item = list_head(sbsref->reflowerindexpr);   /* could be NULL */
   13037   [ #  #  #  #  :           0 :         foreach(uplist_item, sbsref->refupperindexpr)
                   #  # ]
   13038                 :             :         {
   13039                 :           0 :                 appendStringInfoChar(buf, '[');
   13040         [ #  # ]:           0 :                 if (lowlist_item)
   13041                 :             :                 {
   13042                 :             :                         /* If subexpression is NULL, get_rule_expr prints nothing */
   13043                 :           0 :                         get_rule_expr((Node *) lfirst(lowlist_item), context, false);
   13044                 :           0 :                         appendStringInfoChar(buf, ':');
   13045                 :           0 :                         lowlist_item = lnext(sbsref->reflowerindexpr, lowlist_item);
   13046                 :           0 :                 }
   13047                 :             :                 /* If subexpression is NULL, get_rule_expr prints nothing */
   13048                 :           0 :                 get_rule_expr((Node *) lfirst(uplist_item), context, false);
   13049                 :           0 :                 appendStringInfoChar(buf, ']');
   13050                 :           0 :         }
   13051                 :           0 : }
   13052                 :             : 
   13053                 :             : /*
   13054                 :             :  * quote_identifier                     - Quote an identifier only if needed
   13055                 :             :  *
   13056                 :             :  * When quotes are needed, we palloc the required space; slightly
   13057                 :             :  * space-wasteful but well worth it for notational simplicity.
   13058                 :             :  */
   13059                 :             : const char *
   13060                 :           0 : quote_identifier(const char *ident)
   13061                 :             : {
   13062                 :             :         /*
   13063                 :             :          * Can avoid quoting if ident starts with a lowercase letter or underscore
   13064                 :             :          * and contains only lowercase letters, digits, and underscores, *and* is
   13065                 :             :          * not any SQL keyword.  Otherwise, supply quotes.
   13066                 :             :          */
   13067                 :           0 :         int                     nquotes = 0;
   13068                 :           0 :         bool            safe;
   13069                 :           0 :         const char *ptr;
   13070                 :           0 :         char       *result;
   13071                 :           0 :         char       *optr;
   13072                 :             : 
   13073                 :             :         /*
   13074                 :             :          * would like to use <ctype.h> macros here, but they might yield unwanted
   13075                 :             :          * locale-specific results...
   13076                 :             :          */
   13077   [ #  #  #  # ]:           0 :         safe = ((ident[0] >= 'a' && ident[0] <= 'z') || ident[0] == '_');
   13078                 :             : 
   13079         [ #  # ]:           0 :         for (ptr = ident; *ptr; ptr++)
   13080                 :             :         {
   13081                 :           0 :                 char            ch = *ptr;
   13082                 :             : 
   13083         [ #  # ]:           0 :                 if ((ch >= 'a' && ch <= 'z') ||
   13084   [ #  #  #  # ]:           0 :                         (ch >= '0' && ch <= '9') ||
   13085                 :           0 :                         (ch == '_'))
   13086                 :             :                 {
   13087                 :             :                         /* okay */
   13088                 :           0 :                 }
   13089                 :             :                 else
   13090                 :             :                 {
   13091                 :           0 :                         safe = false;
   13092         [ #  # ]:           0 :                         if (ch == '"')
   13093                 :           0 :                                 nquotes++;
   13094                 :             :                 }
   13095                 :           0 :         }
   13096                 :             : 
   13097         [ #  # ]:           0 :         if (quote_all_identifiers)
   13098                 :           0 :                 safe = false;
   13099                 :             : 
   13100         [ #  # ]:           0 :         if (safe)
   13101                 :             :         {
   13102                 :             :                 /*
   13103                 :             :                  * Check for keyword.  We quote keywords except for unreserved ones.
   13104                 :             :                  * (In some cases we could avoid quoting a col_name or type_func_name
   13105                 :             :                  * keyword, but it seems much harder than it's worth to tell that.)
   13106                 :             :                  *
   13107                 :             :                  * Note: ScanKeywordLookup() does case-insensitive comparison, but
   13108                 :             :                  * that's fine, since we already know we have all-lower-case.
   13109                 :             :                  */
   13110                 :           0 :                 int                     kwnum = ScanKeywordLookup(ident, &ScanKeywords);
   13111                 :             : 
   13112   [ #  #  #  # ]:           0 :                 if (kwnum >= 0 && ScanKeywordCategories[kwnum] != UNRESERVED_KEYWORD)
   13113                 :           0 :                         safe = false;
   13114                 :           0 :         }
   13115                 :             : 
   13116         [ #  # ]:           0 :         if (safe)
   13117                 :           0 :                 return ident;                   /* no change needed */
   13118                 :             : 
   13119                 :           0 :         result = (char *) palloc(strlen(ident) + nquotes + 2 + 1);
   13120                 :             : 
   13121                 :           0 :         optr = result;
   13122                 :           0 :         *optr++ = '"';
   13123         [ #  # ]:           0 :         for (ptr = ident; *ptr; ptr++)
   13124                 :             :         {
   13125                 :           0 :                 char            ch = *ptr;
   13126                 :             : 
   13127         [ #  # ]:           0 :                 if (ch == '"')
   13128                 :           0 :                         *optr++ = '"';
   13129                 :           0 :                 *optr++ = ch;
   13130                 :           0 :         }
   13131                 :           0 :         *optr++ = '"';
   13132                 :           0 :         *optr = '\0';
   13133                 :             : 
   13134                 :           0 :         return result;
   13135                 :           0 : }
   13136                 :             : 
   13137                 :             : /*
   13138                 :             :  * quote_qualified_identifier   - Quote a possibly-qualified identifier
   13139                 :             :  *
   13140                 :             :  * Return a name of the form qualifier.ident, or just ident if qualifier
   13141                 :             :  * is NULL, quoting each component if necessary.  The result is palloc'd.
   13142                 :             :  */
   13143                 :             : char *
   13144                 :           0 : quote_qualified_identifier(const char *qualifier,
   13145                 :             :                                                    const char *ident)
   13146                 :             : {
   13147                 :           0 :         StringInfoData buf;
   13148                 :             : 
   13149                 :           0 :         initStringInfo(&buf);
   13150         [ #  # ]:           0 :         if (qualifier)
   13151                 :           0 :                 appendStringInfo(&buf, "%s.", quote_identifier(qualifier));
   13152                 :           0 :         appendStringInfoString(&buf, quote_identifier(ident));
   13153                 :           0 :         return buf.data;
   13154                 :           0 : }
   13155                 :             : 
   13156                 :             : /*
   13157                 :             :  * get_relation_name
   13158                 :             :  *              Get the unqualified name of a relation specified by OID
   13159                 :             :  *
   13160                 :             :  * This differs from the underlying get_rel_name() function in that it will
   13161                 :             :  * throw error instead of silently returning NULL if the OID is bad.
   13162                 :             :  */
   13163                 :             : static char *
   13164                 :           0 : get_relation_name(Oid relid)
   13165                 :             : {
   13166                 :           0 :         char       *relname = get_rel_name(relid);
   13167                 :             : 
   13168         [ #  # ]:           0 :         if (!relname)
   13169   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for relation %u", relid);
   13170                 :           0 :         return relname;
   13171                 :           0 : }
   13172                 :             : 
   13173                 :             : /*
   13174                 :             :  * generate_relation_name
   13175                 :             :  *              Compute the name to display for a relation specified by OID
   13176                 :             :  *
   13177                 :             :  * The result includes all necessary quoting and schema-prefixing.
   13178                 :             :  *
   13179                 :             :  * If namespaces isn't NIL, it must be a list of deparse_namespace nodes.
   13180                 :             :  * We will forcibly qualify the relation name if it equals any CTE name
   13181                 :             :  * visible in the namespace list.
   13182                 :             :  */
   13183                 :             : static char *
   13184                 :           0 : generate_relation_name(Oid relid, List *namespaces)
   13185                 :             : {
   13186                 :           0 :         HeapTuple       tp;
   13187                 :           0 :         Form_pg_class reltup;
   13188                 :           0 :         bool            need_qual;
   13189                 :           0 :         ListCell   *nslist;
   13190                 :           0 :         char       *relname;
   13191                 :           0 :         char       *nspname;
   13192                 :           0 :         char       *result;
   13193                 :             : 
   13194                 :           0 :         tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
   13195         [ #  # ]:           0 :         if (!HeapTupleIsValid(tp))
   13196   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for relation %u", relid);
   13197                 :           0 :         reltup = (Form_pg_class) GETSTRUCT(tp);
   13198                 :           0 :         relname = NameStr(reltup->relname);
   13199                 :             : 
   13200                 :             :         /* Check for conflicting CTE name */
   13201                 :           0 :         need_qual = false;
   13202   [ #  #  #  #  :           0 :         foreach(nslist, namespaces)
                   #  # ]
   13203                 :             :         {
   13204                 :           0 :                 deparse_namespace *dpns = (deparse_namespace *) lfirst(nslist);
   13205                 :           0 :                 ListCell   *ctlist;
   13206                 :             : 
   13207   [ #  #  #  #  :           0 :                 foreach(ctlist, dpns->ctes)
                   #  # ]
   13208                 :             :                 {
   13209                 :           0 :                         CommonTableExpr *cte = (CommonTableExpr *) lfirst(ctlist);
   13210                 :             : 
   13211         [ #  # ]:           0 :                         if (strcmp(cte->ctename, relname) == 0)
   13212                 :             :                         {
   13213                 :           0 :                                 need_qual = true;
   13214                 :           0 :                                 break;
   13215                 :             :                         }
   13216         [ #  # ]:           0 :                 }
   13217         [ #  # ]:           0 :                 if (need_qual)
   13218                 :           0 :                         break;
   13219         [ #  # ]:           0 :         }
   13220                 :             : 
   13221                 :             :         /* Otherwise, qualify the name if not visible in search path */
   13222         [ #  # ]:           0 :         if (!need_qual)
   13223                 :           0 :                 need_qual = !RelationIsVisible(relid);
   13224                 :             : 
   13225         [ #  # ]:           0 :         if (need_qual)
   13226                 :           0 :                 nspname = get_namespace_name_or_temp(reltup->relnamespace);
   13227                 :             :         else
   13228                 :           0 :                 nspname = NULL;
   13229                 :             : 
   13230                 :           0 :         result = quote_qualified_identifier(nspname, relname);
   13231                 :             : 
   13232                 :           0 :         ReleaseSysCache(tp);
   13233                 :             : 
   13234                 :           0 :         return result;
   13235                 :           0 : }
   13236                 :             : 
   13237                 :             : /*
   13238                 :             :  * generate_qualified_relation_name
   13239                 :             :  *              Compute the name to display for a relation specified by OID
   13240                 :             :  *
   13241                 :             :  * As above, but unconditionally schema-qualify the name.
   13242                 :             :  */
   13243                 :             : static char *
   13244                 :           0 : generate_qualified_relation_name(Oid relid)
   13245                 :             : {
   13246                 :           0 :         HeapTuple       tp;
   13247                 :           0 :         Form_pg_class reltup;
   13248                 :           0 :         char       *relname;
   13249                 :           0 :         char       *nspname;
   13250                 :           0 :         char       *result;
   13251                 :             : 
   13252                 :           0 :         tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
   13253         [ #  # ]:           0 :         if (!HeapTupleIsValid(tp))
   13254   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for relation %u", relid);
   13255                 :           0 :         reltup = (Form_pg_class) GETSTRUCT(tp);
   13256                 :           0 :         relname = NameStr(reltup->relname);
   13257                 :             : 
   13258                 :           0 :         nspname = get_namespace_name_or_temp(reltup->relnamespace);
   13259         [ #  # ]:           0 :         if (!nspname)
   13260   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for namespace %u",
   13261                 :             :                          reltup->relnamespace);
   13262                 :             : 
   13263                 :           0 :         result = quote_qualified_identifier(nspname, relname);
   13264                 :             : 
   13265                 :           0 :         ReleaseSysCache(tp);
   13266                 :             : 
   13267                 :           0 :         return result;
   13268                 :           0 : }
   13269                 :             : 
   13270                 :             : /*
   13271                 :             :  * generate_function_name
   13272                 :             :  *              Compute the name to display for a function specified by OID,
   13273                 :             :  *              given that it is being called with the specified actual arg names and
   13274                 :             :  *              types.  (Those matter because of ambiguous-function resolution rules.)
   13275                 :             :  *
   13276                 :             :  * If we're dealing with a potentially variadic function (in practice, this
   13277                 :             :  * means a FuncExpr or Aggref, not some other way of calling a function), then
   13278                 :             :  * has_variadic must specify whether variadic arguments have been merged,
   13279                 :             :  * and *use_variadic_p will be set to indicate whether to print VARIADIC in
   13280                 :             :  * the output.  For non-FuncExpr cases, has_variadic should be false and
   13281                 :             :  * use_variadic_p can be NULL.
   13282                 :             :  *
   13283                 :             :  * inGroupBy must be true if we're deparsing a GROUP BY clause.
   13284                 :             :  *
   13285                 :             :  * The result includes all necessary quoting and schema-prefixing.
   13286                 :             :  */
   13287                 :             : static char *
   13288                 :           0 : generate_function_name(Oid funcid, int nargs, List *argnames, Oid *argtypes,
   13289                 :             :                                            bool has_variadic, bool *use_variadic_p,
   13290                 :             :                                            bool inGroupBy)
   13291                 :             : {
   13292                 :           0 :         char       *result;
   13293                 :           0 :         HeapTuple       proctup;
   13294                 :           0 :         Form_pg_proc procform;
   13295                 :           0 :         char       *proname;
   13296                 :           0 :         bool            use_variadic;
   13297                 :           0 :         char       *nspname;
   13298                 :           0 :         FuncDetailCode p_result;
   13299                 :           0 :         int                     fgc_flags;
   13300                 :           0 :         Oid                     p_funcid;
   13301                 :           0 :         Oid                     p_rettype;
   13302                 :           0 :         bool            p_retset;
   13303                 :           0 :         int                     p_nvargs;
   13304                 :           0 :         Oid                     p_vatype;
   13305                 :           0 :         Oid                *p_true_typeids;
   13306                 :           0 :         bool            force_qualify = false;
   13307                 :             : 
   13308                 :           0 :         proctup = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
   13309         [ #  # ]:           0 :         if (!HeapTupleIsValid(proctup))
   13310   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for function %u", funcid);
   13311                 :           0 :         procform = (Form_pg_proc) GETSTRUCT(proctup);
   13312                 :           0 :         proname = NameStr(procform->proname);
   13313                 :             : 
   13314                 :             :         /*
   13315                 :             :          * Due to parser hacks to avoid needing to reserve CUBE, we need to force
   13316                 :             :          * qualification of some function names within GROUP BY.
   13317                 :             :          */
   13318         [ #  # ]:           0 :         if (inGroupBy)
   13319                 :             :         {
   13320   [ #  #  #  # ]:           0 :                 if (strcmp(proname, "cube") == 0 || strcmp(proname, "rollup") == 0)
   13321                 :           0 :                         force_qualify = true;
   13322                 :           0 :         }
   13323                 :             : 
   13324                 :             :         /*
   13325                 :             :          * Determine whether VARIADIC should be printed.  We must do this first
   13326                 :             :          * since it affects the lookup rules in func_get_detail().
   13327                 :             :          *
   13328                 :             :          * We always print VARIADIC if the function has a merged variadic-array
   13329                 :             :          * argument.  Note that this is always the case for functions taking a
   13330                 :             :          * VARIADIC argument type other than VARIADIC ANY.  If we omitted VARIADIC
   13331                 :             :          * and printed the array elements as separate arguments, the call could
   13332                 :             :          * match a newer non-VARIADIC function.
   13333                 :             :          */
   13334         [ #  # ]:           0 :         if (use_variadic_p)
   13335                 :             :         {
   13336                 :             :                 /* Parser should not have set funcvariadic unless fn is variadic */
   13337   [ #  #  #  # ]:           0 :                 Assert(!has_variadic || OidIsValid(procform->provariadic));
   13338                 :           0 :                 use_variadic = has_variadic;
   13339                 :           0 :                 *use_variadic_p = use_variadic;
   13340                 :           0 :         }
   13341                 :             :         else
   13342                 :             :         {
   13343         [ #  # ]:           0 :                 Assert(!has_variadic);
   13344                 :           0 :                 use_variadic = false;
   13345                 :             :         }
   13346                 :             : 
   13347                 :             :         /*
   13348                 :             :          * The idea here is to schema-qualify only if the parser would fail to
   13349                 :             :          * resolve the correct function given the unqualified func name with the
   13350                 :             :          * specified argtypes and VARIADIC flag.  But if we already decided to
   13351                 :             :          * force qualification, then we can skip the lookup and pretend we didn't
   13352                 :             :          * find it.
   13353                 :             :          */
   13354         [ #  # ]:           0 :         if (!force_qualify)
   13355                 :           0 :                 p_result = func_get_detail(list_make1(makeString(proname)),
   13356                 :           0 :                                                                    NIL, argnames, nargs, argtypes,
   13357                 :           0 :                                                                    !use_variadic, true, false,
   13358                 :             :                                                                    &fgc_flags,
   13359                 :             :                                                                    &p_funcid, &p_rettype,
   13360                 :             :                                                                    &p_retset, &p_nvargs, &p_vatype,
   13361                 :             :                                                                    &p_true_typeids, NULL);
   13362                 :             :         else
   13363                 :             :         {
   13364                 :           0 :                 p_result = FUNCDETAIL_NOTFOUND;
   13365                 :           0 :                 p_funcid = InvalidOid;
   13366                 :             :         }
   13367                 :             : 
   13368         [ #  # ]:           0 :         if ((p_result == FUNCDETAIL_NORMAL ||
   13369         [ #  # ]:           0 :                  p_result == FUNCDETAIL_AGGREGATE ||
   13370         [ #  # ]:           0 :                  p_result == FUNCDETAIL_WINDOWFUNC) &&
   13371                 :           0 :                 p_funcid == funcid)
   13372                 :           0 :                 nspname = NULL;
   13373                 :             :         else
   13374                 :           0 :                 nspname = get_namespace_name_or_temp(procform->pronamespace);
   13375                 :             : 
   13376                 :           0 :         result = quote_qualified_identifier(nspname, proname);
   13377                 :             : 
   13378                 :           0 :         ReleaseSysCache(proctup);
   13379                 :             : 
   13380                 :           0 :         return result;
   13381                 :           0 : }
   13382                 :             : 
   13383                 :             : /*
   13384                 :             :  * generate_operator_name
   13385                 :             :  *              Compute the name to display for an operator specified by OID,
   13386                 :             :  *              given that it is being called with the specified actual arg types.
   13387                 :             :  *              (Arg types matter because of ambiguous-operator resolution rules.
   13388                 :             :  *              Pass InvalidOid for unused arg of a unary operator.)
   13389                 :             :  *
   13390                 :             :  * The result includes all necessary quoting and schema-prefixing,
   13391                 :             :  * plus the OPERATOR() decoration needed to use a qualified operator name
   13392                 :             :  * in an expression.
   13393                 :             :  */
   13394                 :             : static char *
   13395                 :           0 : generate_operator_name(Oid operid, Oid arg1, Oid arg2)
   13396                 :             : {
   13397                 :           0 :         StringInfoData buf;
   13398                 :           0 :         HeapTuple       opertup;
   13399                 :           0 :         Form_pg_operator operform;
   13400                 :           0 :         char       *oprname;
   13401                 :           0 :         char       *nspname;
   13402                 :           0 :         Operator        p_result;
   13403                 :             : 
   13404                 :           0 :         initStringInfo(&buf);
   13405                 :             : 
   13406                 :           0 :         opertup = SearchSysCache1(OPEROID, ObjectIdGetDatum(operid));
   13407         [ #  # ]:           0 :         if (!HeapTupleIsValid(opertup))
   13408   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for operator %u", operid);
   13409                 :           0 :         operform = (Form_pg_operator) GETSTRUCT(opertup);
   13410                 :           0 :         oprname = NameStr(operform->oprname);
   13411                 :             : 
   13412                 :             :         /*
   13413                 :             :          * The idea here is to schema-qualify only if the parser would fail to
   13414                 :             :          * resolve the correct operator given the unqualified op name with the
   13415                 :             :          * specified argtypes.
   13416                 :             :          */
   13417      [ #  #  # ]:           0 :         switch (operform->oprkind)
   13418                 :             :         {
   13419                 :             :                 case 'b':
   13420                 :           0 :                         p_result = oper(NULL, list_make1(makeString(oprname)), arg1, arg2,
   13421                 :             :                                                         true, -1);
   13422                 :           0 :                         break;
   13423                 :             :                 case 'l':
   13424                 :           0 :                         p_result = left_oper(NULL, list_make1(makeString(oprname)), arg2,
   13425                 :             :                                                                  true, -1);
   13426                 :           0 :                         break;
   13427                 :             :                 default:
   13428   [ #  #  #  # ]:           0 :                         elog(ERROR, "unrecognized oprkind: %d", operform->oprkind);
   13429                 :           0 :                         p_result = NULL;        /* keep compiler quiet */
   13430                 :           0 :                         break;
   13431                 :             :         }
   13432                 :             : 
   13433   [ #  #  #  # ]:           0 :         if (p_result != NULL && oprid(p_result) == operid)
   13434                 :           0 :                 nspname = NULL;
   13435                 :             :         else
   13436                 :             :         {
   13437                 :           0 :                 nspname = get_namespace_name_or_temp(operform->oprnamespace);
   13438                 :           0 :                 appendStringInfo(&buf, "OPERATOR(%s.", quote_identifier(nspname));
   13439                 :             :         }
   13440                 :             : 
   13441                 :           0 :         appendStringInfoString(&buf, oprname);
   13442                 :             : 
   13443         [ #  # ]:           0 :         if (nspname)
   13444                 :           0 :                 appendStringInfoChar(&buf, ')');
   13445                 :             : 
   13446         [ #  # ]:           0 :         if (p_result != NULL)
   13447                 :           0 :                 ReleaseSysCache(p_result);
   13448                 :             : 
   13449                 :           0 :         ReleaseSysCache(opertup);
   13450                 :             : 
   13451                 :           0 :         return buf.data;
   13452                 :           0 : }
   13453                 :             : 
   13454                 :             : /*
   13455                 :             :  * generate_operator_clause --- generate a binary-operator WHERE clause
   13456                 :             :  *
   13457                 :             :  * This is used for internally-generated-and-executed SQL queries, where
   13458                 :             :  * precision is essential and readability is secondary.  The basic
   13459                 :             :  * requirement is to append "leftop op rightop" to buf, where leftop and
   13460                 :             :  * rightop are given as strings and are assumed to yield types leftoptype
   13461                 :             :  * and rightoptype; the operator is identified by OID.  The complexity
   13462                 :             :  * comes from needing to be sure that the parser will select the desired
   13463                 :             :  * operator when the query is parsed.  We always name the operator using
   13464                 :             :  * OPERATOR(schema.op) syntax, so as to avoid search-path uncertainties.
   13465                 :             :  * We have to emit casts too, if either input isn't already the input type
   13466                 :             :  * of the operator; else we are at the mercy of the parser's heuristics for
   13467                 :             :  * ambiguous-operator resolution.  The caller must ensure that leftop and
   13468                 :             :  * rightop are suitable arguments for a cast operation; it's best to insert
   13469                 :             :  * parentheses if they aren't just variables or parameters.
   13470                 :             :  */
   13471                 :             : void
   13472                 :           0 : generate_operator_clause(StringInfo buf,
   13473                 :             :                                                  const char *leftop, Oid leftoptype,
   13474                 :             :                                                  Oid opoid,
   13475                 :             :                                                  const char *rightop, Oid rightoptype)
   13476                 :             : {
   13477                 :           0 :         HeapTuple       opertup;
   13478                 :           0 :         Form_pg_operator operform;
   13479                 :           0 :         char       *oprname;
   13480                 :           0 :         char       *nspname;
   13481                 :             : 
   13482                 :           0 :         opertup = SearchSysCache1(OPEROID, ObjectIdGetDatum(opoid));
   13483         [ #  # ]:           0 :         if (!HeapTupleIsValid(opertup))
   13484   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for operator %u", opoid);
   13485                 :           0 :         operform = (Form_pg_operator) GETSTRUCT(opertup);
   13486         [ #  # ]:           0 :         Assert(operform->oprkind == 'b');
   13487                 :           0 :         oprname = NameStr(operform->oprname);
   13488                 :             : 
   13489                 :           0 :         nspname = get_namespace_name(operform->oprnamespace);
   13490                 :             : 
   13491                 :           0 :         appendStringInfoString(buf, leftop);
   13492         [ #  # ]:           0 :         if (leftoptype != operform->oprleft)
   13493                 :           0 :                 add_cast_to(buf, operform->oprleft);
   13494                 :           0 :         appendStringInfo(buf, " OPERATOR(%s.", quote_identifier(nspname));
   13495                 :           0 :         appendStringInfoString(buf, oprname);
   13496                 :           0 :         appendStringInfo(buf, ") %s", rightop);
   13497         [ #  # ]:           0 :         if (rightoptype != operform->oprright)
   13498                 :           0 :                 add_cast_to(buf, operform->oprright);
   13499                 :             : 
   13500                 :           0 :         ReleaseSysCache(opertup);
   13501                 :           0 : }
   13502                 :             : 
   13503                 :             : /*
   13504                 :             :  * Add a cast specification to buf.  We spell out the type name the hard way,
   13505                 :             :  * intentionally not using format_type_be().  This is to avoid corner cases
   13506                 :             :  * for CHARACTER, BIT, and perhaps other types, where specifying the type
   13507                 :             :  * using SQL-standard syntax results in undesirable data truncation.  By
   13508                 :             :  * doing it this way we can be certain that the cast will have default (-1)
   13509                 :             :  * target typmod.
   13510                 :             :  */
   13511                 :             : static void
   13512                 :           0 : add_cast_to(StringInfo buf, Oid typid)
   13513                 :             : {
   13514                 :           0 :         HeapTuple       typetup;
   13515                 :           0 :         Form_pg_type typform;
   13516                 :           0 :         char       *typname;
   13517                 :           0 :         char       *nspname;
   13518                 :             : 
   13519                 :           0 :         typetup = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
   13520         [ #  # ]:           0 :         if (!HeapTupleIsValid(typetup))
   13521   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for type %u", typid);
   13522                 :           0 :         typform = (Form_pg_type) GETSTRUCT(typetup);
   13523                 :             : 
   13524                 :           0 :         typname = NameStr(typform->typname);
   13525                 :           0 :         nspname = get_namespace_name_or_temp(typform->typnamespace);
   13526                 :             : 
   13527                 :           0 :         appendStringInfo(buf, "::%s.%s",
   13528                 :           0 :                                          quote_identifier(nspname), quote_identifier(typname));
   13529                 :             : 
   13530                 :           0 :         ReleaseSysCache(typetup);
   13531                 :           0 : }
   13532                 :             : 
   13533                 :             : /*
   13534                 :             :  * generate_qualified_type_name
   13535                 :             :  *              Compute the name to display for a type specified by OID
   13536                 :             :  *
   13537                 :             :  * This is different from format_type_be() in that we unconditionally
   13538                 :             :  * schema-qualify the name.  That also means no special syntax for
   13539                 :             :  * SQL-standard type names ... although in current usage, this should
   13540                 :             :  * only get used for domains, so such cases wouldn't occur anyway.
   13541                 :             :  */
   13542                 :             : static char *
   13543                 :           0 : generate_qualified_type_name(Oid typid)
   13544                 :             : {
   13545                 :           0 :         HeapTuple       tp;
   13546                 :           0 :         Form_pg_type typtup;
   13547                 :           0 :         char       *typname;
   13548                 :           0 :         char       *nspname;
   13549                 :           0 :         char       *result;
   13550                 :             : 
   13551                 :           0 :         tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
   13552         [ #  # ]:           0 :         if (!HeapTupleIsValid(tp))
   13553   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for type %u", typid);
   13554                 :           0 :         typtup = (Form_pg_type) GETSTRUCT(tp);
   13555                 :           0 :         typname = NameStr(typtup->typname);
   13556                 :             : 
   13557                 :           0 :         nspname = get_namespace_name_or_temp(typtup->typnamespace);
   13558         [ #  # ]:           0 :         if (!nspname)
   13559   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for namespace %u",
   13560                 :             :                          typtup->typnamespace);
   13561                 :             : 
   13562                 :           0 :         result = quote_qualified_identifier(nspname, typname);
   13563                 :             : 
   13564                 :           0 :         ReleaseSysCache(tp);
   13565                 :             : 
   13566                 :           0 :         return result;
   13567                 :           0 : }
   13568                 :             : 
   13569                 :             : /*
   13570                 :             :  * generate_collation_name
   13571                 :             :  *              Compute the name to display for a collation specified by OID
   13572                 :             :  *
   13573                 :             :  * The result includes all necessary quoting and schema-prefixing.
   13574                 :             :  */
   13575                 :             : char *
   13576                 :           0 : generate_collation_name(Oid collid)
   13577                 :             : {
   13578                 :           0 :         HeapTuple       tp;
   13579                 :           0 :         Form_pg_collation colltup;
   13580                 :           0 :         char       *collname;
   13581                 :           0 :         char       *nspname;
   13582                 :           0 :         char       *result;
   13583                 :             : 
   13584                 :           0 :         tp = SearchSysCache1(COLLOID, ObjectIdGetDatum(collid));
   13585         [ #  # ]:           0 :         if (!HeapTupleIsValid(tp))
   13586   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for collation %u", collid);
   13587                 :           0 :         colltup = (Form_pg_collation) GETSTRUCT(tp);
   13588                 :           0 :         collname = NameStr(colltup->collname);
   13589                 :             : 
   13590         [ #  # ]:           0 :         if (!CollationIsVisible(collid))
   13591                 :           0 :                 nspname = get_namespace_name_or_temp(colltup->collnamespace);
   13592                 :             :         else
   13593                 :           0 :                 nspname = NULL;
   13594                 :             : 
   13595                 :           0 :         result = quote_qualified_identifier(nspname, collname);
   13596                 :             : 
   13597                 :           0 :         ReleaseSysCache(tp);
   13598                 :             : 
   13599                 :           0 :         return result;
   13600                 :           0 : }
   13601                 :             : 
   13602                 :             : /*
   13603                 :             :  * Given a C string, produce a TEXT datum.
   13604                 :             :  *
   13605                 :             :  * We assume that the input was palloc'd and may be freed.
   13606                 :             :  */
   13607                 :             : static text *
   13608                 :           0 : string_to_text(char *str)
   13609                 :             : {
   13610                 :           0 :         text       *result;
   13611                 :             : 
   13612                 :           0 :         result = cstring_to_text(str);
   13613                 :           0 :         pfree(str);
   13614                 :           0 :         return result;
   13615                 :           0 : }
   13616                 :             : 
   13617                 :             : /*
   13618                 :             :  * Generate a C string representing a relation options from text[] datum.
   13619                 :             :  */
   13620                 :             : static void
   13621                 :           0 : get_reloptions(StringInfo buf, Datum reloptions)
   13622                 :             : {
   13623                 :           0 :         Datum      *options;
   13624                 :           0 :         int                     noptions;
   13625                 :           0 :         int                     i;
   13626                 :             : 
   13627                 :           0 :         deconstruct_array_builtin(DatumGetArrayTypeP(reloptions), TEXTOID,
   13628                 :             :                                                           &options, NULL, &noptions);
   13629                 :             : 
   13630         [ #  # ]:           0 :         for (i = 0; i < noptions; i++)
   13631                 :             :         {
   13632                 :           0 :                 char       *option = TextDatumGetCString(options[i]);
   13633                 :           0 :                 char       *name;
   13634                 :           0 :                 char       *separator;
   13635                 :           0 :                 char       *value;
   13636                 :             : 
   13637                 :             :                 /*
   13638                 :             :                  * Each array element should have the form name=value.  If the "=" is
   13639                 :             :                  * missing for some reason, treat it like an empty value.
   13640                 :             :                  */
   13641                 :           0 :                 name = option;
   13642                 :           0 :                 separator = strchr(option, '=');
   13643         [ #  # ]:           0 :                 if (separator)
   13644                 :             :                 {
   13645                 :           0 :                         *separator = '\0';
   13646                 :           0 :                         value = separator + 1;
   13647                 :           0 :                 }
   13648                 :             :                 else
   13649                 :           0 :                         value = "";
   13650                 :             : 
   13651         [ #  # ]:           0 :                 if (i > 0)
   13652                 :           0 :                         appendStringInfoString(buf, ", ");
   13653                 :           0 :                 appendStringInfo(buf, "%s=", quote_identifier(name));
   13654                 :             : 
   13655                 :             :                 /*
   13656                 :             :                  * In general we need to quote the value; but to avoid unnecessary
   13657                 :             :                  * clutter, do not quote if it is an identifier that would not need
   13658                 :             :                  * quoting.  (We could also allow numbers, but that is a bit trickier
   13659                 :             :                  * than it looks --- for example, are leading zeroes significant?  We
   13660                 :             :                  * don't want to assume very much here about what custom reloptions
   13661                 :             :                  * might mean.)
   13662                 :             :                  */
   13663         [ #  # ]:           0 :                 if (quote_identifier(value) == value)
   13664                 :           0 :                         appendStringInfoString(buf, value);
   13665                 :             :                 else
   13666                 :           0 :                         simple_quote_literal(buf, value);
   13667                 :             : 
   13668                 :           0 :                 pfree(option);
   13669                 :           0 :         }
   13670                 :           0 : }
   13671                 :             : 
   13672                 :             : /*
   13673                 :             :  * Generate a C string representing a relation's reloptions, or NULL if none.
   13674                 :             :  */
   13675                 :             : static char *
   13676                 :           0 : flatten_reloptions(Oid relid)
   13677                 :             : {
   13678                 :           0 :         char       *result = NULL;
   13679                 :           0 :         HeapTuple       tuple;
   13680                 :           0 :         Datum           reloptions;
   13681                 :           0 :         bool            isnull;
   13682                 :             : 
   13683                 :           0 :         tuple = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
   13684         [ #  # ]:           0 :         if (!HeapTupleIsValid(tuple))
   13685   [ #  #  #  # ]:           0 :                 elog(ERROR, "cache lookup failed for relation %u", relid);
   13686                 :             : 
   13687                 :           0 :         reloptions = SysCacheGetAttr(RELOID, tuple,
   13688                 :             :                                                                  Anum_pg_class_reloptions, &isnull);
   13689         [ #  # ]:           0 :         if (!isnull)
   13690                 :             :         {
   13691                 :           0 :                 StringInfoData buf;
   13692                 :             : 
   13693                 :           0 :                 initStringInfo(&buf);
   13694                 :           0 :                 get_reloptions(&buf, reloptions);
   13695                 :             : 
   13696                 :           0 :                 result = buf.data;
   13697                 :           0 :         }
   13698                 :             : 
   13699                 :           0 :         ReleaseSysCache(tuple);
   13700                 :             : 
   13701                 :           0 :         return result;
   13702                 :           0 : }
   13703                 :             : 
   13704                 :             : /*
   13705                 :             :  * get_range_partbound_string
   13706                 :             :  *              A C string representation of one range partition bound
   13707                 :             :  */
   13708                 :             : char *
   13709                 :           0 : get_range_partbound_string(List *bound_datums)
   13710                 :             : {
   13711                 :           0 :         deparse_context context;
   13712                 :           0 :         StringInfoData buf;
   13713                 :           0 :         ListCell   *cell;
   13714                 :           0 :         char       *sep;
   13715                 :             : 
   13716                 :           0 :         initStringInfo(&buf);
   13717                 :           0 :         memset(&context, 0, sizeof(deparse_context));
   13718                 :           0 :         context.buf = &buf;
   13719                 :             : 
   13720                 :           0 :         appendStringInfoChar(&buf, '(');
   13721                 :           0 :         sep = "";
   13722   [ #  #  #  #  :           0 :         foreach(cell, bound_datums)
                   #  # ]
   13723                 :             :         {
   13724                 :           0 :                 PartitionRangeDatum *datum =
   13725                 :           0 :                         lfirst_node(PartitionRangeDatum, cell);
   13726                 :             : 
   13727                 :           0 :                 appendStringInfoString(&buf, sep);
   13728         [ #  # ]:           0 :                 if (datum->kind == PARTITION_RANGE_DATUM_MINVALUE)
   13729                 :           0 :                         appendStringInfoString(&buf, "MINVALUE");
   13730         [ #  # ]:           0 :                 else if (datum->kind == PARTITION_RANGE_DATUM_MAXVALUE)
   13731                 :           0 :                         appendStringInfoString(&buf, "MAXVALUE");
   13732                 :             :                 else
   13733                 :             :                 {
   13734                 :           0 :                         Const      *val = castNode(Const, datum->value);
   13735                 :             : 
   13736                 :           0 :                         get_const_expr(val, &context, -1);
   13737                 :           0 :                 }
   13738                 :           0 :                 sep = ", ";
   13739                 :           0 :         }
   13740                 :           0 :         appendStringInfoChar(&buf, ')');
   13741                 :             : 
   13742                 :           0 :         return buf.data;
   13743                 :           0 : }
        

Generated by: LCOV version 2.3.2-1