Proper prevention of mail injection in PHP

后端 未结 4 1707
小鲜肉
小鲜肉 2020-11-28 15:17

Could you advise me how do I go about preventing email injection in PHP mail() without losing original message data? E.g. if I need to allow user to use \

相关标签:
4条回答
  • 2020-11-28 15:45

    Suppose you you want to put the email address of the visitor in the optional header field like so:

    $headers = "From: $visitorEmailAddress";
    

    However, if

    $visitorEmailAddress

    contains

    "address@email.com\n\nBCC:spam@v1agra.com"

    you've made yourself a spam host, opening the door for mail injection. This is a very simple example, but creative spammers and malicious hackers can sneak potentially damaging scripts in your email, since email is sent as a plaintext file. Even attachments are converted plaintext, and they can easily send attachements by adding a mimetype content line.

    If your form validation for the FROM and/or TO fields is OK, you have to look at the form validation for the body of the email. I'd strip out the '-=' and '=-' characters, and prevent users from typing plain HTML by using strip_tags().

    0 讨论(0)
  • 2020-11-28 15:48

    Use a designated mime email library, like Mail_Mime:

    <?php
    include 'Mail.php';
    include 'Mail/mime.php' ;
    
    $mime = new Mail_mime();
    
    $mime->setTXTBody("Message goes here");
    $hdrs = $mime->headers(array(
        'From'    => 'you@yourdomain.com',
        'Subject' => 'Test mime message'
    ));
    $body = $mime->get();
    
    $mail = &Mail::factory('mail');
    $mail->send('postmaster@localhost', $hdrs, $body);
    
    ?>
    
    0 讨论(0)
  • 2020-11-28 15:59

    Try this for a review of various options:

    http://www.codeproject.com/Articles/428076/PHP-Mail-Injection-Protection-and-E-Mail-Validatio

    It covers several options and tries to explain the benefits and risks of each.

    0 讨论(0)
  • 2020-11-28 16:01

    To filter valid emails for use in the recipient email field, take a look at filter_var():

    $email = filter_var($_POST['recipient_email'], FILTER_VALIDATE_EMAIL);
    
    if ($email === FALSE) {
        echo 'Invalid email';
        exit(1);
    }
    

    This will make sure your users only supply singular, valid emails, which you can then pass to the mail() function. As far as I know, there's no way to inject headers through the message body using the PHP mail() function, so that data shouldn't need any special processing.

    Update:

    According to the documentation for mail(), when it's talking directly to an SMTP server, you will need to prevent full stops in the message body:

    $body = str_replace("\n.", "\n..", $body);
    

    Update #2:

    Apparently, it's also possible to inject via the subject, as well, but since there is no FILTER_VALIDATE_EMAIL_SUBJECT, you'll need to do the filtering yourself:

    $subject = str_ireplace(array("\r", "\n", '%0A', '%0D'), '', $_POST['subject']);
    
    0 讨论(0)
提交回复
热议问题