I am trying to add some security to the forms on my website. One of the forms uses AJAX and the other is a straightforward "contact us" form. I'm trying to add a CSRF token. The problem I'm having is that the token is only showing up in the HTML "value" some of the time. The rest of the time, the value is empty. Here is the code I am using on the AJAX form:
PHP:
if (!isset($_SESSION)) {
session_start();
$_SESSION['formStarted'] = true;
}
if (!isset($_SESSION['token']))
{$token = md5(uniqid(rand(), TRUE));
$_SESSION['token'] = $token;
}
HTML
<form>
//...
<input type="hidden" name="token" value="<?php echo $token; ?>" />
//...
</form>
Any suggestions?
For security code, please don't generate your tokens this way:
$token = md5(uniqid(rand(), TRUE));
rand()
is predictableuniqid()
only adds up to 29 bits of entropymd5()
doesn't add entropy, it just mixes it deterministicallyTry this out:
Generating a CSRF Token
PHP 7
Sidenote: One of my employer's open source projects is an initiative to backport
random_bytes()
andrandom_int()
into PHP 5 projects. It's MIT licensed and available on Github and Composer as paragonie/random_compat.PHP 5.3+ (or with ext-mcrypt)
Verifying the CSRF Token
Don't just use
==
or even===
, usehash_equals()
(PHP 5.6+ only, but available to earlier versions with the hash-compat library).Going Further with Per-Form Tokens
You can further restrict tokens to only be available for a particular form by using
hash_hmac()
. HMAC is a particular keyed hash function that is safe to use, even with weaker hash functions (e.g. MD5). However, I recommend using the SHA-2 family of hash functions instead.First, generate a second token for use as an HMAC key, then use logic like this to render it:
And then using a congruent operation when verifying the token:
The tokens generated for one form cannot be reused in another context without knowing
$_SESSION['second_token']
. It is important that you use a separate token as an HMAC key than the one you just drop on the page.Bonus: Hybrid Approach + Twig Integration
Anyone who uses the Twig templating engine can benefit from a simplified dual strategy by adding this filter to their Twig environment:
With this Twig function, you can use both the general purpose tokens like so:
Or the locked down variant:
Twig is only concerned with template rendering; you still must validate the tokens properly. In my opinion, the Twig strategy offers greater flexibility and simplicity, while maintaining the possibility for maximum security.
Single-Use CSRF Tokens
If you have a security requirement that each CSRF token is allowed to be usable exactly once, the simplest strategy regenerate it after each successful validation. However, doing so will invalidate every previous token which doesn't mix well with people who browse multiple tabs at once.
Paragon Initiative Enterprises maintains an Anti-CSRF library for these corner cases. It works with one-use per-form tokens, exclusively. When enough tokens are stored in the session data (default configuration: 65535), it will cycle out the oldest unredeemed tokens first.
Looks like you need an else with your if.
The variable
$token
is not being retrieved from the session when it's in there