How to properly store a password in the Database

Article originally posted on my website un How to securely store the password in the Database

When I started LOGaritmical, one of the first functionalities that I implemented was registering a new user. This meant that I had to store the user’s password in the database in a secure way and I will need to be able to verify that the password entered during login is correct. This is something that is quite easy to implement incorrectly and you would be surprised (or not) at how many systems and websites store passwords in an incorrect way.

Just look at the number of data leaks from the past few years. It is very likely that your password was leaked at least once, and from a well known and high-profile source. Also, did you ever register for a website, entered the username and password you wanted, and after registration you got an email with the password in clear text? Chances are that the storing was also as insecure.

So, before I show you how to properly store a password (and what mechanism I used for LOGaritmical), let me tell you straight away how NOT to do it:

  • Clear text – NEVER STORE PASSWORDS IN CLEAR – I can’t believe I have to say this. Passwords should not be stored in clear text because any DB dump can easily extract the email and password combination and try it out on other websites.
  • Use your own algorithm – Unless you are a genius in mathematics and encryption, don’t write your own algorithms for encrypting or hashing the password. It may not be as secure as you think.
  • Simple hash – Even though not as bad as the first two, it is not very far away. There are rainbow-tables attacks that can be done on the DB dump and most of the passwords can be easily recovered. A rainbow table is a pre-computed file that has the known hash for many passwords. The file can store hundreds of thousands of passwords and their hash with different algorithms. Next, it is just a matter of matching the DB hashes to the pre-calculated file. Not all passwords will be recovered, but most of them will.

How to securely store the password in the database

The best way to store a password in a database is by using a salted hash algorithm. This way, even in case of a DB dump leak, it will be almost impossible to recover the passwords. How does adding a salt help? First, let’s explain what the salt is. The hash salt is a randomly selected string (or bytes) that is added to the password before hashing it. It does not really matter if you append it or prepend it, as long as you are consistent and the salt is strong enough.

In the database you store the salt for each user and the obtained hash for the salt+password. Keep in mind (and this is very important) each user must have his own salt and it must be changed whenever the password is changed. Also, in order to be sure that the salt is strong enough, it is recommended to be at least 64 bytes long.

Next, use a strong and widely used hashing algorithm. In the past MD5 was considered good enough, after that SHA-1 was recommended, but nowadays it is SHA-256 or better are considered necessary. There are other algorithms as well, but please read about their strengths before deciding to use one.

Hashing algorithms usually have a number of iterations. Here we must be consistent since it will affect the output even if the salt and passwords are the same. Keep in mind though that the greater the number of iterations, the longer it will take for the hash to be computed. This is great since it minimizes the possibility someone will do a brute-force attack in case a DB dump gets leaked. Do some testing to see what is a good number of iterations for your system, without putting any major load on it.

How to check the password entered is correct?

We have the password stored the right way. We have a salt that is unique for each user and a secure hashing algorithm was used. But hashes are one-way functions, which means we can no longer retrieve the password from a given hash, so how do we check during login that the password the user-submitted is the correct one?

We simply compute the hash again for the user-entered password. We retrieve the salt from the DB, we attach it to the user’s password, and recompute the hash using the same algorithm that was used when saving the user’s information. After that, we compare the resulted hash with the one stored in the database. If they are identical, the password entered is the correct one.

How I stored passwords in LOGaritmical

Now that we have all the details, let’s look on how I stored the password in LOGaritmical and how to write a similar functionality in Java. I created a utility class that handles this part. It can generate a salt and it can compute the hash of a String with a provided salt. I chose the following values for the hash strength:

  • Salt length – 521 bytes
  • Hash byte size – 256
  • Iteration – 1000
  • Algorithm – PBKDF2WithHmacSHA512
  • Always used SecureRandom

Below is a part of the utility class (as it is currently is in the project). For the full and updated version feel free to look over the code in LOGaritmical repo.

