Commit 99389cb6 authored by Tom Lane's avatar Tom Lane

Make psql's \password default to CURRENT_USER, not PQuser(conn).

The documentation says plainly that \password acts on "the current user"
by default.  What it actually acted on, or tried to, was the username
used to log into the current session.  This is not the same thing if
one has since done SET ROLE or SET SESSION AUTHENTICATION.  Aside from
the possible surprise factor, it's quite likely that the current role
doesn't have permissions to set the password of the original role.

To fix, use "SELECT CURRENT_USER" to get the role name to act on.
(This syntax works with servers at least back to 7.0.)  Also, in
hopes of reducing confusion, include the role name that will be
acted on in the password prompt.

The discrepancy from the documentation makes this a bug, so
back-patch to all supported branches.

Patch by me; thanks to Nathan Bossart for review.

Discussion: https://postgr.es/m/747443.1635536754@sss.pgh.pa.us
parent 5f81a480
...@@ -2022,12 +2022,29 @@ exec_command_password(PsqlScanState scan_state, bool active_branch) ...@@ -2022,12 +2022,29 @@ exec_command_password(PsqlScanState scan_state, bool active_branch)
if (active_branch) if (active_branch)
{ {
char *opt0 = psql_scan_slash_option(scan_state, char *user = psql_scan_slash_option(scan_state,
OT_SQLID, NULL, true); OT_SQLID, NULL, true);
char *pw1; char *pw1;
char *pw2; char *pw2;
PQExpBufferData buf;
if (user == NULL)
{
/* By default, the command applies to CURRENT_USER */
PGresult *res;
res = PSQLexec("SELECT CURRENT_USER");
if (!res)
return PSQL_CMD_ERROR;
user = pg_strdup(PQgetvalue(res, 0, 0));
PQclear(res);
}
pw1 = simple_prompt("Enter new password: ", false); initPQExpBuffer(&buf);
printfPQExpBuffer(&buf, _("Enter new password for user \"%s\": "), user);
pw1 = simple_prompt(buf.data, false);
pw2 = simple_prompt("Enter it again: ", false); pw2 = simple_prompt("Enter it again: ", false);
if (strcmp(pw1, pw2) != 0) if (strcmp(pw1, pw2) != 0)
...@@ -2037,14 +2054,8 @@ exec_command_password(PsqlScanState scan_state, bool active_branch) ...@@ -2037,14 +2054,8 @@ exec_command_password(PsqlScanState scan_state, bool active_branch)
} }
else else
{ {
char *user;
char *encrypted_password; char *encrypted_password;
if (opt0)
user = opt0;
else
user = PQuser(pset.db);
encrypted_password = PQencryptPasswordConn(pset.db, pw1, user, NULL); encrypted_password = PQencryptPasswordConn(pset.db, pw1, user, NULL);
if (!encrypted_password) if (!encrypted_password)
...@@ -2054,15 +2065,12 @@ exec_command_password(PsqlScanState scan_state, bool active_branch) ...@@ -2054,15 +2065,12 @@ exec_command_password(PsqlScanState scan_state, bool active_branch)
} }
else else
{ {
PQExpBufferData buf;
PGresult *res; PGresult *res;
initPQExpBuffer(&buf);
printfPQExpBuffer(&buf, "ALTER USER %s PASSWORD ", printfPQExpBuffer(&buf, "ALTER USER %s PASSWORD ",
fmtId(user)); fmtId(user));
appendStringLiteralConn(&buf, encrypted_password, pset.db); appendStringLiteralConn(&buf, encrypted_password, pset.db);
res = PSQLexec(buf.data); res = PSQLexec(buf.data);
termPQExpBuffer(&buf);
if (!res) if (!res)
success = false; success = false;
else else
...@@ -2071,10 +2079,10 @@ exec_command_password(PsqlScanState scan_state, bool active_branch) ...@@ -2071,10 +2079,10 @@ exec_command_password(PsqlScanState scan_state, bool active_branch)
} }
} }
if (opt0) free(user);
free(opt0);
free(pw1); free(pw1);
free(pw2); free(pw2);
termPQExpBuffer(&buf);
} }
else else
ignore_slash_options(scan_state); ignore_slash_options(scan_state);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment