93

I have MySQL password saved on a file foo.php, for example P455w0rd, when I try to use it:

$ cat foo.php | grep '$dbpwd=' | cut -d '"' -f 2 | mysql -U root -p mydb -h friendserver
Enter password: (holds)

$ echo P455w0rd | mysql -u root -p mydb -h friendserver
Enter password: (holds)

Both option still ask for password, what's the correct way to send password from stdin?

Kokizzu
  • 9,699

6 Answers6

102

The mysql client utility can take a password on the command line with either the -p or --password= options.

If you use -p, there must not be any blank space after the option letter:

$ mysql -pmypassword

I prefer the long options in scripts as they are self-documenting:

mysql --password=mypassword --user=me --host=etc
Kusalananda
  • 333,661
  • 21
    This is insecure because any user could view the password - either directly via /proc/$pid/cmdline or via the ps command. It's true that mysql overwrites the password in argv during startup but there is always a time window where another user could observe the password. Also, on some systems the argv overwriting might not work. – maxschlepzig Jul 14 '19 at 14:49
  • Thanks for the hint that there must not be any blank space after the option letter. Saved my day. – Tobias Marschall Oct 07 '20 at 08:11
  • 11
    Of course this is insecure, but most people are doing something like this in a build process that will be deleted shortly afterwards... – Adam Fowler Jun 29 '21 at 22:41
  • List of running processes might be captured by machine monitoring tools and appear then in log files. A short running process is a bad excuse. – dolmen Sep 23 '22 at 08:55
  • Just be a 'real person' and enter the password when prompted rather than sharing it with anyone, including your mother. – Wilson Hauck Mar 09 '24 at 17:09
70

You have to be very careful how you pass passwords to command lines as, if you're not careful, you'll end up leaving it open to sniffing using tools such as ps.


The safest way to do this would be to create a new config file and pass it to mysql using either the --defaults-file= or --defaults-extra-file= command line option.

The difference between the two is that the latter is read in addition to the default config files whereas with the former, only the one file passed as the argument is used.

Your additional configuration file should contain something similar to:

[client]
user=foo
password=P@55w0rd

Make sure that you secure this file.

Then run:

mysql --defaults-extra-file=<path to the new config file> [all my other options]
garethTheRed
  • 33,957
  • 2
    MySQL will modify its argv to overwrite the parameters given to the -p flag. At least that's what's concluded here, together with other relevant info: http://unix.stackexchange.com/questions/78757/securely-feeding-a-program-with-a-password – Kusalananda Jan 23 '17 at 13:40
  • 2
    This is the best solution, as it provides a measure of security. You need to chmod go-rwx, and make sure this argument preceeds all other arguments. – ChuckCottrill Feb 01 '17 at 00:52
  • 3
    @Kusalananda, yes, but as per comments on https://unix.stackexchange.com/q/385339/135943, that does not mean it's safe! – Wildcard Oct 19 '17 at 05:01
  • Might be worth adding a note about --login-path now that it's supported. It's not much better than this, but it is slightly less plain-text (even though the barrier to converting the contents to plaintext is low). – Geoffrey Wiseman Feb 03 '18 at 01:50
  • 27
    Set MYSQL_PWD in the environment (export MYSQL_PWD=muhpassword) and execute your command without the -p. See MySQL Program Environment Variables. In spite of the manual's dire warnings, this is rather safe. Unless you start weird warez in the same shell later. So we run: MYSQL_PWD=$(cat foo.php etc) mysql -u foouser -h barhost – David Tonhofer Mar 01 '18 at 18:02
  • 1
    The --defaults-extra-file must be the first option when executing the mentioned command. For me, it didn't work otherwise. – vinkomlacic Jun 16 '19 at 21:05
  • @DavidTonhofer this should be an answer on its own. – Mladen Jablanović Jun 03 '21 at 13:46
  • defaults-extra-file didn't worked for me, but defaults-file worked ok. Thx, solved my problem (not wanting to put secrets in cmd). – mFlorin Jan 15 '22 at 01:04
12

create a file ~/.my.cnf, make it only accessible by yourself, permission 600.

   [client]
   user=myuser
   password=mypassword

Then you don't need type password any more.

   bash$ mysql -u myuser mydatabase
   mysql>
oldpride
  • 231
  • 1
    Using mysql_config_editor (an official MySQL tool which stores the credentials obfuscated in ~/.mylogin.cnf`) is slightly more secure (or less insecure). https://dev.mysql.com/doc/refman/8.0/en/mysql-config-editor.html – dolmen Sep 23 '22 at 08:53
8

You can use a nifty Linux trick...

/dev/stdin can be used as a file resource for --defaults-file like so:

echo -e "[client]\nuser=xxx\npassword=xxx" | mysql --defaults-file=/dev/stdin -e 'select user()' <database_name>

You can usually use - instead of /dev/stdin, but it's a convention that a lot of programs (including mysql) do not honour.

A little more about the /dev/stdin file resource:

$ ls -la /dev/stdin
lrwxrwxrwx 1 root root 15 Jun 19 08:35 /dev/stdin -> /proc/self/fd/0

This way you can also check empty password situations

  • 2
    you should use password instead of pass to not see the following error: Info: Using unique option prefix 'pass' is error-prone and can break in the future. Please use the full name 'password' instead. – Bash Stack Aug 30 '20 at 10:45
  • What's with the 'select user() bit? I couldn't get this to work. It just outputs user(). – geoidesic Jan 24 '22 at 17:56
  • This is probably the safest answer, as you won't need to write the credentials to file. You could probably also use process substitution instead of stdin, but the idea is the same. Not sure if that makes a difference to whether a different process can read from the file descriptor, but perhaps someone can enlighten us on that. – Steen Schütt Sep 21 '22 at 10:57
  • this one is geeky ❤️, you deserve a +1 – Sylvain Jan 22 '24 at 16:27
5

As @maxschlepzig said in a comment,

Note that there is the MYSQL_PWD environment variable which is read by mysql if you don't specify -p.

So in bash:

read -s -p "Enter the mysql password for $DBUSER @ $DBNAME: " DBPASS 
export MYSQL_PWD="$DBPASS" 
mysqldump -u $DBUSER $DBNAME > dump.sql

once you are done

export MYSQL_PWD=""

  • 4
    From the MySQL 8.0 documentation (below the table): "Use of MYSQL_PWD to specify a MySQL password must be considered extremely insecure and should not be used. [...] MYSQL_PWD is deprecated as of MySQL 8.0; expect it to be removed in a future version of MySQL." – Steen Schütt Sep 21 '22 at 09:26
-2

If you want to start mysql with a password provided you have to fetch the password in a variable first:

MYSQLPASS=`cat foo.php | grep '$dbpwd=' | cut -d '"' -f 2`

Then you can start your mysql command with:

mysql -U root -p ${MYSQLPASS} mydb -h friendserver
Lambert
  • 12,680