org.sonar.l10n.py.rules.python.S2053.html Maven / Gradle / Ivy
This vulnerability increases the likelihood that attackers are able to compute the cleartext of password hashes.
Why is this an issue?
During the process of password hashing, an additional component, known as a "salt," is often integrated to bolster the overall security. This salt,
acting as a defensive measure, primarily wards off certain types of attacks that leverage pre-computed tables to crack passwords.
However, potential risks emerge when the salt is deemed insecure. This can occur when the salt is consistently the same across all users or when it
is too short or predictable. In scenarios where users share the same password and salt, their password hashes will inevitably mirror each other.
Similarly, a short salt heightens the probability of multiple users unintentionally having identical salts, which can potentially lead to identical
password hashes. These identical hashes streamline the process for potential attackers to recover clear-text passwords. Thus, the emphasis on
implementing secure, unique, and sufficiently lengthy salts in password-hashing functions is vital.
What is the potential impact?
Despite best efforts, even well-guarded systems might have vulnerabilities that could allow an attacker to gain access to the hashed passwords.
This could be due to software vulnerabilities, insider threats, or even successful phishing attempts that give attackers the access they need.
Once the attacker has these hashes, they will likely attempt to crack them using a couple of methods. One is brute force, which entails trying
every possible combination until the correct password is found. While this can be time-consuming, having the same salt for all users or a short salt
can make the task significantly easier and faster.
If multiple users have the same password and the same salt, their password hashes would be identical. This means that if an attacker successfully
cracks one hash, they have effectively cracked all identical ones, granting them access to multiple accounts at once.
A short salt, while less critical than a shared one, still increases the odds of different users having the same salt. This might create clusters
of password hashes with identical salt that can then be attacked as explained before.
With short salts, the probability of a collision between two users' passwords and salts couple might be low depending on the salt size. The shorter
the salt, the higher the collision probability. In any case, using longer, cryptographically secure salt should be preferred.
Exceptions
To securely store password hashes, it is a recommended to rely on key derivation functions that are computationally intensive. Examples of such
functions are:
- Argon2
- PBKDF2
- Scrypt
- Bcrypt
When they are used for password storage, using a secure, random salt is required.
However, those functions can also be used for other purposes such as master key derivation or password-based pre-shared key generation. In those
cases, the implemented cryptographic protocol might require using a fixed salt to derive keys in a deterministic way. In such cases, using a fixed
salt is safe and accepted.
How to fix it in Python Standard Library
Code examples
The following code contains examples of hard-coded salts.
Noncompliant code example
import hashlib
hash = hashlib.scrypt(password, salt=b"F3MdWpeHeeSjlUxvKBnzzA", n=2**17, r=8, p=1) # Noncompliant
Compliant solution
import hashlib
import secrets
salt = secrets.token_bytes(32)
hash = hashlib.scrypt(password, salt=salt, n=2**17, r=8, p=1)
How does this work?
This code ensures that each user’s password has a unique salt value associated with it. It generates a salt randomly and with a length that
provides the required security level. It uses a salt length of at least 32 bytes (256 bits), as recommended by industry standards.
Here, the compliant code example ensures the salt is random and has a sufficient length by calling the secrets.token_bytes
function.
This one internally uses a cryptographically secure pseudo random number generator.
How to fix it in pyca
Code examples
The following code contains examples of hard-coded salts.
Noncompliant code example
from cryptography.hazmat.primitives.kdf.pbkdf2 import PBKDF2HMAC
from cryptography.hazmat.primitives import hashes
digest = PBKDF2HMAC(hashes.SHA256(), length=32, salt=b"F3MdWpeHeeSjlUxvKBnzzA", iterations=100000).derive(password)
Compliant solution
import secrets
from cryptography.hazmat.primitives.kdf.pbkdf2 import PBKDF2HMAC
from cryptography.hazmat.primitives import hashes
salt = secrets.token_bytes(32)
digest = PBKDF2HMAC(hashes.SHA256(), length=32, salt=salt, iterations=100000).derive(password)
How does this work?
This code ensures that each user’s password has a unique salt value associated with it. It generates a salt randomly and with a length that
provides the required security level. It uses a salt length of at least 32 bytes (256 bits), as recommended by industry standards.
Here, the compliant code example ensures the salt is random and has a sufficient length by calling the secrets.token_bytes
function.
This one internally uses a cryptographically secure pseudo random number generator.
How to fix it in Cryptodome
Code examples
The following code contains examples of hard-coded salts.
Noncompliant code example
from Crypto.Protocol.KDF import scrypt
digest = scrypt(password, salt=b"F3MdWpeHeeSjlUxvKBnzzA", key_len=32, N=2**17, r=8, p=1) # Noncompliant
Compliant solution
import secrets
from Crypto.Protocol.KDF import scrypt
salt = secrets.token_bytes(32)
digest = scrypt(password, salt=salt, key_len=32, N=2**17, r=8, p=1)
How does this work?
This code ensures that each user’s password has a unique salt value associated with it. It generates a salt randomly and with a length that
provides the required security level. It uses a salt length of at least 32 bytes (256 bits), as recommended by industry standards.
Here, the compliant code example ensures the salt is random and has a sufficient length by calling the secrets.token_bytes
function.
This function internally uses a cryptographically secure pseudo-random number generator.
Resources
Standards
- OWASP - Top 10 2021 Category A2 - Cryptographic Failures
- OWASP - Top 10 2017 Category A3 - Sensitive Data
Exposure
- CWE - CWE-759 - Use of a One-Way Hash without a Salt
- CWE - CWE-760 - Use of a One-Way Hash with a Predictable Salt
- STIG Viewer - Application Security and
Development: V-222542 - The application must only store cryptographic representations of passwords.