mirror of
https://github.com/postgres/postgres.git
synced 2025-09-03 15:22:11 +03:00
Fix connection string handling in src/bin/scripts/ programs.
When told to process all databases, clusterdb, reindexdb, and vacuumdb
would reconnect by replacing their --maintenance-db parameter with the
name of the target database. If that parameter is a connstring (which
has been allowed for a long time, though we failed to document that
before this patch), we'd lose any other options it might specify, for
example SSL or GSS parameters, possibly resulting in failure to connect.
Thus, this is the same bug as commit a45bc8a4f
fixed in pg_dump and
pg_restore. We can fix it in the same way, by using libpq's rules for
handling multiple "dbname" parameters to add the target database name
separately. I chose to apply the same refactoring approach as in that
patch, with a struct to handle the command line parameters that need to
be passed through to connectDatabase. (Maybe someday we can unify the
very similar functions here and in pg_dump/pg_restore.)
Per Peter Eisentraut's comments on bug #16604. Back-patch to all
supported branches.
Discussion: https://postgr.es/m/16604-933f4b8791227b15@postgresql.org
This commit is contained in:
@@ -44,19 +44,16 @@ typedef struct vacuumingOptions
|
||||
} vacuumingOptions;
|
||||
|
||||
|
||||
static void vacuum_one_database(const char *dbname, vacuumingOptions *vacopts,
|
||||
static void vacuum_one_database(const ConnParams *cparams,
|
||||
vacuumingOptions *vacopts,
|
||||
int stage,
|
||||
SimpleStringList *tables,
|
||||
const char *host, const char *port,
|
||||
const char *username, enum trivalue prompt_password,
|
||||
int concurrentCons,
|
||||
const char *progname, bool echo, bool quiet);
|
||||
|
||||
static void vacuum_all_databases(vacuumingOptions *vacopts,
|
||||
static void vacuum_all_databases(ConnParams *cparams,
|
||||
vacuumingOptions *vacopts,
|
||||
bool analyze_in_stages,
|
||||
const char *maintenance_db,
|
||||
const char *host, const char *port,
|
||||
const char *username, enum trivalue prompt_password,
|
||||
int concurrentCons,
|
||||
const char *progname, bool echo, bool quiet);
|
||||
|
||||
@@ -123,6 +120,7 @@ main(int argc, char *argv[])
|
||||
char *port = NULL;
|
||||
char *username = NULL;
|
||||
enum trivalue prompt_password = TRI_DEFAULT;
|
||||
ConnParams cparams;
|
||||
bool echo = false;
|
||||
bool quiet = false;
|
||||
vacuumingOptions vacopts;
|
||||
@@ -249,6 +247,13 @@ main(int argc, char *argv[])
|
||||
/* allow 'and_analyze' with 'analyze_only' */
|
||||
}
|
||||
|
||||
/* fill cparams except for dbname, which is set below */
|
||||
cparams.pghost = host;
|
||||
cparams.pgport = port;
|
||||
cparams.pguser = username;
|
||||
cparams.prompt_password = prompt_password;
|
||||
cparams.override_dbname = NULL;
|
||||
|
||||
setup_cancel_handler();
|
||||
|
||||
/* Avoid opening extra connections. */
|
||||
@@ -270,10 +275,10 @@ main(int argc, char *argv[])
|
||||
exit(1);
|
||||
}
|
||||
|
||||
vacuum_all_databases(&vacopts,
|
||||
cparams.dbname = maintenance_db;
|
||||
|
||||
vacuum_all_databases(&cparams, &vacopts,
|
||||
analyze_in_stages,
|
||||
maintenance_db,
|
||||
host, port, username, prompt_password,
|
||||
concurrentCons,
|
||||
progname, echo, quiet);
|
||||
}
|
||||
@@ -289,25 +294,25 @@ main(int argc, char *argv[])
|
||||
dbname = get_user_name_or_exit(progname);
|
||||
}
|
||||
|
||||
cparams.dbname = dbname;
|
||||
|
||||
if (analyze_in_stages)
|
||||
{
|
||||
int stage;
|
||||
|
||||
for (stage = 0; stage < ANALYZE_NUM_STAGES; stage++)
|
||||
{
|
||||
vacuum_one_database(dbname, &vacopts,
|
||||
vacuum_one_database(&cparams, &vacopts,
|
||||
stage,
|
||||
&tables,
|
||||
host, port, username, prompt_password,
|
||||
concurrentCons,
|
||||
progname, echo, quiet);
|
||||
}
|
||||
}
|
||||
else
|
||||
vacuum_one_database(dbname, &vacopts,
|
||||
vacuum_one_database(&cparams, &vacopts,
|
||||
ANALYZE_NO_STAGE,
|
||||
&tables,
|
||||
host, port, username, prompt_password,
|
||||
concurrentCons,
|
||||
progname, echo, quiet);
|
||||
}
|
||||
@@ -329,11 +334,10 @@ main(int argc, char *argv[])
|
||||
* a list of tables from the database.
|
||||
*/
|
||||
static void
|
||||
vacuum_one_database(const char *dbname, vacuumingOptions *vacopts,
|
||||
vacuum_one_database(const ConnParams *cparams,
|
||||
vacuumingOptions *vacopts,
|
||||
int stage,
|
||||
SimpleStringList *tables,
|
||||
const char *host, const char *port,
|
||||
const char *username, enum trivalue prompt_password,
|
||||
int concurrentCons,
|
||||
const char *progname, bool echo, bool quiet)
|
||||
{
|
||||
@@ -359,8 +363,7 @@ vacuum_one_database(const char *dbname, vacuumingOptions *vacopts,
|
||||
Assert(stage == ANALYZE_NO_STAGE ||
|
||||
(stage >= 0 && stage < ANALYZE_NUM_STAGES));
|
||||
|
||||
conn = connectDatabase(dbname, host, port, username, prompt_password,
|
||||
progname, echo, false, true);
|
||||
conn = connectDatabase(cparams, progname, echo, false, true);
|
||||
|
||||
if (!quiet)
|
||||
{
|
||||
@@ -432,8 +435,7 @@ vacuum_one_database(const char *dbname, vacuumingOptions *vacopts,
|
||||
{
|
||||
for (i = 1; i < concurrentCons; i++)
|
||||
{
|
||||
conn = connectDatabase(dbname, host, port, username, prompt_password,
|
||||
progname, echo, false, true);
|
||||
conn = connectDatabase(cparams, progname, echo, false, true);
|
||||
|
||||
/*
|
||||
* Fail and exit immediately if trying to use a socket in an
|
||||
@@ -556,28 +558,23 @@ finish:
|
||||
* quickly everywhere before generating more detailed ones.
|
||||
*/
|
||||
static void
|
||||
vacuum_all_databases(vacuumingOptions *vacopts,
|
||||
vacuum_all_databases(ConnParams *cparams,
|
||||
vacuumingOptions *vacopts,
|
||||
bool analyze_in_stages,
|
||||
const char *maintenance_db, const char *host,
|
||||
const char *port, const char *username,
|
||||
enum trivalue prompt_password,
|
||||
int concurrentCons,
|
||||
const char *progname, bool echo, bool quiet)
|
||||
{
|
||||
PGconn *conn;
|
||||
PGresult *result;
|
||||
PQExpBufferData connstr;
|
||||
int stage;
|
||||
int i;
|
||||
|
||||
conn = connectMaintenanceDatabase(maintenance_db, host, port, username,
|
||||
prompt_password, progname, echo);
|
||||
conn = connectMaintenanceDatabase(cparams, progname, echo);
|
||||
result = executeQuery(conn,
|
||||
"SELECT datname FROM pg_database WHERE datallowconn ORDER BY 1;",
|
||||
progname, echo);
|
||||
PQfinish(conn);
|
||||
|
||||
initPQExpBuffer(&connstr);
|
||||
if (analyze_in_stages)
|
||||
{
|
||||
/*
|
||||
@@ -592,14 +589,11 @@ vacuum_all_databases(vacuumingOptions *vacopts,
|
||||
{
|
||||
for (i = 0; i < PQntuples(result); i++)
|
||||
{
|
||||
resetPQExpBuffer(&connstr);
|
||||
appendPQExpBuffer(&connstr, "dbname=");
|
||||
appendConnStrVal(&connstr, PQgetvalue(result, i, 0));
|
||||
cparams->override_dbname = PQgetvalue(result, i, 0);
|
||||
|
||||
vacuum_one_database(connstr.data, vacopts,
|
||||
vacuum_one_database(cparams, vacopts,
|
||||
stage,
|
||||
NULL,
|
||||
host, port, username, prompt_password,
|
||||
concurrentCons,
|
||||
progname, echo, quiet);
|
||||
}
|
||||
@@ -609,19 +603,15 @@ vacuum_all_databases(vacuumingOptions *vacopts,
|
||||
{
|
||||
for (i = 0; i < PQntuples(result); i++)
|
||||
{
|
||||
resetPQExpBuffer(&connstr);
|
||||
appendPQExpBuffer(&connstr, "dbname=");
|
||||
appendConnStrVal(&connstr, PQgetvalue(result, i, 0));
|
||||
cparams->override_dbname = PQgetvalue(result, i, 0);
|
||||
|
||||
vacuum_one_database(connstr.data, vacopts,
|
||||
vacuum_one_database(cparams, vacopts,
|
||||
ANALYZE_NO_STAGE,
|
||||
NULL,
|
||||
host, port, username, prompt_password,
|
||||
concurrentCons,
|
||||
progname, echo, quiet);
|
||||
}
|
||||
}
|
||||
termPQExpBuffer(&connstr);
|
||||
|
||||
PQclear(result);
|
||||
}
|
||||
|
Reference in New Issue
Block a user