I stand up for children in need. Please join me in helping this family.

Skip to content
Steven Roland

Implementing Per-Form CSRF Tokens in PHP

PHP , HTML

Building on our previous discussion of simple CSRF tokens, this post will focus on implementing per-form CSRF tokens in PHP. This approach enhances security by generating a unique token for each form, reducing the risk of token reuse and potential attacks.

Why Use Per-Form CSRF Tokens?

Per-form CSRF tokens provide an additional layer of security by ensuring that each form submission is uniquely validated. This method is particularly effective in mitigating the impact of token leakage, as an attacker would only be able to exploit a specific form or action.

Generating Per-Form CSRF Tokens

To implement per-form tokens, we need to generate a unique token for each form and store it in the session with a unique identifier. Here’s how you can achieve this:

1. Token Generation and Storage

For each form, generate a secure token and store it in the session with a specific key associated with the form:

session_start();

function generateFormToken($formName) {
    $token = bin2hex(random_bytes(32));
    $_SESSION['csrf_tokens'][$formName] = $token;
    return $token;
}

$formName = 'contact_form';
$token = generateFormToken($formName);

2. Embedding the Token in Forms

Include the generated token in the form as a hidden input field. This ensures that the token is sent along with the form data:

<form method="post" action="process_form.php">
    <!-- Other form fields -->
    <input type="hidden" name="csrf_token" value="<?php echo htmlspecialchars($token); ?>">
    <input type="hidden" name="form_name" value="<?php echo htmlspecialchars($formName); ?>">
    <input type="submit" value="Submit">
</form>

3. Validating the Token

Upon form submission, validate the token by checking it against the session-stored token for the specific form:

session_start();

function validateFormToken($formName, $receivedToken) {
    if (isset($_SESSION['csrf_tokens'][$formName]) && hash_equals($_SESSION['csrf_tokens'][$formName], $receivedToken)) {
        unset($_SESSION['csrf_tokens'][$formName]); // Invalidate the token after use
        return true;
    }

    return false;
}

if ($_SERVER['REQUEST_METHOD'] === 'POST') {
    $formName = $_POST['form_name'];
    $receivedToken = $_POST['csrf_token'];

    if (validateFormToken($formName, $receivedToken)) {
        // Proceed with the action
    } else {
        // Handle invalid token
        echo "Invalid or expired token.";
    }
}

Advantages of Per-Form Tokens

  • Enhanced Security: Each form has its own token, minimizing the risk of token reuse.

  • Flexibility: Allows handling multiple forms on the same page with distinct tokens.

  • Reduced Impact of Token Leakage: Limits potential exploitation to a specific form or action.

Best Practices

  • Session Management: Ensure secure session handling to prevent session hijacking.

  • HTTPS: Use HTTPS to protect token transmission.

  • Token Expiration: Consider implementing token expiration to further enhance security.

By implementing per-form CSRF tokens, you can significantly bolster the security of your PHP applications, ensuring that each form submission is uniquely validated and protected against potential attacks.

More posts

Understanding the EOS Methodology

The EOS methodology streamlines web development by aligning team vision, clarifying roles, using data-driven decisions, optimizing processes, and enhancing accountability, leading to efficient project execution and improved collaboration.

Enhancing CLI Experiences with Laravel Prompts

Laravel Prompts enhances CLI applications with user-friendly, interactive forms. It offers features like placeholders, real-time validation, and multi-step forms. Use it for interactive installers, configuration wizards, database seeders, custom Artisan commands, and CLI tools. Best practices include clear instructions, appropriate input types, and graceful error handling.

Simplifying Laravel Development with Laravel Sail

Laravel Sail is a lightweight CLI for managing Laravel's Docker development environment. It simplifies running Artisan commands, PHP scripts, tests, and database operations. Key uses include local development, CI/CD pipelines, team collaboration, and multi-version PHP testing. Best practices involve using aliases, customizing services, and regular updates.

"I've learned that people will forget what you said, people will forget what you did, but people will never forget how you made them feel."

Maya Angelou BrainyQuote