Thread: BUG #16304: ECPG won't accept input from STDIN

BUG #16304: ECPG won't accept input from STDIN

From
PG Bug reporting form
Date:
The following bug has been logged on the website:

Bug reference:      16304
Logged by:          James Gray
Email address:      jim_gray@cox.net
PostgreSQL version: 11.5
Operating system:   AIX
Description:

On an older version (9.6.3) of Postgres on CentOS 7.7.1908, ECPG accepts
input from STDIN:

[singha7] (jgray) ~> psql --version
psql (PostgreSQL) 9.6.3
[singha7] (jgray) ~>

[singha7] (jgray) ~> ecpg -o - -
/* Processed by ecpg (4.12.0) */
/* These include files are added by the preprocessor */
#include <ecpglib.h>
#include <ecpgerrno.h>
#include <sqlca.h>
/* End of automatic include section */

#line 1 "stdin"
^C
[singha7] (jgray) ~>

On a newer version (11.5) of Postgres on AIX 7.2, ECPG does NOT accept input
from STDIN:

bash-5.0$ psql --version
psql (PostgreSQL) 11.5
bash-5.0$

bash-5.0$ ecpg -o - -
Try "ecpg --help" for more information.
bash-5.0$

I suspect a problem in the Postgres version of getopt_long.c


Re: BUG #16304: ECPG won't accept input from STDIN

From
Bruce Momjian
Date:
On Mon, Mar 16, 2020 at 08:48:22PM +0000, PG Bug reporting form wrote:
> The following bug has been logged on the website:
> 
> Bug reference:      16304
> Logged by:          James Gray
> Email address:      jim_gray@cox.net
> PostgreSQL version: 11.5
> Operating system:   AIX
> Description:        
> 
> On an older version (9.6.3) of Postgres on CentOS 7.7.1908, ECPG accepts
> input from STDIN:
> 
> [singha7] (jgray) ~> psql --version
> psql (PostgreSQL) 9.6.3
> [singha7] (jgray) ~>
> 
> [singha7] (jgray) ~> ecpg -o - -
> /* Processed by ecpg (4.12.0) */
> /* These include files are added by the preprocessor */
> #include <ecpglib.h>
> #include <ecpgerrno.h>
> #include <sqlca.h>
> /* End of automatic include section */
> 
> #line 1 "stdin"
> ^C
> [singha7] (jgray) ~>
> 
> On a newer version (11.5) of Postgres on AIX 7.2, ECPG does NOT accept input
> from STDIN:
> 
> bash-5.0$ psql --version
> psql (PostgreSQL) 11.5
> bash-5.0$
> 
> bash-5.0$ ecpg -o - -
> Try "ecpg --help" for more information.
> bash-5.0$
> 
> I suspect a problem in the Postgres version of getopt_long.c

Yeah, you might be right.  In some cases we use the libc (maybe GNU)
getopt-long, and in other cases we use our own version, and there are
behavioral differences between the two.  The fact you are testing on
CentOS and the other on AIX probably explains it.  I think using
/dev/stdin should work on both.


-- 
  Bruce Momjian  <bruce@momjian.us>        https://momjian.us
  EnterpriseDB                             https://enterprisedb.com

+ As you are, so once was I.  As I am, so you will be. +
+                      Ancient Roman grave inscription +



Re: BUG #16304: ECPG won't accept input from STDIN

From
Tom Lane
Date:
Bruce Momjian <bruce@momjian.us> writes:
> On Mon, Mar 16, 2020 at 08:48:22PM +0000, PG Bug reporting form wrote:
>> On an older version (9.6.3) of Postgres on CentOS 7.7.1908, ECPG accepts
>> input from STDIN:
>> ...
>> On a newer version (11.5) of Postgres on AIX 7.2, ECPG does NOT accept input
>> from STDIN:
>> I suspect a problem in the Postgres version of getopt_long.c

> Yeah, you might be right.  In some cases we use the libc (maybe GNU)
> getopt-long, and in other cases we use our own version, and there are
> behavioral differences between the two.  The fact you are testing on
> CentOS and the other on AIX probably explains it.  I think using
> /dev/stdin should work on both.

Oooh!  I'd looked at this report and figured that somebody broke ecpg's
command parsing, not thinking hard about the platform dependence angle.
But you are both right: our version of getopt_long() fails to treat
bare "-" correctly.  It starts parsing it as options, fails to find
'\0' in the options string, and returns BADCH.  This is

(a) useless; there's no possible value in that behavior

(b) unlike what GNU getopt_long does

(c) clearly not what ecpg is expecting.  Although it's not documented
(also our bad), a look at ecpg.c clearly shows that it intends to
treat bare "-" as a filename argument meaning stdin

(d) arguably wrong according to POSIX.  POSIX doesn't specify getopt_long
of course, but it does specify getopt(), and that page is absolutely
unambiguous that an argv element containing just "-" is to be treated
as a non-switch argument.  (Which indeed our version of getopt does)

So I think we ought to fix it, as attached.  I've verified that on
a platform using our getopt_long, ecpg misbehaves as reported, and
this fixes it.

We should also fix ecpg's documentation...

            regards, tom lane

diff --git a/src/port/getopt_long.c b/src/port/getopt_long.c
index ff379db..c989276 100644
--- a/src/port/getopt_long.c
+++ b/src/port/getopt_long.c
@@ -79,14 +79,22 @@ getopt_long(int argc, char *const argv[],
 
         place++;
 
-        if (place[0] && place[0] == '-' && place[1] == '\0')
-        {                        /* found "--" */
+        if (!*place)
+        {
+            /* treat "-" as not being an option */
+            place = EMSG;
+            return -1;
+        }
+
+        if (place[0] == '-' && place[1] == '\0')
+        {
+            /* found "--", treat it as end of options */
             ++optind;
             place = EMSG;
             return -1;
         }
 
-        if (place[0] && place[0] == '-' && place[1])
+        if (place[0] == '-' && place[1])
         {
             /* long option */
             size_t        namelen;