Disallow a digit as the first character of a variable name in pgbench.
authorTom Lane
Wed, 13 Jan 2021 19:52:49 +0000 (14:52 -0500)
committerTom Lane
Wed, 13 Jan 2021 19:52:49 +0000 (14:52 -0500)
The point of this restriction is to avoid trying to substitute variables
into timestamp literal values, which may contain strings like '12:34'.

There is a good deal more that should be done to reduce pgbench's
tendency to substitute where it shouldn't.  But this is sufficient to
solve the case complained of by Jaime Soler, and it's simple enough
to back-patch.

Back-patch to v11; before commit 9d36a3866, pgbench had a slightly
different definition of what a variable name is, and anyway it seems
unwise to change long-stable branches for this.

Fabien Coelho

Discussion: https://postgr.es/m/alpine.DEB.2.22.394.2006291740420.805678@pseudo

doc/src/sgml/ref/pgbench.sgml
src/bin/pgbench/pgbench.c

index 7f658848df86d379aca26852e0e83eb40bf006d0..02ea7e026d580edc3cd77ff6580d002e0c540cd2 100644 (file)
@@ -942,7 +942,7 @@ pgbench  options  d
   
    There is a simple variable-substitution facility for script files.
    Variable names must consist of letters (including non-Latin letters),
-   digits, and underscores.
+   digits, and underscores, with the first character not being a digit.
    Variables can be set by the command-line  option,
    explained above, or by the meta commands explained below.
    In addition to any variables preset by  command-line options,
index e0e22140758f62a8496138b55ad74a2fba8e9e94..bf32638b6fdba3a7178c0b9d5b7a5a17ba014fe0 100644 (file)
@@ -1356,6 +1356,7 @@ makeVariableValue(Variable *var)
  * "src/bin/pgbench/exprscan.l".  Also see parseVariable(), below.
  *
  * Note: this static function is copied from "src/bin/psql/variables.c"
+ * but changed to disallow variable names starting with a digit.
  */
 static bool
 valid_variable_name(const char *name)
@@ -1366,6 +1367,15 @@ valid_variable_name(const char *name)
    if (*ptr == '\0')
        return false;
 
+   /* must not start with [0-9] */
+   if (IS_HIGHBIT_SET(*ptr) ||
+       strchr("ABCDEFGHIJKLMNOPQRSTUVWXYZ" "abcdefghijklmnopqrstuvwxyz"
+              "_", *ptr) != NULL)
+       ptr++;
+   else
+       return false;
+
+   /* remaining characters can include [0-9] */
    while (*ptr)
    {
        if (IS_HIGHBIT_SET(*ptr) ||
@@ -1487,23 +1497,27 @@ putVariableInt(CState *st, const char *context, char *name, int64 value)
  *
  * "sql" points at a colon.  If what follows it looks like a valid
  * variable name, return a malloc'd string containing the variable name,
- * and set *eaten to the number of characters consumed.
+ * and set *eaten to the number of characters consumed (including the colon).
  * Otherwise, return NULL.
  */
 static char *
 parseVariable(const char *sql, int *eaten)
 {
-   int         i = 0;
+   int         i = 1;          /* starting at 1 skips the colon */
    char       *name;
 
-   do
-   {
+   /* keep this logic in sync with valid_variable_name() */
+   if (IS_HIGHBIT_SET(sql[i]) ||
+       strchr("ABCDEFGHIJKLMNOPQRSTUVWXYZ" "abcdefghijklmnopqrstuvwxyz"
+              "_", sql[i]) != NULL)
+       i++;
+   else
+       return NULL;
+
+   while (IS_HIGHBIT_SET(sql[i]) ||
+          strchr("ABCDEFGHIJKLMNOPQRSTUVWXYZ" "abcdefghijklmnopqrstuvwxyz"
+                 "_0123456789", sql[i]) != NULL)
        i++;
-   } while (IS_HIGHBIT_SET(sql[i]) ||
-            strchr("ABCDEFGHIJKLMNOPQRSTUVWXYZ" "abcdefghijklmnopqrstuvwxyz"
-                   "_0123456789", sql[i]) != NULL);
-   if (i == 1)
-       return NULL;            /* no valid variable name chars */
 
    name = pg_malloc(i);
    memcpy(name, &sql[1], i - 1);