LCOV - code coverage report
Current view: top level - src/backend/executor - nodeTableFuncscan.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 156 163 95.7 %
Date: 2025-01-18 05:15:39 Functions: 8 9 88.9 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * nodeTableFuncscan.c
       4             :  *    Support routines for scanning RangeTableFunc (XMLTABLE like functions).
       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/executor/nodeTableFuncscan.c
      12             :  *
      13             :  *-------------------------------------------------------------------------
      14             :  */
      15             : /*
      16             :  * INTERFACE ROUTINES
      17             :  *      ExecTableFuncScan       scans a function.
      18             :  *      ExecFunctionNext        retrieve next tuple in sequential order.
      19             :  *      ExecInitTableFuncScan   creates and initializes a TableFuncscan node.
      20             :  *      ExecEndTableFuncScan        releases any storage allocated.
      21             :  *      ExecReScanTableFuncScan rescans the function
      22             :  */
      23             : #include "postgres.h"
      24             : 
      25             : #include "executor/executor.h"
      26             : #include "executor/nodeTableFuncscan.h"
      27             : #include "executor/tablefunc.h"
      28             : #include "miscadmin.h"
      29             : #include "nodes/execnodes.h"
      30             : #include "utils/builtins.h"
      31             : #include "utils/jsonpath.h"
      32             : #include "utils/lsyscache.h"
      33             : #include "utils/memutils.h"
      34             : #include "utils/xml.h"
      35             : 
      36             : static TupleTableSlot *TableFuncNext(TableFuncScanState *node);
      37             : static bool TableFuncRecheck(TableFuncScanState *node, TupleTableSlot *slot);
      38             : 
      39             : static void tfuncFetchRows(TableFuncScanState *tstate, ExprContext *econtext);
      40             : static void tfuncInitialize(TableFuncScanState *tstate, ExprContext *econtext, Datum doc);
      41             : static void tfuncLoadRows(TableFuncScanState *tstate, ExprContext *econtext);
      42             : 
      43             : /* ----------------------------------------------------------------
      44             :  *                      Scan Support
      45             :  * ----------------------------------------------------------------
      46             :  */
      47             : /* ----------------------------------------------------------------
      48             :  *      TableFuncNext
      49             :  *
      50             :  *      This is a workhorse for ExecTableFuncScan
      51             :  * ----------------------------------------------------------------
      52             :  */
      53             : static TupleTableSlot *
      54       24192 : TableFuncNext(TableFuncScanState *node)
      55             : {
      56             :     TupleTableSlot *scanslot;
      57             : 
      58       24192 :     scanslot = node->ss.ss_ScanTupleSlot;
      59             : 
      60             :     /*
      61             :      * If first time through, read all tuples from function and put them in a
      62             :      * tuplestore. Subsequent calls just fetch tuples from tuplestore.
      63             :      */
      64       24192 :     if (node->tupstore == NULL)
      65         794 :         tfuncFetchRows(node, node->ss.ps.ps_ExprContext);
      66             : 
      67             :     /*
      68             :      * Get the next tuple from tuplestore.
      69             :      */
      70       24078 :     (void) tuplestore_gettupleslot(node->tupstore,
      71             :                                    true,
      72             :                                    false,
      73             :                                    scanslot);
      74       24078 :     return scanslot;
      75             : }
      76             : 
      77             : /*
      78             :  * TableFuncRecheck -- access method routine to recheck a tuple in EvalPlanQual
      79             :  */
      80             : static bool
      81           0 : TableFuncRecheck(TableFuncScanState *node, TupleTableSlot *slot)
      82             : {
      83             :     /* nothing to check */
      84           0 :     return true;
      85             : }
      86             : 
      87             : /* ----------------------------------------------------------------
      88             :  *      ExecTableFuncScan(node)
      89             :  *
      90             :  *      Scans the function sequentially and returns the next qualifying
      91             :  *      tuple.
      92             :  *      We call the ExecScan() routine and pass it the appropriate
      93             :  *      access method functions.
      94             :  * ----------------------------------------------------------------
      95             :  */
      96             : static TupleTableSlot *
      97       24138 : ExecTableFuncScan(PlanState *pstate)
      98             : {
      99       24138 :     TableFuncScanState *node = castNode(TableFuncScanState, pstate);
     100             : 
     101       24138 :     return ExecScan(&node->ss,
     102             :                     (ExecScanAccessMtd) TableFuncNext,
     103             :                     (ExecScanRecheckMtd) TableFuncRecheck);
     104             : }
     105             : 
     106             : /* ----------------------------------------------------------------
     107             :  *      ExecInitTableFuncScan
     108             :  * ----------------------------------------------------------------
     109             :  */
     110             : TableFuncScanState *
     111         626 : ExecInitTableFuncScan(TableFuncScan *node, EState *estate, int eflags)
     112             : {
     113             :     TableFuncScanState *scanstate;
     114         626 :     TableFunc  *tf = node->tablefunc;
     115             :     TupleDesc   tupdesc;
     116             :     int         i;
     117             : 
     118             :     /* check for unsupported flags */
     119             :     Assert(!(eflags & EXEC_FLAG_MARK));
     120             : 
     121             :     /*
     122             :      * TableFuncscan should not have any children.
     123             :      */
     124             :     Assert(outerPlan(node) == NULL);
     125             :     Assert(innerPlan(node) == NULL);
     126             : 
     127             :     /*
     128             :      * create new ScanState for node
     129             :      */
     130         626 :     scanstate = makeNode(TableFuncScanState);
     131         626 :     scanstate->ss.ps.plan = (Plan *) node;
     132         626 :     scanstate->ss.ps.state = estate;
     133         626 :     scanstate->ss.ps.ExecProcNode = ExecTableFuncScan;
     134             : 
     135             :     /*
     136             :      * Miscellaneous initialization
     137             :      *
     138             :      * create expression context for node
     139             :      */
     140         626 :     ExecAssignExprContext(estate, &scanstate->ss.ps);
     141             : 
     142             :     /*
     143             :      * initialize source tuple type
     144             :      */
     145         626 :     tupdesc = BuildDescFromLists(tf->colnames,
     146         626 :                                  tf->coltypes,
     147         626 :                                  tf->coltypmods,
     148         626 :                                  tf->colcollations);
     149             :     /* and the corresponding scan slot */
     150         626 :     ExecInitScanTupleSlot(estate, &scanstate->ss, tupdesc,
     151             :                           &TTSOpsMinimalTuple);
     152             : 
     153             :     /*
     154             :      * Initialize result type and projection.
     155             :      */
     156         626 :     ExecInitResultTypeTL(&scanstate->ss.ps);
     157         626 :     ExecAssignScanProjectionInfo(&scanstate->ss);
     158             : 
     159             :     /*
     160             :      * initialize child expressions
     161             :      */
     162         626 :     scanstate->ss.ps.qual =
     163         626 :         ExecInitQual(node->scan.plan.qual, &scanstate->ss.ps);
     164             : 
     165             :     /* Only XMLTABLE and JSON_TABLE are supported currently */
     166         626 :     scanstate->routine =
     167         626 :         tf->functype == TFT_XMLTABLE ? &XmlTableRoutine : &JsonbTableRoutine;
     168             : 
     169         626 :     scanstate->perTableCxt =
     170         626 :         AllocSetContextCreate(CurrentMemoryContext,
     171             :                               "TableFunc per value context",
     172             :                               ALLOCSET_DEFAULT_SIZES);
     173         626 :     scanstate->opaque = NULL;    /* initialized at runtime */
     174             : 
     175         626 :     scanstate->ns_names = tf->ns_names;
     176             : 
     177         626 :     scanstate->ns_uris =
     178         626 :         ExecInitExprList(tf->ns_uris, (PlanState *) scanstate);
     179         626 :     scanstate->docexpr =
     180         626 :         ExecInitExpr((Expr *) tf->docexpr, (PlanState *) scanstate);
     181         626 :     scanstate->rowexpr =
     182         626 :         ExecInitExpr((Expr *) tf->rowexpr, (PlanState *) scanstate);
     183         626 :     scanstate->colexprs =
     184         626 :         ExecInitExprList(tf->colexprs, (PlanState *) scanstate);
     185         626 :     scanstate->coldefexprs =
     186         626 :         ExecInitExprList(tf->coldefexprs, (PlanState *) scanstate);
     187         626 :     scanstate->colvalexprs =
     188         626 :         ExecInitExprList(tf->colvalexprs, (PlanState *) scanstate);
     189         626 :     scanstate->passingvalexprs =
     190         626 :         ExecInitExprList(tf->passingvalexprs, (PlanState *) scanstate);
     191             : 
     192         626 :     scanstate->notnulls = tf->notnulls;
     193             : 
     194             :     /* these are allocated now and initialized later */
     195         626 :     scanstate->in_functions = palloc(sizeof(FmgrInfo) * tupdesc->natts);
     196         626 :     scanstate->typioparams = palloc(sizeof(Oid) * tupdesc->natts);
     197             : 
     198             :     /*
     199             :      * Fill in the necessary fmgr infos.
     200             :      */
     201        2290 :     for (i = 0; i < tupdesc->natts; i++)
     202             :     {
     203             :         Oid         in_funcid;
     204             : 
     205        1664 :         getTypeInputInfo(TupleDescAttr(tupdesc, i)->atttypid,
     206        1664 :                          &in_funcid, &scanstate->typioparams[i]);
     207        1664 :         fmgr_info(in_funcid, &scanstate->in_functions[i]);
     208             :     }
     209             : 
     210         626 :     return scanstate;
     211             : }
     212             : 
     213             : /* ----------------------------------------------------------------
     214             :  *      ExecEndTableFuncScan
     215             :  *
     216             :  *      frees any storage allocated through C routines.
     217             :  * ----------------------------------------------------------------
     218             :  */
     219             : void
     220         512 : ExecEndTableFuncScan(TableFuncScanState *node)
     221             : {
     222             :     /*
     223             :      * Release tuplestore resources
     224             :      */
     225         512 :     if (node->tupstore != NULL)
     226         434 :         tuplestore_end(node->tupstore);
     227         512 :     node->tupstore = NULL;
     228         512 : }
     229             : 
     230             : /* ----------------------------------------------------------------
     231             :  *      ExecReScanTableFuncScan
     232             :  *
     233             :  *      Rescans the relation.
     234             :  * ----------------------------------------------------------------
     235             :  */
     236             : void
     237         444 : ExecReScanTableFuncScan(TableFuncScanState *node)
     238             : {
     239         444 :     Bitmapset  *chgparam = node->ss.ps.chgParam;
     240             : 
     241         444 :     if (node->ss.ps.ps_ResultTupleSlot)
     242           0 :         ExecClearTuple(node->ss.ps.ps_ResultTupleSlot);
     243         444 :     ExecScanReScan(&node->ss);
     244             : 
     245             :     /*
     246             :      * Recompute when parameters are changed.
     247             :      */
     248         444 :     if (chgparam)
     249             :     {
     250         444 :         if (node->tupstore != NULL)
     251             :         {
     252         246 :             tuplestore_end(node->tupstore);
     253         246 :             node->tupstore = NULL;
     254             :         }
     255             :     }
     256             : 
     257         444 :     if (node->tupstore != NULL)
     258           0 :         tuplestore_rescan(node->tupstore);
     259         444 : }
     260             : 
     261             : /* ----------------------------------------------------------------
     262             :  *      tfuncFetchRows
     263             :  *
     264             :  *      Read rows from a TableFunc producer
     265             :  * ----------------------------------------------------------------
     266             :  */
     267             : static void
     268         794 : tfuncFetchRows(TableFuncScanState *tstate, ExprContext *econtext)
     269             : {
     270         794 :     const TableFuncRoutine *routine = tstate->routine;
     271             :     MemoryContext oldcxt;
     272             :     Datum       value;
     273             :     bool        isnull;
     274             : 
     275             :     Assert(tstate->opaque == NULL);
     276             : 
     277             :     /* build tuplestore for the result */
     278         794 :     oldcxt = MemoryContextSwitchTo(econtext->ecxt_per_query_memory);
     279         794 :     tstate->tupstore = tuplestore_begin_heap(false, false, work_mem);
     280             : 
     281             :     /*
     282             :      * Each call to fetch a new set of rows - of which there may be very many
     283             :      * if XMLTABLE or JSON_TABLE is being used in a lateral join - will
     284             :      * allocate a possibly substantial amount of memory, so we cannot use the
     285             :      * per-query context here. perTableCxt now serves the same function as
     286             :      * "argcontext" does in FunctionScan - a place to store per-one-call (i.e.
     287             :      * one result table) lifetime data (as opposed to per-query or
     288             :      * per-result-tuple).
     289             :      */
     290         794 :     MemoryContextSwitchTo(tstate->perTableCxt);
     291             : 
     292         794 :     PG_TRY();
     293             :     {
     294         794 :         routine->InitOpaque(tstate,
     295         794 :                             tstate->ss.ss_ScanTupleSlot->tts_tupleDescriptor->natts);
     296             : 
     297             :         /*
     298             :          * If evaluating the document expression returns NULL, the table
     299             :          * expression is empty and we return immediately.
     300             :          */
     301         794 :         value = ExecEvalExpr(tstate->docexpr, econtext, &isnull);
     302             : 
     303         794 :         if (!isnull)
     304             :         {
     305             :             /* otherwise, pass the document value to the table builder */
     306         788 :             tfuncInitialize(tstate, econtext, value);
     307             : 
     308             :             /* initialize ordinality counter */
     309         776 :             tstate->ordinal = 1;
     310             : 
     311             :             /* Load all rows into the tuplestore, and we're done */
     312         776 :             tfuncLoadRows(tstate, econtext);
     313             :         }
     314             :     }
     315         114 :     PG_CATCH();
     316             :     {
     317         114 :         if (tstate->opaque != NULL)
     318         114 :             routine->DestroyOpaque(tstate);
     319         114 :         PG_RE_THROW();
     320             :     }
     321         680 :     PG_END_TRY();
     322             : 
     323             :     /* clean up and return to original memory context */
     324             : 
     325         680 :     if (tstate->opaque != NULL)
     326             :     {
     327         680 :         routine->DestroyOpaque(tstate);
     328         680 :         tstate->opaque = NULL;
     329             :     }
     330             : 
     331         680 :     MemoryContextSwitchTo(oldcxt);
     332         680 :     MemoryContextReset(tstate->perTableCxt);
     333         680 : }
     334             : 
     335             : /*
     336             :  * Fill in namespace declarations, the row filter, and column filters in a
     337             :  * table expression builder context.
     338             :  */
     339             : static void
     340         788 : tfuncInitialize(TableFuncScanState *tstate, ExprContext *econtext, Datum doc)
     341             : {
     342         788 :     const TableFuncRoutine *routine = tstate->routine;
     343             :     TupleDesc   tupdesc;
     344             :     ListCell   *lc1,
     345             :                *lc2;
     346             :     bool        isnull;
     347             :     int         colno;
     348             :     Datum       value;
     349         788 :     int         ordinalitycol =
     350         788 :         ((TableFuncScan *) (tstate->ss.ps.plan))->tablefunc->ordinalitycol;
     351             : 
     352             :     /*
     353             :      * Install the document as a possibly-toasted Datum into the tablefunc
     354             :      * context.
     355             :      */
     356         788 :     routine->SetDocument(tstate, doc);
     357             : 
     358             :     /* Evaluate namespace specifications */
     359         794 :     forboth(lc1, tstate->ns_uris, lc2, tstate->ns_names)
     360             :     {
     361          18 :         ExprState  *expr = (ExprState *) lfirst(lc1);
     362          18 :         String     *ns_node = lfirst_node(String, lc2);
     363             :         char       *ns_uri;
     364             :         char       *ns_name;
     365             : 
     366          18 :         value = ExecEvalExpr((ExprState *) expr, econtext, &isnull);
     367          18 :         if (isnull)
     368           0 :             ereport(ERROR,
     369             :                     (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
     370             :                      errmsg("namespace URI must not be null")));
     371          18 :         ns_uri = TextDatumGetCString(value);
     372             : 
     373             :         /* DEFAULT is passed down to SetNamespace as NULL */
     374          18 :         ns_name = ns_node ? strVal(ns_node) : NULL;
     375             : 
     376          18 :         routine->SetNamespace(tstate, ns_name, ns_uri);
     377             :     }
     378             : 
     379             :     /*
     380             :      * Install the row filter expression, if any, into the table builder
     381             :      * context.
     382             :      */
     383         776 :     if (routine->SetRowFilter)
     384             :     {
     385         258 :         value = ExecEvalExpr(tstate->rowexpr, econtext, &isnull);
     386         258 :         if (isnull)
     387           0 :             ereport(ERROR,
     388             :                     (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
     389             :                      errmsg("row filter expression must not be null")));
     390             : 
     391         258 :         routine->SetRowFilter(tstate, TextDatumGetCString(value));
     392             :     }
     393             : 
     394             :     /*
     395             :      * Install the column filter expressions into the table builder context.
     396             :      * If an expression is given, use that; otherwise the column name itself
     397             :      * is the column filter.
     398             :      */
     399         776 :     colno = 0;
     400         776 :     tupdesc = tstate->ss.ss_ScanTupleSlot->tts_tupleDescriptor;
     401        1622 :     foreach(lc1, tstate->colexprs)
     402             :     {
     403             :         char       *colfilter;
     404         846 :         Form_pg_attribute att = TupleDescAttr(tupdesc, colno);
     405             : 
     406         846 :         if (colno != ordinalitycol)
     407             :         {
     408         774 :             ExprState  *colexpr = lfirst(lc1);
     409             : 
     410         774 :             if (colexpr != NULL)
     411             :             {
     412         606 :                 value = ExecEvalExpr(colexpr, econtext, &isnull);
     413         606 :                 if (isnull)
     414           0 :                     ereport(ERROR,
     415             :                             (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
     416             :                              errmsg("column filter expression must not be null"),
     417             :                              errdetail("Filter for column \"%s\" is null.",
     418             :                                        NameStr(att->attname))));
     419         606 :                 colfilter = TextDatumGetCString(value);
     420             :             }
     421             :             else
     422         168 :                 colfilter = NameStr(att->attname);
     423             : 
     424         774 :             routine->SetColumnFilter(tstate, colfilter, colno);
     425             :         }
     426             : 
     427         846 :         colno++;
     428             :     }
     429         776 : }
     430             : 
     431             : /*
     432             :  * Load all the rows from the TableFunc table builder into a tuplestore.
     433             :  */
     434             : static void
     435         776 : tfuncLoadRows(TableFuncScanState *tstate, ExprContext *econtext)
     436             : {
     437         776 :     const TableFuncRoutine *routine = tstate->routine;
     438         776 :     TupleTableSlot *slot = tstate->ss.ss_ScanTupleSlot;
     439         776 :     TupleDesc   tupdesc = slot->tts_tupleDescriptor;
     440         776 :     Datum      *values = slot->tts_values;
     441         776 :     bool       *nulls = slot->tts_isnull;
     442         776 :     int         natts = tupdesc->natts;
     443             :     MemoryContext oldcxt;
     444             :     int         ordinalitycol;
     445             : 
     446         776 :     ordinalitycol =
     447         776 :         ((TableFuncScan *) (tstate->ss.ps.plan))->tablefunc->ordinalitycol;
     448             : 
     449             :     /*
     450             :      * We need a short-lived memory context that we can clean up each time
     451             :      * around the loop, to avoid wasting space. Our default per-tuple context
     452             :      * is fine for the job, since we won't have used it for anything yet in
     453             :      * this tuple cycle.
     454             :      */
     455         776 :     oldcxt = MemoryContextSwitchTo(econtext->ecxt_per_tuple_memory);
     456             : 
     457             :     /*
     458             :      * Keep requesting rows from the table builder until there aren't any.
     459             :      */
     460       24174 :     while (routine->FetchRow(tstate))
     461             :     {
     462       23500 :         ListCell   *cell = list_head(tstate->coldefexprs);
     463             :         int         colno;
     464             : 
     465       23500 :         CHECK_FOR_INTERRUPTS();
     466             : 
     467       23500 :         ExecClearTuple(tstate->ss.ss_ScanTupleSlot);
     468             : 
     469             :         /*
     470             :          * Obtain the value of each column for this row, installing them into
     471             :          * the slot; then add the tuple to the tuplestore.
     472             :          */
     473      161526 :         for (colno = 0; colno < natts; colno++)
     474             :         {
     475      138128 :             Form_pg_attribute att = TupleDescAttr(tupdesc, colno);
     476             : 
     477      138128 :             if (colno == ordinalitycol)
     478             :             {
     479             :                 /* Fast path for ordinality column */
     480         270 :                 values[colno] = Int32GetDatum(tstate->ordinal++);
     481         270 :                 nulls[colno] = false;
     482             :             }
     483             :             else
     484             :             {
     485             :                 bool        isnull;
     486             : 
     487      137858 :                 values[colno] = routine->GetValue(tstate,
     488             :                                                   colno,
     489             :                                                   att->atttypid,
     490             :                                                   att->atttypmod,
     491             :                                                   &isnull);
     492             : 
     493             :                 /* No value?  Evaluate and apply the default, if any */
     494      137762 :                 if (isnull && cell != NULL)
     495             :                 {
     496       22298 :                     ExprState  *coldefexpr = (ExprState *) lfirst(cell);
     497             : 
     498       22298 :                     if (coldefexpr != NULL)
     499         222 :                         values[colno] = ExecEvalExpr(coldefexpr, econtext,
     500             :                                                      &isnull);
     501             :                 }
     502             : 
     503             :                 /* Verify a possible NOT NULL constraint */
     504      137762 :                 if (isnull && bms_is_member(colno, tstate->notnulls))
     505           6 :                     ereport(ERROR,
     506             :                             (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
     507             :                              errmsg("null is not allowed in column \"%s\"",
     508             :                                     NameStr(att->attname))));
     509             : 
     510      137756 :                 nulls[colno] = isnull;
     511             :             }
     512             : 
     513             :             /* advance list of default expressions */
     514      138026 :             if (cell != NULL)
     515      133428 :                 cell = lnext(tstate->coldefexprs, cell);
     516             :         }
     517             : 
     518       23398 :         tuplestore_putvalues(tstate->tupstore, tupdesc, values, nulls);
     519             : 
     520       23398 :         MemoryContextReset(econtext->ecxt_per_tuple_memory);
     521             :     }
     522             : 
     523         674 :     MemoryContextSwitchTo(oldcxt);
     524         674 : }

Generated by: LCOV version 1.14