Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * parse_target.c
4 : * handle target lists
5 : *
6 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
7 : * Portions Copyright (c) 1994, Regents of the University of California
8 : *
9 : *
10 : * IDENTIFICATION
11 : * src/backend/parser/parse_target.c
12 : *
13 : *-------------------------------------------------------------------------
14 : */
15 : #include "postgres.h"
16 :
17 : #include "catalog/namespace.h"
18 : #include "catalog/pg_type.h"
19 : #include "commands/dbcommands.h"
20 : #include "funcapi.h"
21 : #include "miscadmin.h"
22 : #include "nodes/makefuncs.h"
23 : #include "nodes/nodeFuncs.h"
24 : #include "parser/parse_coerce.h"
25 : #include "parser/parse_expr.h"
26 : #include "parser/parse_relation.h"
27 : #include "parser/parse_target.h"
28 : #include "parser/parse_type.h"
29 : #include "parser/parsetree.h"
30 : #include "utils/builtins.h"
31 : #include "utils/lsyscache.h"
32 : #include "utils/rel.h"
33 :
34 : static void markTargetListOrigin(ParseState *pstate, TargetEntry *tle,
35 : Var *var, int levelsup);
36 : static Node *transformAssignmentSubscripts(ParseState *pstate,
37 : Node *basenode,
38 : const char *targetName,
39 : Oid targetTypeId,
40 : int32 targetTypMod,
41 : Oid targetCollation,
42 : List *subscripts,
43 : List *indirection,
44 : ListCell *next_indirection,
45 : Node *rhs,
46 : CoercionContext ccontext,
47 : int location);
48 : static List *ExpandColumnRefStar(ParseState *pstate, ColumnRef *cref,
49 : bool make_target_entry);
50 : static List *ExpandAllTables(ParseState *pstate, int location);
51 : static List *ExpandIndirectionStar(ParseState *pstate, A_Indirection *ind,
52 : bool make_target_entry, ParseExprKind exprKind);
53 : static List *ExpandSingleTable(ParseState *pstate, ParseNamespaceItem *nsitem,
54 : int sublevels_up, int location,
55 : bool make_target_entry);
56 : static List *ExpandRowReference(ParseState *pstate, Node *expr,
57 : bool make_target_entry);
58 : static int FigureColnameInternal(Node *node, char **name);
59 :
60 :
61 : /*
62 : * transformTargetEntry()
63 : * Transform any ordinary "expression-type" node into a targetlist entry.
64 : * This is exported so that parse_clause.c can generate targetlist entries
65 : * for ORDER/GROUP BY items that are not already in the targetlist.
66 : *
67 : * node the (untransformed) parse tree for the value expression.
68 : * expr the transformed expression, or NULL if caller didn't do it yet.
69 : * exprKind expression kind (EXPR_KIND_SELECT_TARGET, etc)
70 : * colname the column name to be assigned, or NULL if none yet set.
71 : * resjunk true if the target should be marked resjunk, ie, it is not
72 : * wanted in the final projected tuple.
73 : */
74 : TargetEntry *
75 1049074 : transformTargetEntry(ParseState *pstate,
76 : Node *node,
77 : Node *expr,
78 : ParseExprKind exprKind,
79 : char *colname,
80 : bool resjunk)
81 : {
82 : /* Transform the node if caller didn't do it already */
83 1049074 : if (expr == NULL)
84 : {
85 : /*
86 : * If it's a SetToDefault node and we should allow that, pass it
87 : * through unmodified. (transformExpr will throw the appropriate
88 : * error if we're disallowing it.)
89 : */
90 1024470 : if (exprKind == EXPR_KIND_UPDATE_SOURCE && IsA(node, SetToDefault))
91 172 : expr = node;
92 : else
93 1024298 : expr = transformExpr(pstate, node, exprKind);
94 : }
95 :
96 1044224 : if (colname == NULL && !resjunk)
97 : {
98 : /*
99 : * Generate a suitable column name for a column without any explicit
100 : * 'AS ColumnName' clause.
101 : */
102 796724 : colname = FigureColname(node);
103 : }
104 :
105 2088448 : return makeTargetEntry((Expr *) expr,
106 1044224 : (AttrNumber) pstate->p_next_resno++,
107 : colname,
108 : resjunk);
109 : }
110 :
111 :
112 : /*
113 : * transformTargetList()
114 : * Turns a list of ResTarget's into a list of TargetEntry's.
115 : *
116 : * This code acts mostly the same for SELECT, UPDATE, or RETURNING lists;
117 : * the main thing is to transform the given expressions (the "val" fields).
118 : * The exprKind parameter distinguishes these cases when necessary.
119 : */
120 : List *
121 478080 : transformTargetList(ParseState *pstate, List *targetlist,
122 : ParseExprKind exprKind)
123 : {
124 478080 : List *p_target = NIL;
125 : bool expand_star;
126 : ListCell *o_target;
127 :
128 : /* Shouldn't have any leftover multiassign items at start */
129 : Assert(pstate->p_multiassign_exprs == NIL);
130 :
131 : /* Expand "something.*" in SELECT and RETURNING, but not UPDATE */
132 478080 : expand_star = (exprKind != EXPR_KIND_UPDATE_SOURCE);
133 :
134 1551986 : foreach(o_target, targetlist)
135 : {
136 1078762 : ResTarget *res = (ResTarget *) lfirst(o_target);
137 :
138 : /*
139 : * Check for "something.*". Depending on the complexity of the
140 : * "something", the star could appear as the last field in ColumnRef,
141 : * or as the last indirection item in A_Indirection.
142 : */
143 1078762 : if (expand_star)
144 : {
145 1059028 : if (IsA(res->val, ColumnRef))
146 : {
147 568530 : ColumnRef *cref = (ColumnRef *) res->val;
148 :
149 568530 : if (IsA(llast(cref->fields), A_Star))
150 : {
151 : /* It is something.*, expand into multiple items */
152 52944 : p_target = list_concat(p_target,
153 52950 : ExpandColumnRefStar(pstate,
154 : cref,
155 : true));
156 52944 : continue;
157 : }
158 : }
159 490498 : else if (IsA(res->val, A_Indirection))
160 : {
161 3478 : A_Indirection *ind = (A_Indirection *) res->val;
162 :
163 3478 : if (IsA(llast(ind->indirection), A_Star))
164 : {
165 : /* It is something.*, expand into multiple items */
166 1342 : p_target = list_concat(p_target,
167 1342 : ExpandIndirectionStar(pstate,
168 : ind,
169 : true,
170 : exprKind));
171 1342 : continue;
172 : }
173 : }
174 : }
175 :
176 : /*
177 : * Not "something.*", or we want to treat that as a plain whole-row
178 : * variable, so transform as a single expression
179 : */
180 1019620 : p_target = lappend(p_target,
181 1024470 : transformTargetEntry(pstate,
182 : res->val,
183 : NULL,
184 : exprKind,
185 : res->name,
186 : false));
187 : }
188 :
189 : /*
190 : * If any multiassign resjunk items were created, attach them to the end
191 : * of the targetlist. This should only happen in an UPDATE tlist. We
192 : * don't need to worry about numbering of these items; transformUpdateStmt
193 : * will set their resnos.
194 : */
195 473224 : if (pstate->p_multiassign_exprs)
196 : {
197 : Assert(exprKind == EXPR_KIND_UPDATE_SOURCE);
198 138 : p_target = list_concat(p_target, pstate->p_multiassign_exprs);
199 138 : pstate->p_multiassign_exprs = NIL;
200 : }
201 :
202 473224 : return p_target;
203 : }
204 :
205 :
206 : /*
207 : * transformExpressionList()
208 : *
209 : * This is the identical transformation to transformTargetList, except that
210 : * the input list elements are bare expressions without ResTarget decoration,
211 : * and the output elements are likewise just expressions without TargetEntry
212 : * decoration. Also, we don't expect any multiassign constructs within the
213 : * list, so there's nothing to do for that. We use this for ROW() and
214 : * VALUES() constructs.
215 : *
216 : * exprKind is not enough to tell us whether to allow SetToDefault, so
217 : * an additional flag is needed for that.
218 : */
219 : List *
220 92042 : transformExpressionList(ParseState *pstate, List *exprlist,
221 : ParseExprKind exprKind, bool allowDefault)
222 : {
223 92042 : List *result = NIL;
224 : ListCell *lc;
225 :
226 274976 : foreach(lc, exprlist)
227 : {
228 182978 : Node *e = (Node *) lfirst(lc);
229 :
230 : /*
231 : * Check for "something.*". Depending on the complexity of the
232 : * "something", the star could appear as the last field in ColumnRef,
233 : * or as the last indirection item in A_Indirection.
234 : */
235 182978 : if (IsA(e, ColumnRef))
236 : {
237 8336 : ColumnRef *cref = (ColumnRef *) e;
238 :
239 8336 : if (IsA(llast(cref->fields), A_Star))
240 : {
241 : /* It is something.*, expand into multiple items */
242 312 : result = list_concat(result,
243 312 : ExpandColumnRefStar(pstate, cref,
244 : false));
245 312 : continue;
246 : }
247 : }
248 174642 : else if (IsA(e, A_Indirection))
249 : {
250 24 : A_Indirection *ind = (A_Indirection *) e;
251 :
252 24 : if (IsA(llast(ind->indirection), A_Star))
253 : {
254 : /* It is something.*, expand into multiple items */
255 0 : result = list_concat(result,
256 0 : ExpandIndirectionStar(pstate, ind,
257 : false, exprKind));
258 0 : continue;
259 : }
260 : }
261 :
262 : /*
263 : * Not "something.*", so transform as a single expression. If it's a
264 : * SetToDefault node and we should allow that, pass it through
265 : * unmodified. (transformExpr will throw the appropriate error if
266 : * we're disallowing it.)
267 : */
268 182666 : if (allowDefault && IsA(e, SetToDefault))
269 : /* do nothing */ ;
270 : else
271 181460 : e = transformExpr(pstate, e, exprKind);
272 :
273 182622 : result = lappend(result, e);
274 : }
275 :
276 91998 : return result;
277 : }
278 :
279 :
280 : /*
281 : * resolveTargetListUnknowns()
282 : * Convert any unknown-type targetlist entries to type TEXT.
283 : *
284 : * We do this after we've exhausted all other ways of identifying the output
285 : * column types of a query.
286 : */
287 : void
288 426484 : resolveTargetListUnknowns(ParseState *pstate, List *targetlist)
289 : {
290 : ListCell *l;
291 :
292 1541246 : foreach(l, targetlist)
293 : {
294 1114762 : TargetEntry *tle = (TargetEntry *) lfirst(l);
295 1114762 : Oid restype = exprType((Node *) tle->expr);
296 :
297 1114762 : if (restype == UNKNOWNOID)
298 : {
299 6988 : tle->expr = (Expr *) coerce_type(pstate, (Node *) tle->expr,
300 : restype, TEXTOID, -1,
301 : COERCION_IMPLICIT,
302 : COERCE_IMPLICIT_CAST,
303 : -1);
304 : }
305 : }
306 426484 : }
307 :
308 :
309 : /*
310 : * markTargetListOrigins()
311 : * Mark targetlist columns that are simple Vars with the source
312 : * table's OID and column number.
313 : *
314 : * Currently, this is done only for SELECT targetlists and RETURNING lists,
315 : * since we only need the info if we are going to send it to the frontend.
316 : */
317 : void
318 452310 : markTargetListOrigins(ParseState *pstate, List *targetlist)
319 : {
320 : ListCell *l;
321 :
322 1635480 : foreach(l, targetlist)
323 : {
324 1183170 : TargetEntry *tle = (TargetEntry *) lfirst(l);
325 :
326 1183170 : markTargetListOrigin(pstate, tle, (Var *) tle->expr, 0);
327 : }
328 452310 : }
329 :
330 : /*
331 : * markTargetListOrigin()
332 : * If 'var' is a Var of a plain relation, mark 'tle' with its origin
333 : *
334 : * levelsup is an extra offset to interpret the Var's varlevelsup correctly.
335 : *
336 : * Note that we do not drill down into views, but report the view as the
337 : * column owner. There's also no need to drill down into joins: if we see
338 : * a join alias Var, it must be a merged JOIN USING column (or possibly a
339 : * whole-row Var); that is not a direct reference to any plain table column,
340 : * so we don't report it.
341 : */
342 : static void
343 1183170 : markTargetListOrigin(ParseState *pstate, TargetEntry *tle,
344 : Var *var, int levelsup)
345 : {
346 : int netlevelsup;
347 : RangeTblEntry *rte;
348 : AttrNumber attnum;
349 :
350 1183170 : if (var == NULL || !IsA(var, Var))
351 494374 : return;
352 688796 : netlevelsup = var->varlevelsup + levelsup;
353 688796 : rte = GetRTEByRangeTablePosn(pstate, var->varno, netlevelsup);
354 688796 : attnum = var->varattno;
355 :
356 688796 : switch (rte->rtekind)
357 : {
358 572350 : case RTE_RELATION:
359 : /* It's a table or view, report it */
360 572350 : tle->resorigtbl = rte->relid;
361 572350 : tle->resorigcol = attnum;
362 572350 : break;
363 14908 : case RTE_SUBQUERY:
364 : /* Subselect-in-FROM: copy up from the subselect */
365 14908 : if (attnum != InvalidAttrNumber)
366 : {
367 14848 : TargetEntry *ste = get_tle_by_resno(rte->subquery->targetList,
368 : attnum);
369 :
370 14848 : if (ste == NULL || ste->resjunk)
371 0 : elog(ERROR, "subquery %s does not have attribute %d",
372 : rte->eref->aliasname, attnum);
373 14848 : tle->resorigtbl = ste->resorigtbl;
374 14848 : tle->resorigcol = ste->resorigcol;
375 : }
376 14908 : break;
377 93766 : case RTE_JOIN:
378 : case RTE_FUNCTION:
379 : case RTE_VALUES:
380 : case RTE_TABLEFUNC:
381 : case RTE_NAMEDTUPLESTORE:
382 : case RTE_RESULT:
383 : /* not a simple relation, leave it unmarked */
384 93766 : break;
385 7772 : case RTE_CTE:
386 :
387 : /*
388 : * CTE reference: copy up from the subquery, if possible. If the
389 : * RTE is a recursive self-reference then we can't do anything
390 : * because we haven't finished analyzing it yet. However, it's no
391 : * big loss because we must be down inside the recursive term of a
392 : * recursive CTE, and so any markings on the current targetlist
393 : * are not going to affect the results anyway.
394 : */
395 7772 : if (attnum != InvalidAttrNumber && !rte->self_reference)
396 : {
397 7268 : CommonTableExpr *cte = GetCTEForRTE(pstate, rte, netlevelsup);
398 : TargetEntry *ste;
399 7268 : List *tl = GetCTETargetList(cte);
400 7268 : int extra_cols = 0;
401 :
402 : /*
403 : * RTE for CTE will already have the search and cycle columns
404 : * added, but the subquery won't, so skip looking those up.
405 : */
406 7268 : if (cte->search_clause)
407 294 : extra_cols += 1;
408 7268 : if (cte->cycle_clause)
409 288 : extra_cols += 2;
410 7778 : if (extra_cols &&
411 510 : attnum > list_length(tl) &&
412 186 : attnum <= list_length(tl) + extra_cols)
413 186 : break;
414 :
415 7082 : ste = get_tle_by_resno(tl, attnum);
416 7082 : if (ste == NULL || ste->resjunk)
417 0 : elog(ERROR, "CTE %s does not have attribute %d",
418 : rte->eref->aliasname, attnum);
419 7082 : tle->resorigtbl = ste->resorigtbl;
420 7082 : tle->resorigcol = ste->resorigcol;
421 : }
422 7586 : break;
423 0 : case RTE_GROUP:
424 : /* We couldn't get here: the RTE_GROUP RTE has not been added */
425 0 : break;
426 : }
427 : }
428 :
429 :
430 : /*
431 : * transformAssignedExpr()
432 : * This is used in INSERT and UPDATE statements only. It prepares an
433 : * expression for assignment to a column of the target table.
434 : * This includes coercing the given value to the target column's type
435 : * (if necessary), and dealing with any subfield names or subscripts
436 : * attached to the target column itself. The input expression has
437 : * already been through transformExpr().
438 : *
439 : * pstate parse state
440 : * expr expression to be modified
441 : * exprKind indicates which type of statement we're dealing with
442 : * colname target column name (ie, name of attribute to be assigned to)
443 : * attrno target attribute number
444 : * indirection subscripts/field names for target column, if any
445 : * location error cursor position for the target column, or -1
446 : *
447 : * Returns the modified expression.
448 : *
449 : * Note: location points at the target column name (SET target or INSERT
450 : * column name list entry), and must therefore be -1 in an INSERT that
451 : * omits the column name list. So we should usually prefer to use
452 : * exprLocation(expr) for errors that can happen in a default INSERT.
453 : */
454 : Expr *
455 184544 : transformAssignedExpr(ParseState *pstate,
456 : Expr *expr,
457 : ParseExprKind exprKind,
458 : const char *colname,
459 : int attrno,
460 : List *indirection,
461 : int location)
462 : {
463 184544 : Relation rd = pstate->p_target_relation;
464 : Oid type_id; /* type of value provided */
465 : Oid attrtype; /* type of target column */
466 : int32 attrtypmod;
467 : Oid attrcollation; /* collation of target column */
468 : ParseExprKind sv_expr_kind;
469 :
470 : /*
471 : * Save and restore identity of expression type we're parsing. We must
472 : * set p_expr_kind here because we can parse subscripts without going
473 : * through transformExpr().
474 : */
475 : Assert(exprKind != EXPR_KIND_NONE);
476 184544 : sv_expr_kind = pstate->p_expr_kind;
477 184544 : pstate->p_expr_kind = exprKind;
478 :
479 : Assert(rd != NULL);
480 184544 : if (attrno <= 0)
481 0 : ereport(ERROR,
482 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
483 : errmsg("cannot assign to system column \"%s\"",
484 : colname),
485 : parser_errposition(pstate, location)));
486 184544 : attrtype = attnumTypeId(rd, attrno);
487 184544 : attrtypmod = TupleDescAttr(rd->rd_att, attrno - 1)->atttypmod;
488 184544 : attrcollation = TupleDescAttr(rd->rd_att, attrno - 1)->attcollation;
489 :
490 : /*
491 : * If the expression is a DEFAULT placeholder, insert the attribute's
492 : * type/typmod/collation into it so that exprType etc will report the
493 : * right things. (We expect that the eventually substituted default
494 : * expression will in fact have this type and typmod. The collation
495 : * likely doesn't matter, but let's set it correctly anyway.) Also,
496 : * reject trying to update a subfield or array element with DEFAULT, since
497 : * there can't be any default for portions of a column.
498 : */
499 184544 : if (expr && IsA(expr, SetToDefault))
500 : {
501 1348 : SetToDefault *def = (SetToDefault *) expr;
502 :
503 1348 : def->typeId = attrtype;
504 1348 : def->typeMod = attrtypmod;
505 1348 : def->collation = attrcollation;
506 1348 : if (indirection)
507 : {
508 24 : if (IsA(linitial(indirection), A_Indices))
509 12 : ereport(ERROR,
510 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
511 : errmsg("cannot set an array element to DEFAULT"),
512 : parser_errposition(pstate, location)));
513 : else
514 12 : ereport(ERROR,
515 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
516 : errmsg("cannot set a subfield to DEFAULT"),
517 : parser_errposition(pstate, location)));
518 : }
519 : }
520 :
521 : /* Now we can use exprType() safely. */
522 184520 : type_id = exprType((Node *) expr);
523 :
524 : /*
525 : * If there is indirection on the target column, prepare an array or
526 : * subfield assignment expression. This will generate a new column value
527 : * that the source value has been inserted into, which can then be placed
528 : * in the new tuple constructed by INSERT or UPDATE.
529 : */
530 184520 : if (indirection)
531 : {
532 : Node *colVar;
533 :
534 1842 : if (pstate->p_is_insert)
535 : {
536 : /*
537 : * The command is INSERT INTO table (col.something) ... so there
538 : * is not really a source value to work with. Insert a NULL
539 : * constant as the source value.
540 : */
541 1050 : colVar = (Node *) makeNullConst(attrtype, attrtypmod,
542 : attrcollation);
543 : }
544 : else
545 : {
546 : /*
547 : * Build a Var for the column to be updated.
548 : */
549 : Var *var;
550 :
551 792 : var = makeVar(pstate->p_target_nsitem->p_rtindex, attrno,
552 : attrtype, attrtypmod, attrcollation, 0);
553 792 : var->location = location;
554 :
555 792 : colVar = (Node *) var;
556 : }
557 :
558 : expr = (Expr *)
559 1842 : transformAssignmentIndirection(pstate,
560 : colVar,
561 : colname,
562 : false,
563 : attrtype,
564 : attrtypmod,
565 : attrcollation,
566 : indirection,
567 : list_head(indirection),
568 : (Node *) expr,
569 : COERCION_ASSIGNMENT,
570 : location);
571 : }
572 : else
573 : {
574 : /*
575 : * For normal non-qualified target column, do type checking and
576 : * coercion.
577 : */
578 182678 : Node *orig_expr = (Node *) expr;
579 :
580 : expr = (Expr *)
581 182678 : coerce_to_target_type(pstate,
582 : orig_expr, type_id,
583 : attrtype, attrtypmod,
584 : COERCION_ASSIGNMENT,
585 : COERCE_IMPLICIT_CAST,
586 : -1);
587 181658 : if (expr == NULL)
588 142 : ereport(ERROR,
589 : (errcode(ERRCODE_DATATYPE_MISMATCH),
590 : errmsg("column \"%s\" is of type %s"
591 : " but expression is of type %s",
592 : colname,
593 : format_type_be(attrtype),
594 : format_type_be(type_id)),
595 : errhint("You will need to rewrite or cast the expression."),
596 : parser_errposition(pstate, exprLocation(orig_expr))));
597 : }
598 :
599 183340 : pstate->p_expr_kind = sv_expr_kind;
600 :
601 183340 : return expr;
602 : }
603 :
604 :
605 : /*
606 : * updateTargetListEntry()
607 : * This is used in UPDATE statements (and ON CONFLICT DO UPDATE)
608 : * only. It prepares an UPDATE TargetEntry for assignment to a
609 : * column of the target table. This includes coercing the given
610 : * value to the target column's type (if necessary), and dealing with
611 : * any subfield names or subscripts attached to the target column
612 : * itself.
613 : *
614 : * pstate parse state
615 : * tle target list entry to be modified
616 : * colname target column name (ie, name of attribute to be assigned to)
617 : * attrno target attribute number
618 : * indirection subscripts/field names for target column, if any
619 : * location error cursor position (should point at column name), or -1
620 : */
621 : void
622 19662 : updateTargetListEntry(ParseState *pstate,
623 : TargetEntry *tle,
624 : char *colname,
625 : int attrno,
626 : List *indirection,
627 : int location)
628 : {
629 : /* Fix up expression as needed */
630 19662 : tle->expr = transformAssignedExpr(pstate,
631 : tle->expr,
632 : EXPR_KIND_UPDATE_TARGET,
633 : colname,
634 : attrno,
635 : indirection,
636 : location);
637 :
638 : /*
639 : * Set the resno to identify the target column --- the rewriter and
640 : * planner depend on this. We also set the resname to identify the target
641 : * column, but this is only for debugging purposes; it should not be
642 : * relied on. (In particular, it might be out of date in a stored rule.)
643 : */
644 19650 : tle->resno = (AttrNumber) attrno;
645 19650 : tle->resname = colname;
646 19650 : }
647 :
648 :
649 : /*
650 : * Process indirection (field selection or subscripting) of the target
651 : * column in INSERT/UPDATE/assignment. This routine recurses for multiple
652 : * levels of indirection --- but note that several adjacent A_Indices nodes
653 : * in the indirection list are treated as a single multidimensional subscript
654 : * operation.
655 : *
656 : * In the initial call, basenode is a Var for the target column in UPDATE,
657 : * or a null Const of the target's type in INSERT, or a Param for the target
658 : * variable in PL/pgSQL assignment. In recursive calls, basenode is NULL,
659 : * indicating that a substitute node should be consed up if needed.
660 : *
661 : * targetName is the name of the field or subfield we're assigning to, and
662 : * targetIsSubscripting is true if we're subscripting it. These are just for
663 : * error reporting.
664 : *
665 : * targetTypeId, targetTypMod, targetCollation indicate the datatype and
666 : * collation of the object to be assigned to (initially the target column,
667 : * later some subobject).
668 : *
669 : * indirection is the list of indirection nodes, and indirection_cell is the
670 : * start of the sublist remaining to process. When it's NULL, we're done
671 : * recursing and can just coerce and return the RHS.
672 : *
673 : * rhs is the already-transformed value to be assigned; note it has not been
674 : * coerced to any particular type.
675 : *
676 : * ccontext is the coercion level to use while coercing the rhs. For
677 : * normal statements it'll be COERCION_ASSIGNMENT, but PL/pgSQL uses
678 : * a special value.
679 : *
680 : * location is the cursor error position for any errors. (Note: this points
681 : * to the head of the target clause, eg "foo" in "foo.bar[baz]". Later we
682 : * might want to decorate indirection cells with their own location info,
683 : * in which case the location argument could probably be dropped.)
684 : */
685 : Node *
686 4564 : transformAssignmentIndirection(ParseState *pstate,
687 : Node *basenode,
688 : const char *targetName,
689 : bool targetIsSubscripting,
690 : Oid targetTypeId,
691 : int32 targetTypMod,
692 : Oid targetCollation,
693 : List *indirection,
694 : ListCell *indirection_cell,
695 : Node *rhs,
696 : CoercionContext ccontext,
697 : int location)
698 : {
699 : Node *result;
700 4564 : List *subscripts = NIL;
701 : ListCell *i;
702 :
703 4564 : if (indirection_cell && !basenode)
704 : {
705 : /*
706 : * Set up a substitution. We abuse CaseTestExpr for this. It's safe
707 : * to do so because the only nodes that will be above the CaseTestExpr
708 : * in the finished expression will be FieldStore and SubscriptingRef
709 : * nodes. (There could be other stuff in the tree, but it will be
710 : * within other child fields of those node types.)
711 : */
712 674 : CaseTestExpr *ctest = makeNode(CaseTestExpr);
713 :
714 674 : ctest->typeId = targetTypeId;
715 674 : ctest->typeMod = targetTypMod;
716 674 : ctest->collation = targetCollation;
717 674 : basenode = (Node *) ctest;
718 : }
719 :
720 : /*
721 : * We have to split any field-selection operations apart from
722 : * subscripting. Adjacent A_Indices nodes have to be treated as a single
723 : * multidimensional subscript operation.
724 : */
725 6652 : for_each_cell(i, indirection, indirection_cell)
726 : {
727 3276 : Node *n = lfirst(i);
728 :
729 3276 : if (IsA(n, A_Indices))
730 2088 : subscripts = lappend(subscripts, n);
731 1188 : else if (IsA(n, A_Star))
732 : {
733 0 : ereport(ERROR,
734 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
735 : errmsg("row expansion via \"*\" is not supported here"),
736 : parser_errposition(pstate, location)));
737 : }
738 : else
739 : {
740 : FieldStore *fstore;
741 : Oid baseTypeId;
742 : int32 baseTypeMod;
743 : Oid typrelid;
744 : AttrNumber attnum;
745 : Oid fieldTypeId;
746 : int32 fieldTypMod;
747 : Oid fieldCollation;
748 :
749 : Assert(IsA(n, String));
750 :
751 : /* process subscripts before this field selection */
752 1188 : if (subscripts)
753 : {
754 : /* recurse, and then return because we're done */
755 344 : return transformAssignmentSubscripts(pstate,
756 : basenode,
757 : targetName,
758 : targetTypeId,
759 : targetTypMod,
760 : targetCollation,
761 : subscripts,
762 : indirection,
763 : i,
764 : rhs,
765 : ccontext,
766 : location);
767 : }
768 :
769 : /* No subscripts, so can process field selection here */
770 :
771 : /*
772 : * Look up the composite type, accounting for possibility that
773 : * what we are given is a domain over composite.
774 : */
775 844 : baseTypeMod = targetTypMod;
776 844 : baseTypeId = getBaseTypeAndTypmod(targetTypeId, &baseTypeMod);
777 :
778 844 : typrelid = typeidTypeRelid(baseTypeId);
779 844 : if (!typrelid)
780 2 : ereport(ERROR,
781 : (errcode(ERRCODE_DATATYPE_MISMATCH),
782 : errmsg("cannot assign to field \"%s\" of column \"%s\" because its type %s is not a composite type",
783 : strVal(n), targetName,
784 : format_type_be(targetTypeId)),
785 : parser_errposition(pstate, location)));
786 :
787 842 : attnum = get_attnum(typrelid, strVal(n));
788 842 : if (attnum == InvalidAttrNumber)
789 4 : ereport(ERROR,
790 : (errcode(ERRCODE_UNDEFINED_COLUMN),
791 : errmsg("cannot assign to field \"%s\" of column \"%s\" because there is no such column in data type %s",
792 : strVal(n), targetName,
793 : format_type_be(targetTypeId)),
794 : parser_errposition(pstate, location)));
795 838 : if (attnum < 0)
796 0 : ereport(ERROR,
797 : (errcode(ERRCODE_UNDEFINED_COLUMN),
798 : errmsg("cannot assign to system column \"%s\"",
799 : strVal(n)),
800 : parser_errposition(pstate, location)));
801 :
802 838 : get_atttypetypmodcoll(typrelid, attnum,
803 : &fieldTypeId, &fieldTypMod, &fieldCollation);
804 :
805 : /* recurse to create appropriate RHS for field assign */
806 838 : rhs = transformAssignmentIndirection(pstate,
807 : NULL,
808 838 : strVal(n),
809 : false,
810 : fieldTypeId,
811 : fieldTypMod,
812 : fieldCollation,
813 : indirection,
814 : lnext(indirection, i),
815 : rhs,
816 : ccontext,
817 : location);
818 :
819 : /* and build a FieldStore node */
820 832 : fstore = makeNode(FieldStore);
821 832 : fstore->arg = (Expr *) basenode;
822 832 : fstore->newvals = list_make1(rhs);
823 832 : fstore->fieldnums = list_make1_int(attnum);
824 832 : fstore->resulttype = baseTypeId;
825 :
826 : /*
827 : * If target is a domain, apply constraints. Notice that this
828 : * isn't totally right: the expression tree we build would check
829 : * the domain's constraints on a composite value with only this
830 : * one field populated or updated, possibly leading to an unwanted
831 : * failure. The rewriter will merge together any subfield
832 : * assignments to the same table column, resulting in the domain's
833 : * constraints being checked only once after we've assigned to all
834 : * the fields that the INSERT or UPDATE means to.
835 : */
836 832 : if (baseTypeId != targetTypeId)
837 432 : return coerce_to_domain((Node *) fstore,
838 : baseTypeId, baseTypeMod,
839 : targetTypeId,
840 : COERCION_IMPLICIT,
841 : COERCE_IMPLICIT_CAST,
842 : location,
843 : false);
844 :
845 400 : return (Node *) fstore;
846 : }
847 : }
848 :
849 : /* process trailing subscripts, if any */
850 3376 : if (subscripts)
851 : {
852 : /* recurse, and then return because we're done */
853 1436 : return transformAssignmentSubscripts(pstate,
854 : basenode,
855 : targetName,
856 : targetTypeId,
857 : targetTypMod,
858 : targetCollation,
859 : subscripts,
860 : indirection,
861 : NULL,
862 : rhs,
863 : ccontext,
864 : location);
865 : }
866 :
867 : /* base case: just coerce RHS to match target type ID */
868 :
869 1940 : result = coerce_to_target_type(pstate,
870 : rhs, exprType(rhs),
871 : targetTypeId, targetTypMod,
872 : ccontext,
873 : COERCE_IMPLICIT_CAST,
874 : -1);
875 1940 : if (result == NULL)
876 : {
877 18 : if (targetIsSubscripting)
878 12 : ereport(ERROR,
879 : (errcode(ERRCODE_DATATYPE_MISMATCH),
880 : errmsg("subscripted assignment to \"%s\" requires type %s"
881 : " but expression is of type %s",
882 : targetName,
883 : format_type_be(targetTypeId),
884 : format_type_be(exprType(rhs))),
885 : errhint("You will need to rewrite or cast the expression."),
886 : parser_errposition(pstate, location)));
887 : else
888 6 : ereport(ERROR,
889 : (errcode(ERRCODE_DATATYPE_MISMATCH),
890 : errmsg("subfield \"%s\" is of type %s"
891 : " but expression is of type %s",
892 : targetName,
893 : format_type_be(targetTypeId),
894 : format_type_be(exprType(rhs))),
895 : errhint("You will need to rewrite or cast the expression."),
896 : parser_errposition(pstate, location)));
897 : }
898 :
899 1922 : return result;
900 : }
901 :
902 : /*
903 : * helper for transformAssignmentIndirection: process container assignment
904 : */
905 : static Node *
906 1780 : transformAssignmentSubscripts(ParseState *pstate,
907 : Node *basenode,
908 : const char *targetName,
909 : Oid targetTypeId,
910 : int32 targetTypMod,
911 : Oid targetCollation,
912 : List *subscripts,
913 : List *indirection,
914 : ListCell *next_indirection,
915 : Node *rhs,
916 : CoercionContext ccontext,
917 : int location)
918 : {
919 : Node *result;
920 : SubscriptingRef *sbsref;
921 : Oid containerType;
922 : int32 containerTypMod;
923 : Oid typeNeeded;
924 : int32 typmodNeeded;
925 : Oid collationNeeded;
926 :
927 : Assert(subscripts != NIL);
928 :
929 : /* Identify the actual container type involved */
930 1780 : containerType = targetTypeId;
931 1780 : containerTypMod = targetTypMod;
932 1780 : transformContainerType(&containerType, &containerTypMod);
933 :
934 : /* Process subscripts and identify required type for RHS */
935 1780 : sbsref = transformContainerSubscripts(pstate,
936 : basenode,
937 : containerType,
938 : containerTypMod,
939 : subscripts,
940 : true);
941 :
942 1776 : typeNeeded = sbsref->refrestype;
943 1776 : typmodNeeded = sbsref->reftypmod;
944 :
945 : /*
946 : * Container normally has same collation as its elements, but there's an
947 : * exception: we might be subscripting a domain over a container type. In
948 : * that case use collation of the base type. (This is shaky for arbitrary
949 : * subscripting semantics, but it doesn't matter all that much since we
950 : * only use this to label the collation of a possible CaseTestExpr.)
951 : */
952 1776 : if (containerType == targetTypeId)
953 1474 : collationNeeded = targetCollation;
954 : else
955 302 : collationNeeded = get_typcollation(containerType);
956 :
957 : /* recurse to create appropriate RHS for container assign */
958 1776 : rhs = transformAssignmentIndirection(pstate,
959 : NULL,
960 : targetName,
961 : true,
962 : typeNeeded,
963 : typmodNeeded,
964 : collationNeeded,
965 : indirection,
966 : next_indirection,
967 : rhs,
968 : ccontext,
969 : location);
970 :
971 : /*
972 : * Insert the already-properly-coerced RHS into the SubscriptingRef. Then
973 : * set refrestype and reftypmod back to the container type's values.
974 : */
975 1762 : sbsref->refassgnexpr = (Expr *) rhs;
976 1762 : sbsref->refrestype = containerType;
977 1762 : sbsref->reftypmod = containerTypMod;
978 :
979 1762 : result = (Node *) sbsref;
980 :
981 : /*
982 : * If target was a domain over container, need to coerce up to the domain.
983 : * As in transformAssignmentIndirection, this coercion is premature if the
984 : * query assigns to multiple elements of the container; but we'll fix that
985 : * during query rewrite.
986 : */
987 1762 : if (containerType != targetTypeId)
988 : {
989 302 : Oid resulttype = exprType(result);
990 :
991 302 : result = coerce_to_target_type(pstate,
992 : result, resulttype,
993 : targetTypeId, targetTypMod,
994 : ccontext,
995 : COERCE_IMPLICIT_CAST,
996 : -1);
997 : /* can fail if we had int2vector/oidvector, but not for true domains */
998 302 : if (result == NULL)
999 0 : ereport(ERROR,
1000 : (errcode(ERRCODE_CANNOT_COERCE),
1001 : errmsg("cannot cast type %s to %s",
1002 : format_type_be(resulttype),
1003 : format_type_be(targetTypeId)),
1004 : parser_errposition(pstate, location)));
1005 : }
1006 :
1007 1762 : return result;
1008 : }
1009 :
1010 :
1011 : /*
1012 : * checkInsertTargets -
1013 : * generate a list of INSERT column targets if not supplied, or
1014 : * test supplied column names to make sure they are in target table.
1015 : * Also return an integer list of the columns' attribute numbers.
1016 : */
1017 : List *
1018 73806 : checkInsertTargets(ParseState *pstate, List *cols, List **attrnos)
1019 : {
1020 73806 : *attrnos = NIL;
1021 :
1022 73806 : if (cols == NIL)
1023 : {
1024 : /*
1025 : * Generate default column list for INSERT.
1026 : */
1027 59648 : int numcol = RelationGetNumberOfAttributes(pstate->p_target_relation);
1028 :
1029 : int i;
1030 :
1031 171070 : for (i = 0; i < numcol; i++)
1032 : {
1033 : ResTarget *col;
1034 : Form_pg_attribute attr;
1035 :
1036 111422 : attr = TupleDescAttr(pstate->p_target_relation->rd_att, i);
1037 :
1038 111422 : if (attr->attisdropped)
1039 276 : continue;
1040 :
1041 111146 : col = makeNode(ResTarget);
1042 111146 : col->name = pstrdup(NameStr(attr->attname));
1043 111146 : col->indirection = NIL;
1044 111146 : col->val = NULL;
1045 111146 : col->location = -1;
1046 111146 : cols = lappend(cols, col);
1047 111146 : *attrnos = lappend_int(*attrnos, i + 1);
1048 : }
1049 : }
1050 : else
1051 : {
1052 : /*
1053 : * Do initial validation of user-supplied INSERT column list.
1054 : */
1055 14158 : Bitmapset *wholecols = NULL;
1056 14158 : Bitmapset *partialcols = NULL;
1057 : ListCell *tl;
1058 :
1059 43776 : foreach(tl, cols)
1060 : {
1061 29666 : ResTarget *col = (ResTarget *) lfirst(tl);
1062 29666 : char *name = col->name;
1063 : int attrno;
1064 :
1065 : /* Lookup column name, ereport on failure */
1066 29666 : attrno = attnameAttNum(pstate->p_target_relation, name, false);
1067 29666 : if (attrno == InvalidAttrNumber)
1068 48 : ereport(ERROR,
1069 : (errcode(ERRCODE_UNDEFINED_COLUMN),
1070 : errmsg("column \"%s\" of relation \"%s\" does not exist",
1071 : name,
1072 : RelationGetRelationName(pstate->p_target_relation)),
1073 : parser_errposition(pstate, col->location)));
1074 :
1075 : /*
1076 : * Check for duplicates, but only of whole columns --- we allow
1077 : * INSERT INTO foo (col.subcol1, col.subcol2)
1078 : */
1079 29618 : if (col->indirection == NIL)
1080 : {
1081 : /* whole column; must not have any other assignment */
1082 57784 : if (bms_is_member(attrno, wholecols) ||
1083 28892 : bms_is_member(attrno, partialcols))
1084 0 : ereport(ERROR,
1085 : (errcode(ERRCODE_DUPLICATE_COLUMN),
1086 : errmsg("column \"%s\" specified more than once",
1087 : name),
1088 : parser_errposition(pstate, col->location)));
1089 28892 : wholecols = bms_add_member(wholecols, attrno);
1090 : }
1091 : else
1092 : {
1093 : /* partial column; must not have any whole assignment */
1094 726 : if (bms_is_member(attrno, wholecols))
1095 0 : ereport(ERROR,
1096 : (errcode(ERRCODE_DUPLICATE_COLUMN),
1097 : errmsg("column \"%s\" specified more than once",
1098 : name),
1099 : parser_errposition(pstate, col->location)));
1100 726 : partialcols = bms_add_member(partialcols, attrno);
1101 : }
1102 :
1103 29618 : *attrnos = lappend_int(*attrnos, attrno);
1104 : }
1105 : }
1106 :
1107 73758 : return cols;
1108 : }
1109 :
1110 : /*
1111 : * ExpandColumnRefStar()
1112 : * Transforms foo.* into a list of expressions or targetlist entries.
1113 : *
1114 : * This handles the case where '*' appears as the last or only item in a
1115 : * ColumnRef. The code is shared between the case of foo.* at the top level
1116 : * in a SELECT target list (where we want TargetEntry nodes in the result)
1117 : * and foo.* in a ROW() or VALUES() construct (where we want just bare
1118 : * expressions).
1119 : *
1120 : * The referenced columns are marked as requiring SELECT access.
1121 : */
1122 : static List *
1123 53262 : ExpandColumnRefStar(ParseState *pstate, ColumnRef *cref,
1124 : bool make_target_entry)
1125 : {
1126 53262 : List *fields = cref->fields;
1127 53262 : int numnames = list_length(fields);
1128 :
1129 53262 : if (numnames == 1)
1130 : {
1131 : /*
1132 : * Target item is a bare '*', expand all tables
1133 : *
1134 : * (e.g., SELECT * FROM emp, dept)
1135 : *
1136 : * Since the grammar only accepts bare '*' at top level of SELECT, we
1137 : * need not handle the make_target_entry==false case here.
1138 : */
1139 : Assert(make_target_entry);
1140 49632 : return ExpandAllTables(pstate, cref->location);
1141 : }
1142 : else
1143 : {
1144 : /*
1145 : * Target item is relation.*, expand that table
1146 : *
1147 : * (e.g., SELECT emp.*, dname FROM emp, dept)
1148 : *
1149 : * Note: this code is a lot like transformColumnRef; it's tempting to
1150 : * call that instead and then replace the resulting whole-row Var with
1151 : * a list of Vars. However, that would leave us with the relation's
1152 : * selectedCols bitmap showing the whole row as needing select
1153 : * permission, as well as the individual columns. That would be
1154 : * incorrect (since columns added later shouldn't need select
1155 : * permissions). We could try to remove the whole-row permission bit
1156 : * after the fact, but duplicating code is less messy.
1157 : */
1158 3630 : char *nspname = NULL;
1159 3630 : char *relname = NULL;
1160 3630 : ParseNamespaceItem *nsitem = NULL;
1161 : int levels_up;
1162 : enum
1163 : {
1164 : CRSERR_NO_RTE,
1165 : CRSERR_WRONG_DB,
1166 : CRSERR_TOO_MANY
1167 3630 : } crserr = CRSERR_NO_RTE;
1168 :
1169 : /*
1170 : * Give the PreParseColumnRefHook, if any, first shot. If it returns
1171 : * non-null then we should use that expression.
1172 : */
1173 3630 : if (pstate->p_pre_columnref_hook != NULL)
1174 : {
1175 : Node *node;
1176 :
1177 58 : node = pstate->p_pre_columnref_hook(pstate, cref);
1178 58 : if (node != NULL)
1179 0 : return ExpandRowReference(pstate, node, make_target_entry);
1180 : }
1181 :
1182 3630 : switch (numnames)
1183 : {
1184 3630 : case 2:
1185 3630 : relname = strVal(linitial(fields));
1186 3630 : nsitem = refnameNamespaceItem(pstate, nspname, relname,
1187 : cref->location,
1188 : &levels_up);
1189 3630 : break;
1190 0 : case 3:
1191 0 : nspname = strVal(linitial(fields));
1192 0 : relname = strVal(lsecond(fields));
1193 0 : nsitem = refnameNamespaceItem(pstate, nspname, relname,
1194 : cref->location,
1195 : &levels_up);
1196 0 : break;
1197 0 : case 4:
1198 : {
1199 0 : char *catname = strVal(linitial(fields));
1200 :
1201 : /*
1202 : * We check the catalog name and then ignore it.
1203 : */
1204 0 : if (strcmp(catname, get_database_name(MyDatabaseId)) != 0)
1205 : {
1206 0 : crserr = CRSERR_WRONG_DB;
1207 0 : break;
1208 : }
1209 0 : nspname = strVal(lsecond(fields));
1210 0 : relname = strVal(lthird(fields));
1211 0 : nsitem = refnameNamespaceItem(pstate, nspname, relname,
1212 : cref->location,
1213 : &levels_up);
1214 0 : break;
1215 : }
1216 0 : default:
1217 0 : crserr = CRSERR_TOO_MANY;
1218 0 : break;
1219 : }
1220 :
1221 : /*
1222 : * Now give the PostParseColumnRefHook, if any, a chance. We cheat a
1223 : * bit by passing the RangeTblEntry, not a Var, as the planned
1224 : * translation. (A single Var wouldn't be strictly correct anyway.
1225 : * This convention allows hooks that really care to know what is
1226 : * happening. It might be better to pass the nsitem, but we'd have to
1227 : * promote that struct to a full-fledged Node type so that callees
1228 : * could identify its type.)
1229 : */
1230 3630 : if (pstate->p_post_columnref_hook != NULL)
1231 : {
1232 : Node *node;
1233 :
1234 184 : node = pstate->p_post_columnref_hook(pstate, cref,
1235 : (Node *) (nsitem ? nsitem->p_rte : NULL));
1236 184 : if (node != NULL)
1237 : {
1238 94 : if (nsitem != NULL)
1239 0 : ereport(ERROR,
1240 : (errcode(ERRCODE_AMBIGUOUS_COLUMN),
1241 : errmsg("column reference \"%s\" is ambiguous",
1242 : NameListToString(cref->fields)),
1243 : parser_errposition(pstate, cref->location)));
1244 94 : return ExpandRowReference(pstate, node, make_target_entry);
1245 : }
1246 : }
1247 :
1248 : /*
1249 : * Throw error if no translation found.
1250 : */
1251 3536 : if (nsitem == NULL)
1252 : {
1253 6 : switch (crserr)
1254 : {
1255 6 : case CRSERR_NO_RTE:
1256 6 : errorMissingRTE(pstate, makeRangeVar(nspname, relname,
1257 : cref->location));
1258 : break;
1259 0 : case CRSERR_WRONG_DB:
1260 0 : ereport(ERROR,
1261 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
1262 : errmsg("cross-database references are not implemented: %s",
1263 : NameListToString(cref->fields)),
1264 : parser_errposition(pstate, cref->location)));
1265 : break;
1266 0 : case CRSERR_TOO_MANY:
1267 0 : ereport(ERROR,
1268 : (errcode(ERRCODE_SYNTAX_ERROR),
1269 : errmsg("improper qualified name (too many dotted names): %s",
1270 : NameListToString(cref->fields)),
1271 : parser_errposition(pstate, cref->location)));
1272 : break;
1273 : }
1274 3530 : }
1275 :
1276 : /*
1277 : * OK, expand the nsitem into fields.
1278 : */
1279 3530 : return ExpandSingleTable(pstate, nsitem, levels_up, cref->location,
1280 : make_target_entry);
1281 : }
1282 : }
1283 :
1284 : /*
1285 : * ExpandAllTables()
1286 : * Transforms '*' (in the target list) into a list of targetlist entries.
1287 : *
1288 : * tlist entries are generated for each relation visible for unqualified
1289 : * column name access. We do not consider qualified-name-only entries because
1290 : * that would include input tables of aliasless JOINs, NEW/OLD pseudo-entries,
1291 : * etc.
1292 : *
1293 : * The referenced relations/columns are marked as requiring SELECT access.
1294 : */
1295 : static List *
1296 49632 : ExpandAllTables(ParseState *pstate, int location)
1297 : {
1298 49632 : List *target = NIL;
1299 49632 : bool found_table = false;
1300 : ListCell *l;
1301 :
1302 108748 : foreach(l, pstate->p_namespace)
1303 : {
1304 59116 : ParseNamespaceItem *nsitem = (ParseNamespaceItem *) lfirst(l);
1305 :
1306 : /* Ignore table-only items */
1307 59116 : if (!nsitem->p_cols_visible)
1308 8292 : continue;
1309 : /* Should not have any lateral-only items when parsing targetlist */
1310 : Assert(!nsitem->p_lateral_only);
1311 : /* Remember we found a p_cols_visible item */
1312 50824 : found_table = true;
1313 :
1314 50824 : target = list_concat(target,
1315 50824 : expandNSItemAttrs(pstate,
1316 : nsitem,
1317 : 0,
1318 : true,
1319 : location));
1320 : }
1321 :
1322 : /*
1323 : * Check for "SELECT *;". We do it this way, rather than checking for
1324 : * target == NIL, because we want to allow SELECT * FROM a zero_column
1325 : * table.
1326 : */
1327 49632 : if (!found_table)
1328 0 : ereport(ERROR,
1329 : (errcode(ERRCODE_SYNTAX_ERROR),
1330 : errmsg("SELECT * with no tables specified is not valid"),
1331 : parser_errposition(pstate, location)));
1332 :
1333 49632 : return target;
1334 : }
1335 :
1336 : /*
1337 : * ExpandIndirectionStar()
1338 : * Transforms foo.* into a list of expressions or targetlist entries.
1339 : *
1340 : * This handles the case where '*' appears as the last item in A_Indirection.
1341 : * The code is shared between the case of foo.* at the top level in a SELECT
1342 : * target list (where we want TargetEntry nodes in the result) and foo.* in
1343 : * a ROW() or VALUES() construct (where we want just bare expressions).
1344 : * For robustness, we use a separate "make_target_entry" flag to control
1345 : * this rather than relying on exprKind.
1346 : */
1347 : static List *
1348 1342 : ExpandIndirectionStar(ParseState *pstate, A_Indirection *ind,
1349 : bool make_target_entry, ParseExprKind exprKind)
1350 : {
1351 : Node *expr;
1352 :
1353 : /* Strip off the '*' to create a reference to the rowtype object */
1354 1342 : ind = copyObject(ind);
1355 1342 : ind->indirection = list_truncate(ind->indirection,
1356 1342 : list_length(ind->indirection) - 1);
1357 :
1358 : /* And transform that */
1359 1342 : expr = transformExpr(pstate, (Node *) ind, exprKind);
1360 :
1361 : /* Expand the rowtype expression into individual fields */
1362 1342 : return ExpandRowReference(pstate, expr, make_target_entry);
1363 : }
1364 :
1365 : /*
1366 : * ExpandSingleTable()
1367 : * Transforms foo.* into a list of expressions or targetlist entries.
1368 : *
1369 : * This handles the case where foo has been determined to be a simple
1370 : * reference to an RTE, so we can just generate Vars for the expressions.
1371 : *
1372 : * The referenced columns are marked as requiring SELECT access.
1373 : */
1374 : static List *
1375 3530 : ExpandSingleTable(ParseState *pstate, ParseNamespaceItem *nsitem,
1376 : int sublevels_up, int location, bool make_target_entry)
1377 : {
1378 3530 : if (make_target_entry)
1379 : {
1380 : /* expandNSItemAttrs handles permissions marking */
1381 3310 : return expandNSItemAttrs(pstate, nsitem, sublevels_up, true, location);
1382 : }
1383 : else
1384 : {
1385 220 : RangeTblEntry *rte = nsitem->p_rte;
1386 220 : RTEPermissionInfo *perminfo = nsitem->p_perminfo;
1387 : List *vars;
1388 : ListCell *l;
1389 :
1390 220 : vars = expandNSItemVars(pstate, nsitem, sublevels_up, location, NULL);
1391 :
1392 : /*
1393 : * Require read access to the table. This is normally redundant with
1394 : * the markVarForSelectPriv calls below, but not if the table has zero
1395 : * columns. We need not do anything if the nsitem is for a join: its
1396 : * component tables will have been marked ACL_SELECT when they were
1397 : * added to the rangetable. (This step changes things only for the
1398 : * target relation of UPDATE/DELETE, which cannot be under a join.)
1399 : */
1400 220 : if (rte->rtekind == RTE_RELATION)
1401 : {
1402 : Assert(perminfo != NULL);
1403 128 : perminfo->requiredPerms |= ACL_SELECT;
1404 : }
1405 :
1406 : /* Require read access to each column */
1407 920 : foreach(l, vars)
1408 : {
1409 700 : Var *var = (Var *) lfirst(l);
1410 :
1411 700 : markVarForSelectPriv(pstate, var);
1412 : }
1413 :
1414 220 : return vars;
1415 : }
1416 : }
1417 :
1418 : /*
1419 : * ExpandRowReference()
1420 : * Transforms foo.* into a list of expressions or targetlist entries.
1421 : *
1422 : * This handles the case where foo is an arbitrary expression of composite
1423 : * type.
1424 : */
1425 : static List *
1426 1436 : ExpandRowReference(ParseState *pstate, Node *expr,
1427 : bool make_target_entry)
1428 : {
1429 1436 : List *result = NIL;
1430 : TupleDesc tupleDesc;
1431 : int numAttrs;
1432 : int i;
1433 :
1434 : /*
1435 : * If the rowtype expression is a whole-row Var, we can expand the fields
1436 : * as simple Vars. Note: if the RTE is a relation, this case leaves us
1437 : * with its RTEPermissionInfo's selectedCols bitmap showing the whole row
1438 : * as needing select permission, as well as the individual columns.
1439 : * However, we can only get here for weird notations like (table.*).*, so
1440 : * it's not worth trying to clean up --- arguably, the permissions marking
1441 : * is correct anyway for such cases.
1442 : */
1443 1436 : if (IsA(expr, Var) &&
1444 78 : ((Var *) expr)->varattno == InvalidAttrNumber)
1445 : {
1446 0 : Var *var = (Var *) expr;
1447 : ParseNamespaceItem *nsitem;
1448 :
1449 0 : nsitem = GetNSItemByRangeTablePosn(pstate, var->varno, var->varlevelsup);
1450 0 : return ExpandSingleTable(pstate, nsitem, var->varlevelsup, var->location, make_target_entry);
1451 : }
1452 :
1453 : /*
1454 : * Otherwise we have to do it the hard way. Our current implementation is
1455 : * to generate multiple copies of the expression and do FieldSelects.
1456 : * (This can be pretty inefficient if the expression involves nontrivial
1457 : * computation :-(.)
1458 : *
1459 : * Verify it's a composite type, and get the tupdesc.
1460 : * get_expr_result_tupdesc() handles this conveniently.
1461 : *
1462 : * If it's a Var of type RECORD, we have to work even harder: we have to
1463 : * find what the Var refers to, and pass that to get_expr_result_tupdesc.
1464 : * That task is handled by expandRecordVariable().
1465 : */
1466 1436 : if (IsA(expr, Var) &&
1467 78 : ((Var *) expr)->vartype == RECORDOID)
1468 6 : tupleDesc = expandRecordVariable(pstate, (Var *) expr, 0);
1469 : else
1470 1430 : tupleDesc = get_expr_result_tupdesc(expr, false);
1471 : Assert(tupleDesc);
1472 :
1473 : /* Generate a list of references to the individual fields */
1474 1436 : numAttrs = tupleDesc->natts;
1475 6892 : for (i = 0; i < numAttrs; i++)
1476 : {
1477 5456 : Form_pg_attribute att = TupleDescAttr(tupleDesc, i);
1478 : FieldSelect *fselect;
1479 :
1480 5456 : if (att->attisdropped)
1481 8 : continue;
1482 :
1483 5448 : fselect = makeNode(FieldSelect);
1484 5448 : fselect->arg = (Expr *) copyObject(expr);
1485 5448 : fselect->fieldnum = i + 1;
1486 5448 : fselect->resulttype = att->atttypid;
1487 5448 : fselect->resulttypmod = att->atttypmod;
1488 : /* save attribute's collation for parse_collate.c */
1489 5448 : fselect->resultcollid = att->attcollation;
1490 :
1491 5448 : if (make_target_entry)
1492 : {
1493 : /* add TargetEntry decoration */
1494 : TargetEntry *te;
1495 :
1496 10416 : te = makeTargetEntry((Expr *) fselect,
1497 5208 : (AttrNumber) pstate->p_next_resno++,
1498 5208 : pstrdup(NameStr(att->attname)),
1499 : false);
1500 5208 : result = lappend(result, te);
1501 : }
1502 : else
1503 240 : result = lappend(result, fselect);
1504 : }
1505 :
1506 1436 : return result;
1507 : }
1508 :
1509 : /*
1510 : * expandRecordVariable
1511 : * Get the tuple descriptor for a Var of type RECORD, if possible.
1512 : *
1513 : * Since no actual table or view column is allowed to have type RECORD, such
1514 : * a Var must refer to a JOIN or FUNCTION RTE or to a subquery output. We
1515 : * drill down to find the ultimate defining expression and attempt to infer
1516 : * the tupdesc from it. We ereport if we can't determine the tupdesc.
1517 : *
1518 : * levelsup is an extra offset to interpret the Var's varlevelsup correctly
1519 : * when recursing. Outside callers should pass zero.
1520 : */
1521 : TupleDesc
1522 1832 : expandRecordVariable(ParseState *pstate, Var *var, int levelsup)
1523 : {
1524 : TupleDesc tupleDesc;
1525 : int netlevelsup;
1526 : RangeTblEntry *rte;
1527 : AttrNumber attnum;
1528 : Node *expr;
1529 :
1530 : /* Check my caller didn't mess up */
1531 : Assert(IsA(var, Var));
1532 : Assert(var->vartype == RECORDOID);
1533 :
1534 : /*
1535 : * Note: it's tempting to use GetNSItemByRangeTablePosn here so that we
1536 : * can use expandNSItemVars instead of expandRTE; but that does not work
1537 : * for some of the recursion cases below, where we have consed up a
1538 : * ParseState that lacks p_namespace data.
1539 : */
1540 1832 : netlevelsup = var->varlevelsup + levelsup;
1541 1832 : rte = GetRTEByRangeTablePosn(pstate, var->varno, netlevelsup);
1542 1832 : attnum = var->varattno;
1543 :
1544 1832 : if (attnum == InvalidAttrNumber)
1545 : {
1546 : /* Whole-row reference to an RTE, so expand the known fields */
1547 : List *names,
1548 : *vars;
1549 : ListCell *lname,
1550 : *lvar;
1551 : int i;
1552 :
1553 30 : expandRTE(rte, var->varno, 0, var->varreturningtype,
1554 : var->location, false, &names, &vars);
1555 :
1556 30 : tupleDesc = CreateTemplateTupleDesc(list_length(vars));
1557 30 : i = 1;
1558 90 : forboth(lname, names, lvar, vars)
1559 : {
1560 60 : char *label = strVal(lfirst(lname));
1561 60 : Node *varnode = (Node *) lfirst(lvar);
1562 :
1563 60 : TupleDescInitEntry(tupleDesc, i,
1564 : label,
1565 : exprType(varnode),
1566 : exprTypmod(varnode),
1567 : 0);
1568 60 : TupleDescInitEntryCollation(tupleDesc, i,
1569 : exprCollation(varnode));
1570 60 : i++;
1571 : }
1572 : Assert(lname == NULL && lvar == NULL); /* lists same length? */
1573 :
1574 30 : return tupleDesc;
1575 : }
1576 :
1577 1802 : expr = (Node *) var; /* default if we can't drill down */
1578 :
1579 1802 : switch (rte->rtekind)
1580 : {
1581 0 : case RTE_RELATION:
1582 : case RTE_VALUES:
1583 : case RTE_NAMEDTUPLESTORE:
1584 : case RTE_RESULT:
1585 :
1586 : /*
1587 : * This case should not occur: a column of a table, values list,
1588 : * or ENR shouldn't have type RECORD. Fall through and fail (most
1589 : * likely) at the bottom.
1590 : */
1591 0 : break;
1592 1724 : case RTE_SUBQUERY:
1593 : {
1594 : /* Subselect-in-FROM: examine sub-select's output expr */
1595 1724 : TargetEntry *ste = get_tle_by_resno(rte->subquery->targetList,
1596 : attnum);
1597 :
1598 1724 : if (ste == NULL || ste->resjunk)
1599 0 : elog(ERROR, "subquery %s does not have attribute %d",
1600 : rte->eref->aliasname, attnum);
1601 1724 : expr = (Node *) ste->expr;
1602 1724 : if (IsA(expr, Var))
1603 : {
1604 : /*
1605 : * Recurse into the sub-select to see what its Var refers
1606 : * to. We have to build an additional level of ParseState
1607 : * to keep in step with varlevelsup in the subselect;
1608 : * furthermore, the subquery RTE might be from an outer
1609 : * query level, in which case the ParseState for the
1610 : * subselect must have that outer level as parent.
1611 : */
1612 36 : ParseState mypstate = {0};
1613 : Index levelsup;
1614 :
1615 : /* this loop must work, since GetRTEByRangeTablePosn did */
1616 54 : for (levelsup = 0; levelsup < netlevelsup; levelsup++)
1617 18 : pstate = pstate->parentParseState;
1618 36 : mypstate.parentParseState = pstate;
1619 36 : mypstate.p_rtable = rte->subquery->rtable;
1620 : /* don't bother filling the rest of the fake pstate */
1621 :
1622 36 : return expandRecordVariable(&mypstate, (Var *) expr, 0);
1623 : }
1624 : /* else fall through to inspect the expression */
1625 : }
1626 1688 : break;
1627 0 : case RTE_JOIN:
1628 : /* Join RTE --- recursively inspect the alias variable */
1629 : Assert(attnum > 0 && attnum <= list_length(rte->joinaliasvars));
1630 0 : expr = (Node *) list_nth(rte->joinaliasvars, attnum - 1);
1631 : Assert(expr != NULL);
1632 : /* We intentionally don't strip implicit coercions here */
1633 0 : if (IsA(expr, Var))
1634 0 : return expandRecordVariable(pstate, (Var *) expr, netlevelsup);
1635 : /* else fall through to inspect the expression */
1636 0 : break;
1637 0 : case RTE_FUNCTION:
1638 :
1639 : /*
1640 : * We couldn't get here unless a function is declared with one of
1641 : * its result columns as RECORD, which is not allowed.
1642 : */
1643 0 : break;
1644 0 : case RTE_TABLEFUNC:
1645 :
1646 : /*
1647 : * Table function cannot have columns with RECORD type.
1648 : */
1649 0 : break;
1650 78 : case RTE_CTE:
1651 : /* CTE reference: examine subquery's output expr */
1652 78 : if (!rte->self_reference)
1653 : {
1654 78 : CommonTableExpr *cte = GetCTEForRTE(pstate, rte, netlevelsup);
1655 : TargetEntry *ste;
1656 :
1657 78 : ste = get_tle_by_resno(GetCTETargetList(cte), attnum);
1658 78 : if (ste == NULL || ste->resjunk)
1659 0 : elog(ERROR, "CTE %s does not have attribute %d",
1660 : rte->eref->aliasname, attnum);
1661 78 : expr = (Node *) ste->expr;
1662 78 : if (IsA(expr, Var))
1663 : {
1664 : /*
1665 : * Recurse into the CTE to see what its Var refers to. We
1666 : * have to build an additional level of ParseState to keep
1667 : * in step with varlevelsup in the CTE; furthermore it
1668 : * could be an outer CTE (compare SUBQUERY case above).
1669 : */
1670 42 : ParseState mypstate = {0};
1671 : Index levelsup;
1672 :
1673 : /* this loop must work, since GetCTEForRTE did */
1674 42 : for (levelsup = 0;
1675 78 : levelsup < rte->ctelevelsup + netlevelsup;
1676 36 : levelsup++)
1677 36 : pstate = pstate->parentParseState;
1678 42 : mypstate.parentParseState = pstate;
1679 42 : mypstate.p_rtable = ((Query *) cte->ctequery)->rtable;
1680 : /* don't bother filling the rest of the fake pstate */
1681 :
1682 42 : return expandRecordVariable(&mypstate, (Var *) expr, 0);
1683 : }
1684 : /* else fall through to inspect the expression */
1685 : }
1686 36 : break;
1687 0 : case RTE_GROUP:
1688 :
1689 : /*
1690 : * We couldn't get here: the RTE_GROUP RTE has not been added.
1691 : */
1692 0 : break;
1693 : }
1694 :
1695 : /*
1696 : * We now have an expression we can't expand any more, so see if
1697 : * get_expr_result_tupdesc() can do anything with it.
1698 : */
1699 1724 : return get_expr_result_tupdesc(expr, false);
1700 : }
1701 :
1702 :
1703 : /*
1704 : * FigureColname -
1705 : * if the name of the resulting column is not specified in the target
1706 : * list, we have to guess a suitable name. The SQL spec provides some
1707 : * guidance, but not much...
1708 : *
1709 : * Note that the argument is the *untransformed* parse tree for the target
1710 : * item. This is a shade easier to work with than the transformed tree.
1711 : */
1712 : char *
1713 840336 : FigureColname(Node *node)
1714 : {
1715 840336 : char *name = NULL;
1716 :
1717 840336 : (void) FigureColnameInternal(node, &name);
1718 840336 : if (name != NULL)
1719 765740 : return name;
1720 : /* default result if we can't guess anything */
1721 74596 : return "?column?";
1722 : }
1723 :
1724 : /*
1725 : * FigureIndexColname -
1726 : * choose the name for an expression column in an index
1727 : *
1728 : * This is actually just like FigureColname, except we return NULL if
1729 : * we can't pick a good name.
1730 : */
1731 : char *
1732 898 : FigureIndexColname(Node *node)
1733 : {
1734 898 : char *name = NULL;
1735 :
1736 898 : (void) FigureColnameInternal(node, &name);
1737 898 : return name;
1738 : }
1739 :
1740 : /*
1741 : * FigureColnameInternal -
1742 : * internal workhorse for FigureColname
1743 : *
1744 : * Return value indicates strength of confidence in result:
1745 : * 0 - no information
1746 : * 1 - second-best name choice
1747 : * 2 - good name choice
1748 : * The return value is actually only used internally.
1749 : * If the result isn't zero, *name is set to the chosen name.
1750 : */
1751 : static int
1752 924040 : FigureColnameInternal(Node *node, char **name)
1753 : {
1754 924040 : int strength = 0;
1755 :
1756 924040 : if (node == NULL)
1757 440 : return strength;
1758 :
1759 923600 : switch (nodeTag(node))
1760 : {
1761 506288 : case T_ColumnRef:
1762 : {
1763 506288 : char *fname = NULL;
1764 : ListCell *l;
1765 :
1766 : /* find last field name, if any, ignoring "*" */
1767 1253574 : foreach(l, ((ColumnRef *) node)->fields)
1768 : {
1769 747286 : Node *i = lfirst(l);
1770 :
1771 747286 : if (IsA(i, String))
1772 747166 : fname = strVal(i);
1773 : }
1774 506288 : if (fname)
1775 : {
1776 506288 : *name = fname;
1777 506288 : return 2;
1778 : }
1779 : }
1780 0 : break;
1781 1712 : case T_A_Indirection:
1782 : {
1783 1712 : A_Indirection *ind = (A_Indirection *) node;
1784 1712 : char *fname = NULL;
1785 : ListCell *l;
1786 :
1787 : /* find last field name, if any, ignoring "*" and subscripts */
1788 3714 : foreach(l, ind->indirection)
1789 : {
1790 2002 : Node *i = lfirst(l);
1791 :
1792 2002 : if (IsA(i, String))
1793 592 : fname = strVal(i);
1794 : }
1795 1712 : if (fname)
1796 : {
1797 570 : *name = fname;
1798 570 : return 2;
1799 : }
1800 1142 : return FigureColnameInternal(ind->arg, name);
1801 : }
1802 : break;
1803 216522 : case T_FuncCall:
1804 216522 : *name = strVal(llast(((FuncCall *) node)->funcname));
1805 216522 : return 2;
1806 32330 : case T_A_Expr:
1807 32330 : if (((A_Expr *) node)->kind == AEXPR_NULLIF)
1808 : {
1809 : /* make nullif() act like a regular function */
1810 46 : *name = "nullif";
1811 46 : return 2;
1812 : }
1813 32284 : break;
1814 49750 : case T_TypeCast:
1815 49750 : strength = FigureColnameInternal(((TypeCast *) node)->arg,
1816 : name);
1817 49750 : if (strength <= 1)
1818 : {
1819 11798 : if (((TypeCast *) node)->typeName != NULL)
1820 : {
1821 11798 : *name = strVal(llast(((TypeCast *) node)->typeName->names));
1822 11798 : return 1;
1823 : }
1824 : }
1825 37952 : break;
1826 130 : case T_CollateClause:
1827 130 : return FigureColnameInternal(((CollateClause *) node)->arg, name);
1828 296 : case T_GroupingFunc:
1829 : /* make GROUPING() act like a regular function */
1830 296 : *name = "grouping";
1831 296 : return 2;
1832 120 : case T_MergeSupportFunc:
1833 : /* make MERGE_ACTION() act like a regular function */
1834 120 : *name = "merge_action";
1835 120 : return 2;
1836 5164 : case T_SubLink:
1837 5164 : switch (((SubLink *) node)->subLinkType)
1838 : {
1839 92 : case EXISTS_SUBLINK:
1840 92 : *name = "exists";
1841 92 : return 2;
1842 72 : case ARRAY_SUBLINK:
1843 72 : *name = "array";
1844 72 : return 2;
1845 4952 : case EXPR_SUBLINK:
1846 : {
1847 : /* Get column name of the subquery's single target */
1848 4952 : SubLink *sublink = (SubLink *) node;
1849 4952 : Query *query = (Query *) sublink->subselect;
1850 :
1851 : /*
1852 : * The subquery has probably already been transformed,
1853 : * but let's be careful and check that. (The reason
1854 : * we can see a transformed subquery here is that
1855 : * transformSubLink is lazy and modifies the SubLink
1856 : * node in-place.)
1857 : */
1858 4952 : if (IsA(query, Query))
1859 : {
1860 4952 : TargetEntry *te = (TargetEntry *) linitial(query->targetList);
1861 :
1862 4952 : if (te->resname)
1863 : {
1864 4952 : *name = te->resname;
1865 4952 : return 2;
1866 : }
1867 : }
1868 : }
1869 0 : break;
1870 : /* As with other operator-like nodes, these have no names */
1871 48 : case MULTIEXPR_SUBLINK:
1872 : case ALL_SUBLINK:
1873 : case ANY_SUBLINK:
1874 : case ROWCOMPARE_SUBLINK:
1875 : case CTE_SUBLINK:
1876 48 : break;
1877 : }
1878 48 : break;
1879 31784 : case T_CaseExpr:
1880 31784 : strength = FigureColnameInternal((Node *) ((CaseExpr *) node)->defresult,
1881 : name);
1882 31784 : if (strength <= 1)
1883 : {
1884 21258 : *name = "case";
1885 21258 : return 1;
1886 : }
1887 10526 : break;
1888 746 : case T_A_ArrayExpr:
1889 : /* make ARRAY[] act like a function */
1890 746 : *name = "array";
1891 746 : return 2;
1892 528 : case T_RowExpr:
1893 : /* make ROW() act like a function */
1894 528 : *name = "row";
1895 528 : return 2;
1896 226 : case T_CoalesceExpr:
1897 : /* make coalesce() act like a regular function */
1898 226 : *name = "coalesce";
1899 226 : return 2;
1900 164 : case T_MinMaxExpr:
1901 : /* make greatest/least act like a regular function */
1902 164 : switch (((MinMaxExpr *) node)->op)
1903 : {
1904 60 : case IS_GREATEST:
1905 60 : *name = "greatest";
1906 60 : return 2;
1907 104 : case IS_LEAST:
1908 104 : *name = "least";
1909 104 : return 2;
1910 : }
1911 0 : break;
1912 362 : case T_SQLValueFunction:
1913 : /* make these act like a function or variable */
1914 362 : switch (((SQLValueFunction *) node)->op)
1915 : {
1916 18 : case SVFOP_CURRENT_DATE:
1917 18 : *name = "current_date";
1918 18 : return 2;
1919 12 : case SVFOP_CURRENT_TIME:
1920 : case SVFOP_CURRENT_TIME_N:
1921 12 : *name = "current_time";
1922 12 : return 2;
1923 22 : case SVFOP_CURRENT_TIMESTAMP:
1924 : case SVFOP_CURRENT_TIMESTAMP_N:
1925 22 : *name = "current_timestamp";
1926 22 : return 2;
1927 12 : case SVFOP_LOCALTIME:
1928 : case SVFOP_LOCALTIME_N:
1929 12 : *name = "localtime";
1930 12 : return 2;
1931 18 : case SVFOP_LOCALTIMESTAMP:
1932 : case SVFOP_LOCALTIMESTAMP_N:
1933 18 : *name = "localtimestamp";
1934 18 : return 2;
1935 42 : case SVFOP_CURRENT_ROLE:
1936 42 : *name = "current_role";
1937 42 : return 2;
1938 114 : case SVFOP_CURRENT_USER:
1939 114 : *name = "current_user";
1940 114 : return 2;
1941 6 : case SVFOP_USER:
1942 6 : *name = "user";
1943 6 : return 2;
1944 88 : case SVFOP_SESSION_USER:
1945 88 : *name = "session_user";
1946 88 : return 2;
1947 6 : case SVFOP_CURRENT_CATALOG:
1948 6 : *name = "current_catalog";
1949 6 : return 2;
1950 24 : case SVFOP_CURRENT_SCHEMA:
1951 24 : *name = "current_schema";
1952 24 : return 2;
1953 : }
1954 0 : break;
1955 450 : case T_XmlExpr:
1956 : /* make SQL/XML functions act like a regular function */
1957 450 : switch (((XmlExpr *) node)->op)
1958 : {
1959 48 : case IS_XMLCONCAT:
1960 48 : *name = "xmlconcat";
1961 48 : return 2;
1962 108 : case IS_XMLELEMENT:
1963 108 : *name = "xmlelement";
1964 108 : return 2;
1965 6 : case IS_XMLFOREST:
1966 6 : *name = "xmlforest";
1967 6 : return 2;
1968 138 : case IS_XMLPARSE:
1969 138 : *name = "xmlparse";
1970 138 : return 2;
1971 78 : case IS_XMLPI:
1972 78 : *name = "xmlpi";
1973 78 : return 2;
1974 60 : case IS_XMLROOT:
1975 60 : *name = "xmlroot";
1976 60 : return 2;
1977 0 : case IS_XMLSERIALIZE:
1978 0 : *name = "xmlserialize";
1979 0 : return 2;
1980 12 : case IS_DOCUMENT:
1981 : /* nothing */
1982 12 : break;
1983 : }
1984 12 : break;
1985 174 : case T_XmlSerialize:
1986 : /* make XMLSERIALIZE act like a regular function */
1987 174 : *name = "xmlserialize";
1988 174 : return 2;
1989 116 : case T_JsonParseExpr:
1990 : /* make JSON act like a regular function */
1991 116 : *name = "json";
1992 116 : return 2;
1993 112 : case T_JsonScalarExpr:
1994 : /* make JSON_SCALAR act like a regular function */
1995 112 : *name = "json_scalar";
1996 112 : return 2;
1997 92 : case T_JsonSerializeExpr:
1998 : /* make JSON_SERIALIZE act like a regular function */
1999 92 : *name = "json_serialize";
2000 92 : return 2;
2001 344 : case T_JsonObjectConstructor:
2002 : /* make JSON_OBJECT act like a regular function */
2003 344 : *name = "json_object";
2004 344 : return 2;
2005 182 : case T_JsonArrayConstructor:
2006 : case T_JsonArrayQueryConstructor:
2007 : /* make JSON_ARRAY act like a regular function */
2008 182 : *name = "json_array";
2009 182 : return 2;
2010 156 : case T_JsonObjectAgg:
2011 : /* make JSON_OBJECTAGG act like a regular function */
2012 156 : *name = "json_objectagg";
2013 156 : return 2;
2014 120 : case T_JsonArrayAgg:
2015 : /* make JSON_ARRAYAGG act like a regular function */
2016 120 : *name = "json_arrayagg";
2017 120 : return 2;
2018 1476 : case T_JsonFuncExpr:
2019 : /* make SQL/JSON functions act like a regular function */
2020 1476 : switch (((JsonFuncExpr *) node)->op)
2021 : {
2022 156 : case JSON_EXISTS_OP:
2023 156 : *name = "json_exists";
2024 156 : return 2;
2025 798 : case JSON_QUERY_OP:
2026 798 : *name = "json_query";
2027 798 : return 2;
2028 522 : case JSON_VALUE_OP:
2029 522 : *name = "json_value";
2030 522 : return 2;
2031 : /* JSON_TABLE_OP can't happen here. */
2032 0 : default:
2033 0 : elog(ERROR, "unrecognized JsonExpr op: %d",
2034 : (int) ((JsonFuncExpr *) node)->op);
2035 : }
2036 : break;
2037 74256 : default:
2038 74256 : break;
2039 : }
2040 :
2041 155078 : return strength;
2042 : }
|