aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorTom Lane <tgl@sss.pgh.pa.us>2003-10-04 21:05:21 +0000
committerTom Lane <tgl@sss.pgh.pa.us>2003-10-04 21:05:21 +0000
commitfa09ee60263aa508b20efdc97e36fc725aaf7025 (patch)
tree0ebe72793ba9bb4f7434bf15026afe75374ba284 /src
parentb833c3d4a468975bfea1abc11feee1ca47de84b1 (diff)
downloadpostgresql-fa09ee60263aa508b20efdc97e36fc725aaf7025.tar.gz
postgresql-fa09ee60263aa508b20efdc97e36fc725aaf7025.zip
Document the always-true-but-previously-undocumented fact that PQfnumber()
will downcase the supplied field name unless it is double-quoted. Also, upgrade the routine's handling of double quotes to match the backend, in particular support doubled double quotes within quoted identifiers. Per pgsql-interfaces discussion a couple weeks ago.
Diffstat (limited to 'src')
-rw-r--r--src/interfaces/libpq/fe-exec.c68
1 files changed, 58 insertions, 10 deletions
diff --git a/src/interfaces/libpq/fe-exec.c b/src/interfaces/libpq/fe-exec.c
index 6d159a9a35a..810a753ed3c 100644
--- a/src/interfaces/libpq/fe-exec.c
+++ b/src/interfaces/libpq/fe-exec.c
@@ -8,7 +8,7 @@
*
*
* IDENTIFICATION
- * $Header: /cvsroot/pgsql/src/interfaces/libpq/fe-exec.c,v 1.150 2003/10/03 18:26:14 tgl Exp $
+ * $Header: /cvsroot/pgsql/src/interfaces/libpq/fe-exec.c,v 1.151 2003/10/04 21:05:21 tgl Exp $
*
*-------------------------------------------------------------------------
*/
@@ -1803,32 +1803,80 @@ PQfname(const PGresult *res, int field_num)
}
/*
- * returns -1 on a bad field name
+ * PQfnumber: find column number given column name
+ *
+ * The column name is parsed as if it were in a SQL statement, including
+ * case-folding and double-quote processing. But note a possible gotcha:
+ * downcasing in the frontend might follow different locale rules than
+ * downcasing in the backend...
+ *
+ * Returns -1 if no match. In the present backend it is also possible
+ * to have multiple matches, in which case the first one is found.
*/
int
PQfnumber(const PGresult *res, const char *field_name)
{
- int i;
char *field_case;
+ bool in_quotes;
+ char *iptr;
+ char *optr;
+ int i;
if (!res)
return -1;
+ /*
+ * Note: it is correct to reject a zero-length input string; the proper
+ * input to match a zero-length field name would be "".
+ */
if (field_name == NULL ||
field_name[0] == '\0' ||
res->attDescs == NULL)
return -1;
+ /*
+ * Note: this code will not reject partially quoted strings, eg
+ * foo"BAR"foo will become fooBARfoo when it probably ought to be
+ * an error condition.
+ */
field_case = strdup(field_name);
- if (*field_case == '"')
+ if (field_case == NULL)
+ return -1; /* grotty */
+
+ in_quotes = false;
+ optr = field_case;
+ for (iptr = field_case; *iptr; iptr++)
{
- strcpy(field_case, field_case + 1);
- *(field_case + strlen(field_case) - 1) = '\0';
+ char c = *iptr;
+
+ if (in_quotes)
+ {
+ if (c == '"')
+ {
+ if (iptr[1] == '"')
+ {
+ /* doubled quotes become a single quote */
+ *optr++ = '"';
+ iptr++;
+ }
+ else
+ in_quotes = false;
+ }
+ else
+ *optr++ = c;
+ }
+ else if (c == '"')
+ {
+ in_quotes = true;
+ }
+ else
+ {
+ if (isupper((unsigned char) c))
+ c = tolower((unsigned char) c);
+ *optr++ = c;
+ }
}
- else
- for (i = 0; field_case[i]; i++)
- if (isupper((unsigned char) field_case[i]))
- field_case[i] = tolower((unsigned char) field_case[i]);
+ *optr = '\0';
for (i = 0; i < res->numAttributes; i++)
{