LCOV - code coverage report
Current view: top level - src/bin/pg_dump - connectdb.c (source / functions) Coverage Total Hit
Test: PostgreSQL 19devel Lines: 79.0 % 124 98
Test Date: 2026-03-03 18:14:56 Functions: 100.0 % 3 3
Legend: Lines:     hit not hit

            Line data    Source code
       1              : /*-------------------------------------------------------------------------
       2              :  *
       3              :  * connectdb.c
       4              :  *    This is a common file connection to the database.
       5              :  *
       6              :  * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
       7              :  * Portions Copyright (c) 1994, Regents of the University of California
       8              :  *
       9              :  * IDENTIFICATION
      10              :  *    src/bin/pg_dump/connectdb.c
      11              :  *
      12              :  *-------------------------------------------------------------------------
      13              :  */
      14              : 
      15              : #include "postgres_fe.h"
      16              : 
      17              : #include "common/connect.h"
      18              : #include "common/logging.h"
      19              : #include "common/string.h"
      20              : #include "connectdb.h"
      21              : #include "dumputils.h"
      22              : #include "fe_utils/string_utils.h"
      23              : 
      24              : static char *constructConnStr(const char **keywords, const char **values);
      25              : 
      26              : /*
      27              :  * ConnectDatabase
      28              :  *
      29              :  * Make a database connection with the given parameters.  An
      30              :  * interactive password prompt is automatically issued if required.
      31              :  *
      32              :  * If fail_on_error is false, we return NULL without printing any message
      33              :  * on failure, but preserve any prompted password for the next try.
      34              :  *
      35              :  * On success, the 'connstr' is set to a connection string containing
      36              :  * the options used and 'server_version' is set to version so that caller
      37              :  * can use them.
      38              :  */
      39              : PGconn *
      40          436 : ConnectDatabase(const char *dbname, const char *connection_string,
      41              :                 const char *pghost, const char *pgport, const char *pguser,
      42              :                 trivalue prompt_password, bool fail_on_error, const char *progname,
      43              :                 const char **connstr, int *server_version, char *password,
      44              :                 char *override_dbname)
      45              : {
      46              :     PGconn     *conn;
      47              :     bool        new_pass;
      48              :     const char *remoteversion_str;
      49              :     int         my_version;
      50          436 :     const char **keywords = NULL;
      51          436 :     const char **values = NULL;
      52          436 :     PQconninfoOption *conn_opts = NULL;
      53              :     int         server_version_temp;
      54              : 
      55          436 :     if (prompt_password == TRI_YES && !password)
      56            0 :         password = simple_prompt("Password: ", false);
      57              : 
      58              :     /*
      59              :      * Start the connection.  Loop until we have a password if requested by
      60              :      * backend.
      61              :      */
      62              :     do
      63              :     {
      64          436 :         int         argcount = 8;
      65              :         PQconninfoOption *conn_opt;
      66          436 :         char       *err_msg = NULL;
      67          436 :         int         i = 0;
      68              : 
      69          436 :         free(keywords);
      70          436 :         free(values);
      71          436 :         PQconninfoFree(conn_opts);
      72              : 
      73              :         /*
      74              :          * Merge the connection info inputs given in form of connection string
      75              :          * and other options.  Explicitly discard any dbname value in the
      76              :          * connection string; otherwise, PQconnectdbParams() would interpret
      77              :          * that value as being itself a connection string.
      78              :          */
      79          436 :         if (connection_string)
      80              :         {
      81           41 :             conn_opts = PQconninfoParse(connection_string, &err_msg);
      82           41 :             if (conn_opts == NULL)
      83            0 :                 pg_fatal("%s", err_msg);
      84              : 
      85         2132 :             for (conn_opt = conn_opts; conn_opt->keyword != NULL; conn_opt++)
      86              :             {
      87         2091 :                 if (conn_opt->val != NULL && conn_opt->val[0] != '\0' &&
      88           25 :                     strcmp(conn_opt->keyword, "dbname") != 0)
      89           16 :                     argcount++;
      90              :             }
      91              : 
      92           41 :             keywords = pg_malloc0_array(const char *, (argcount + 1));
      93           41 :             values = pg_malloc0_array(const char *, (argcount + 1));
      94              : 
      95         2132 :             for (conn_opt = conn_opts; conn_opt->keyword != NULL; conn_opt++)
      96              :             {
      97         2091 :                 if (conn_opt->val != NULL && conn_opt->val[0] != '\0' &&
      98           25 :                     strcmp(conn_opt->keyword, "dbname") != 0)
      99              :                 {
     100           16 :                     keywords[i] = conn_opt->keyword;
     101           16 :                     values[i] = conn_opt->val;
     102           16 :                     i++;
     103              :                 }
     104              :             }
     105              :         }
     106              :         else
     107              :         {
     108          395 :             keywords = pg_malloc0_array(const char *, (argcount + 1));
     109          395 :             values = pg_malloc0_array(const char *, (argcount + 1));
     110              :         }
     111              : 
     112          436 :         if (pghost)
     113              :         {
     114          138 :             keywords[i] = "host";
     115          138 :             values[i] = pghost;
     116          138 :             i++;
     117              :         }
     118          436 :         if (pgport)
     119              :         {
     120          176 :             keywords[i] = "port";
     121          176 :             values[i] = pgport;
     122          176 :             i++;
     123              :         }
     124          436 :         if (pguser)
     125              :         {
     126          156 :             keywords[i] = "user";
     127          156 :             values[i] = pguser;
     128          156 :             i++;
     129              :         }
     130          436 :         if (password)
     131              :         {
     132            0 :             keywords[i] = "password";
     133            0 :             values[i] = password;
     134            0 :             i++;
     135              :         }
     136          436 :         if (dbname)
     137              :         {
     138          422 :             keywords[i] = "dbname";
     139          422 :             values[i] = dbname;
     140          422 :             i++;
     141              :         }
     142          436 :         if (override_dbname)
     143              :         {
     144           64 :             keywords[i] = "dbname";
     145           64 :             values[i] = override_dbname;
     146           64 :             i++;
     147              :         }
     148              : 
     149          436 :         keywords[i] = "fallback_application_name";
     150          436 :         values[i] = progname;
     151          436 :         i++;
     152              : 
     153          436 :         new_pass = false;
     154          436 :         conn = PQconnectdbParams(keywords, values, true);
     155              : 
     156          436 :         if (!conn)
     157            0 :             pg_fatal("could not connect to database \"%s\"", dbname);
     158              : 
     159          439 :         if (PQstatus(conn) == CONNECTION_BAD &&
     160            3 :             PQconnectionNeedsPassword(conn) &&
     161            0 :             !password &&
     162              :             prompt_password != TRI_NO)
     163              :         {
     164            0 :             PQfinish(conn);
     165            0 :             password = simple_prompt("Password: ", false);
     166            0 :             new_pass = true;
     167              :         }
     168          436 :     } while (new_pass);
     169              : 
     170              :     /* check to see that the backend connection was successfully made */
     171          436 :     if (PQstatus(conn) == CONNECTION_BAD)
     172              :     {
     173            3 :         if (fail_on_error)
     174            3 :             pg_fatal("%s", PQerrorMessage(conn));
     175              :         else
     176              :         {
     177            0 :             PQfinish(conn);
     178              : 
     179            0 :             free(keywords);
     180            0 :             free(values);
     181            0 :             PQconninfoFree(conn_opts);
     182              : 
     183            0 :             return NULL;
     184              :         }
     185              :     }
     186              : 
     187              :     /*
     188              :      * Ok, connected successfully. If requested, remember the options used, in
     189              :      * the form of a connection string.
     190              :      */
     191          433 :     if (connstr)
     192           41 :         *connstr = constructConnStr(keywords, values);
     193              : 
     194          433 :     free(keywords);
     195          433 :     free(values);
     196          433 :     PQconninfoFree(conn_opts);
     197              : 
     198              :     /* Check version */
     199          433 :     remoteversion_str = PQparameterStatus(conn, "server_version");
     200          433 :     if (!remoteversion_str)
     201            0 :         pg_fatal("could not get server version");
     202              : 
     203          433 :     server_version_temp = PQserverVersion(conn);
     204          433 :     if (server_version_temp == 0)
     205            0 :         pg_fatal("could not parse server version \"%s\"",
     206              :                  remoteversion_str);
     207              : 
     208              :     /* If requested, then copy server version to out variable. */
     209          433 :     if (server_version)
     210           41 :         *server_version = server_version_temp;
     211              : 
     212          433 :     my_version = PG_VERSION_NUM;
     213              : 
     214              :     /*
     215              :      * We allow the server to be back to 9.2, and up to any minor release of
     216              :      * our own major version.  (See also version check in pg_dump.c.)
     217              :      */
     218          433 :     if (my_version != server_version_temp
     219            0 :         && (server_version_temp < 90200 ||
     220            0 :             (server_version_temp / 100) > (my_version / 100)))
     221              :     {
     222            0 :         pg_log_error("aborting because of server version mismatch");
     223            0 :         pg_log_error_detail("server version: %s; %s version: %s",
     224              :                             remoteversion_str, progname, PG_VERSION);
     225            0 :         exit_nicely(1);
     226              :     }
     227              : 
     228          433 :     PQclear(executeQuery(conn, ALWAYS_SECURE_SEARCH_PATH_SQL));
     229              : 
     230          433 :     return conn;
     231              : }
     232              : 
     233              : /*
     234              :  * constructConnStr
     235              :  *
     236              :  * Construct a connection string from the given keyword/value pairs. It is
     237              :  * used to pass the connection options to the pg_dump subprocess.
     238              :  *
     239              :  * The following parameters are excluded:
     240              :  *  dbname      - varies in each pg_dump invocation
     241              :  *  password    - it's not secure to pass a password on the command line
     242              :  *  fallback_application_name - we'll let pg_dump set it
     243              :  */
     244              : static char *
     245           41 : constructConnStr(const char **keywords, const char **values)
     246              : {
     247           41 :     PQExpBuffer buf = createPQExpBuffer();
     248              :     char       *connstr;
     249              :     int         i;
     250           41 :     bool        firstkeyword = true;
     251              : 
     252              :     /* Construct a new connection string in key='value' format. */
     253          182 :     for (i = 0; keywords[i] != NULL; i++)
     254              :     {
     255          141 :         if (strcmp(keywords[i], "dbname") == 0 ||
     256          100 :             strcmp(keywords[i], "password") == 0 ||
     257          100 :             strcmp(keywords[i], "fallback_application_name") == 0)
     258           82 :             continue;
     259              : 
     260           59 :         if (!firstkeyword)
     261           34 :             appendPQExpBufferChar(buf, ' ');
     262           59 :         firstkeyword = false;
     263           59 :         appendPQExpBuffer(buf, "%s=", keywords[i]);
     264           59 :         appendConnStrVal(buf, values[i]);
     265              :     }
     266              : 
     267           41 :     connstr = pg_strdup(buf->data);
     268           41 :     destroyPQExpBuffer(buf);
     269           41 :     return connstr;
     270              : }
     271              : 
     272              : /*
     273              :  * executeQuery
     274              :  *
     275              :  * Run a query, return the results, exit program on failure.
     276              :  */
     277              : PGresult *
     278         1056 : executeQuery(PGconn *conn, const char *query)
     279              : {
     280              :     PGresult   *res;
     281              : 
     282         1056 :     pg_log_info("executing %s", query);
     283              : 
     284         1056 :     res = PQexec(conn, query);
     285         2112 :     if (!res ||
     286         1056 :         PQresultStatus(res) != PGRES_TUPLES_OK)
     287              :     {
     288            0 :         pg_log_error("query failed: %s", PQerrorMessage(conn));
     289            0 :         pg_log_error_detail("Query was: %s", query);
     290            0 :         PQfinish(conn);
     291            0 :         exit_nicely(1);
     292              :     }
     293              : 
     294         1056 :     return res;
     295              : }
        

Generated by: LCOV version 2.0-1