<span>/** * Generates a cryptographically secure 512 bytes string that can be used as a password salt */</span>
<span>public</span> <span>static</span> <span>String</span> <span>generateSalt</span><span>()</span> <span>{</span>
<span>byte</span><span>[]</span> <span>bytes</span> <span>=</span> <span>new</span> <span>byte</span><span>[</span><span>512</span><span>];</span>
<span>secureRandom</span><span>.</span><span>nextBytes</span><span>(</span><span>bytes</span><span>);</span>
<span>return</span> <span>Base64</span><span>.</span><span>getEncoder</span><span>().</span><span>encodeToString</span><span>(</span><span>bytes</span><span>);</span>
<span>}</span>
<span>/** * Generate a hash for a password */</span>
<span>public</span> <span>static</span> <span>String</span> <span>generateHashForPassword</span><span>(</span><span>String</span> <span>password</span><span>,</span> <span>String</span> <span>salt</span><span>)</span> <span>throws</span> <span>NoSuchAlgorithmException</span><span>,</span> <span>InvalidKeySpecException</span> <span>{</span>
<span>SecretKeyFactory</span> <span>secretKeyFactory</span> <span>=</span> <span>SecretKeyFactory</span><span>.</span><span>getInstance</span><span>(</span><span>"PBKDF2WithHmacSHA512"</span><span>);</span>
<span>PBEKeySpec</span> <span>spec</span> <span>=</span> <span>new</span> <span>PBEKeySpec</span><span>(</span><span>password</span><span>.</span><span>toCharArray</span><span>(),</span> <span>salt</span><span>.</span><span>getBytes</span><span>(),</span> <span>PBKDF2_ITERATIONS</span><span>,</span> <span>HASH_BYTE_SIZE</span><span>);</span>
<span>byte</span><span>[]</span> <span>hash</span> <span>=</span> <span>secretKeyFactory</span><span>.</span><span>generateSecret</span><span>(</span><span>spec</span><span>).</span><span>getEncoded</span><span>();</span>
<span>return</span> <span>Base64</span><span>.</span><span>getEncoder</span><span>().</span><span>encodeToString</span><span>(</span><span>hash</span><span>);</span>
<span>}</span>
<span>/** * Generates a cryptographically secure 512 bytes string that can be used as a password salt */</span>
<span>public</span> <span>static</span> <span>String</span> <span>generateSalt</span><span>()</span> <span>{</span>
    <span>byte</span><span>[]</span> <span>bytes</span> <span>=</span> <span>new</span> <span>byte</span><span>[</span><span>512</span><span>];</span>
    <span>secureRandom</span><span>.</span><span>nextBytes</span><span>(</span><span>bytes</span><span>);</span>
    <span>return</span> <span>Base64</span><span>.</span><span>getEncoder</span><span>().</span><span>encodeToString</span><span>(</span><span>bytes</span><span>);</span>
<span>}</span>

<span>/** * Generate a hash for a password */</span>
<span>public</span> <span>static</span> <span>String</span> <span>generateHashForPassword</span><span>(</span><span>String</span> <span>password</span><span>,</span> <span>String</span> <span>salt</span><span>)</span> <span>throws</span> <span>NoSuchAlgorithmException</span><span>,</span> <span>InvalidKeySpecException</span> <span>{</span>
    <span>SecretKeyFactory</span> <span>secretKeyFactory</span> <span>=</span> <span>SecretKeyFactory</span><span>.</span><span>getInstance</span><span>(</span><span>"PBKDF2WithHmacSHA512"</span><span>);</span>
    <span>PBEKeySpec</span> <span>spec</span> <span>=</span> <span>new</span> <span>PBEKeySpec</span><span>(</span><span>password</span><span>.</span><span>toCharArray</span><span>(),</span> <span>salt</span><span>.</span><span>getBytes</span><span>(),</span> <span>PBKDF2_ITERATIONS</span><span>,</span> <span>HASH_BYTE_SIZE</span><span>);</span>
    <span>byte</span><span>[]</span> <span>hash</span> <span>=</span> <span>secretKeyFactory</span><span>.</span><span>generateSecret</span><span>(</span><span>spec</span><span>).</span><span>getEncoded</span><span>();</span>
    <span>return</span> <span>Base64</span><span>.</span><span>getEncoder</span><span>().</span><span>encodeToString</span><span>(</span><span>hash</span><span>);</span>
<span>}</span>
/** * Generates a cryptographically secure 512 bytes string that can be used as a password salt */ public static String generateSalt() { byte[] bytes = new byte[512]; secureRandom.nextBytes(bytes); return Base64.getEncoder().encodeToString(bytes); } /** * Generate a hash for a password */ public static String generateHashForPassword(String password, String salt) throws NoSuchAlgorithmException, InvalidKeySpecException { SecretKeyFactory secretKeyFactory = SecretKeyFactory.getInstance("PBKDF2WithHmacSHA512"); PBEKeySpec spec = new PBEKeySpec(password.toCharArray(), salt.getBytes(), PBKDF2_ITERATIONS, HASH_BYTE_SIZE); byte[] hash = secretKeyFactory.generateSecret(spec).getEncoded(); return Base64.getEncoder().encodeToString(hash); }

Enter fullscreen mode Exit fullscreen mode

原文链接:How to properly store a password in the Database

© 版权声明
THE END
喜欢就支持一下吧
点赞7 分享
Don’t worry about what others are doing better than you. Concentrate on beating your own records every day.
不要担心别人会做得比你好。你只需要每天都做得比前一天好就可以了
评论 抢沙发

请登录后发表评论

    暂无评论内容