logo
podcast Podcast
get help Get Unstuck

Why does Laravel set Bcrypt rounds to 4 in testing?

Why not 1?

Joel Clermont
Joel Clermont
2025-04-02

If you look in a default Laravel 12 application, BCRYPT_ROUNDS is set to 12 in the .env.example file, but it's overridden to 4 in the phpunit.xml file, which is used when tests run. Why the difference?

That "cost factor" is a measure of how long it takes to hash a password. The higher the number, the longer it takes to hash. This is important for security, but in testing, we want our tests to run quickly.

In production, 12 rounds is a good default based on the current speed of computers. In the future, as computers get faster, that default number will likely increase to compensate.

But in testing, we don't care how secure a hash is. We drop that number to 4 to speed up the tests. There is no security tradeoff here.

But then the next question might be: Why not drop it even further? Why not set it to 1 or even 0? Wouldn't our tests be even faster still?

Laravel's Hash facade uses the password_hash function under the hood, which is a wrapper around the crypt function. The crypt function requires at least 4 rounds to work properly. You can see in the crypt docs for Blowfish (that's the B in Bcrypt), that the acceptable range is between 4 and 31. If you set it to 1 or 0, it will throw an error Invalid bcrypt cost parameter specified.

As a side note, if you ever bump into this in application code, the error emitted by crypt might get swallowed by the framework, and you'd get a more generic Bcrypt hashing is not supported error instead.

Here to help,

Joel

P.S. When you run into weird errors in your app, and you're not sure what's causing them, having a trusted community of helpful Laravel devs is a great way to get help.

Toss a coin in the jar if you found this helpful.
Want a tip like this in your inbox every weekday? Sign up below 👇🏼
email
No spam. Only real-world advice you can use.