Fix portal management code to support non-default command completion tags for
authorTom Lane <tgl@sss.pgh.pa.us>
Sun, 18 Feb 2007 19:49:42 +0000 (19:49 +0000)
committerTom Lane <tgl@sss.pgh.pa.us>
Sun, 18 Feb 2007 19:49:42 +0000 (19:49 +0000)
portals using PORTAL_UTIL_SELECT strategy.  This is currently significant only
for FETCH queries, which are supposed to include a count in the tag.  Seems
it's been broken since 7.4, but nobody noticed before Knut Lehre.

src/backend/tcop/pquery.c

index e44ab59db2a74a0051b7c881d953f5526e8c5db3..dc46a49eeba6b97cfd54a33f013d2f0e5b512247 100644 (file)
@@ -8,7 +8,7 @@
  *
  *
  * IDENTIFICATION
- *   $PostgreSQL: pgsql/src/backend/tcop/pquery.c,v 1.89.4.2 2005/02/10 20:36:49 tgl Exp $
+ *   $PostgreSQL: pgsql/src/backend/tcop/pquery.c,v 1.89.4.3 2007/02/18 19:49:42 tgl Exp $
  *
  *-------------------------------------------------------------------------
  */
@@ -37,6 +37,7 @@ static void ProcessQuery(Query *parsetree,
             ParamListInfo params,
             DestReceiver *dest,
             char *completionTag);
+static void FillPortalStore(Portal portal);
 static uint32 RunFromStore(Portal portal, ScanDirection direction, long count,
             DestReceiver *dest);
 static long PortalRunSelect(Portal portal, bool forward, long count,
@@ -580,16 +581,7 @@ PortalRun(Portal portal, long count,
                 * storing its results in the portal's tuplestore.
                 */
                if (!portal->portalUtilReady)
-               {
-                   DestReceiver *treceiver;
-
-                   PortalCreateHoldStore(portal);
-                   treceiver = CreateDestReceiver(Tuplestore, portal);
-                   PortalRunUtility(portal, linitial(portal->parseTrees),
-                                    treceiver, NULL);
-                   (*treceiver->rDestroy) (treceiver);
-                   portal->portalUtilReady = true;
-               }
+                   FillPortalStore(portal);
 
                /*
                 * Now fetch desired portion of results.
@@ -819,6 +811,35 @@ PortalRunSelect(Portal portal,
    return nprocessed;
 }
 
+/*
+ * FillPortalStore
+ *     Run the query and load result tuples into the portal's tuple store.
+ *
+ * This is used for PORTAL_UTIL_SELECT cases only.
+ */
+static void
+FillPortalStore(Portal portal)
+{
+   DestReceiver *treceiver;
+   char        completionTag[COMPLETION_TAG_BUFSIZE];
+
+   PortalCreateHoldStore(portal);
+   treceiver = CreateDestReceiver(Tuplestore, portal);
+
+   completionTag[0] = '\0';
+
+   PortalRunUtility(portal, linitial(portal->parseTrees),
+                    treceiver, completionTag);
+
+   /* Override default completion tag with actual command result */
+   if (completionTag[0] != '\0')
+       portal->commandTag = pstrdup(completionTag);
+
+   (*treceiver->rDestroy) (treceiver);
+
+   portal->portalUtilReady = true;
+}
+
 /*
  * RunFromStore
  *     Fetch tuples from the portal's tuple store.
@@ -1127,16 +1148,7 @@ PortalRunFetch(Portal portal,
                 * storing its results in the portal's tuplestore.
                 */
                if (!portal->portalUtilReady)
-               {
-                   DestReceiver *treceiver;
-
-                   PortalCreateHoldStore(portal);
-                   treceiver = CreateDestReceiver(Tuplestore, portal);
-                   PortalRunUtility(portal, linitial(portal->parseTrees),
-                                    treceiver, NULL);
-                   (*treceiver->rDestroy) (treceiver);
-                   portal->portalUtilReady = true;
-               }
+                   FillPortalStore(portal);
 
                /*
                 * Now fetch desired portion of results.