PHPでメールフォームを作成したので、脆弱性がないかアドバイスいただけないでしょうか。
エンジニアでもなければ、PHPもろくに書けない雑魚ですが、「php メールフォーム 作り方」でググって表示されるサイトを見ると、「んんんんん???」と思うところがあります。
これらを参考にしたり、コピペする方は、記述されているコードの良し悪しは判断できないかと思います。
そのような方々が参考にできるメールフォームを作りたいという思いで、調べて作りました。
周りに書いたコードを確認してもらえる人もいないので、皆様からのアドバイスがほしいです((_ _ (´ω` )ペコ
このメールフォームは、下記を対象者としています。
- php メールフォーム 作り方 でググってコピペする方
- コピペして動けばいいと考えている方
- if文や関数など基本的な記述はわかるけど、クラスとか理解していない方
- 脆弱性?なにそれ?という方
そのために、このメールフォームは下記の点を意識して作成しました。
- コピペしたら動く
- なるべく難しい記述はしない、記述量を減らす
- 処理を中断する die を使っていないので、表示したい箇所にコピペすれば動く
- PHP5.2.xまで想定し、環境(magic_quotes_gpc、セーフモード)に依存しない
- 肝心の脆弱性対策はクリックジャッキング、CSRF、XSS、メールヘッダインジェクションを考慮
最終的にはコード+説明した記事を作成してQittaに投稿する予定です。
問題があれば、詳しい方々からアドバイスやご指摘があるはずなので...
皆様から見て、脆弱性や気になる点などあれば、アドバイスもらえると嬉しいです((_ _ (´ω` )ペコ
至らないところがあれば、調べますので、ご指摘だけでも頂けると嬉しいです。
お願いします。
入力(index.php
) → 確認(confirm.php
) → 送信(send.php
) と画面を遷移してメールを送ります。
php
1<?php 2// 他のサイトでインラインフレーム表示を禁止する(クリックジャッキング対策) 3header('X-FRAME-OPTIONS: SAMEORIGIN'); 4 5// セッション開始 6session_start(); 7 8// HTML特殊文字をエスケープする関数 9function h($str) { 10 return htmlspecialchars($str,ENT_QUOTES,'UTF-8'); 11} 12 13/* -------------------------------------------------- 14 トークンの作成(CSRF対策) 15 16 ※使用しているPHPのバージョン・環境にあわせて 17 トークンを選んでね。不要なトークは削除してね。 18-------------------------------------------------- */ 19 20// PHP 7.0 以降 21if(!isset($_SESSION['token'])) { 22 $_SESSION['token'] = bin2hex(random_bytes(32)); 23} 24 25// PHP 5.3 ~ 5.x ※OPENSSL導入済 26if(!isset($_SESSION['token'])) { 27 $_SESSION['token'] = bin2hex(openssl_random_pseudo_bytes(32)); 28} 29 30// PHP 5.3 未満 31if(!isset($_SESSION['token'])) { 32 $_SESSION['token'] = hash('sha256', session_id()); 33} 34 35// トークンを代入 36$token = $_SESSION['token']; 37 38?> 39<!DOCTYPE html> 40<html lang="ja"> 41<head> 42<meta charset="UTF-8"> 43<title>入力画面</title> 44</head> 45<body> 46 47<h1>お問い合わせ(入力画面)</h1> 48 49<form method="post" action="confirm.php"> 50<table> 51 <tr> 52 <th>お名前(必須)</th> 53 <td><input type="text" name="name"></td> 54 </tr> 55 <tr> 56 <th>ふりがな(必須)</th> 57 <td><input type="text" name="ruby"></td> 58 </tr> 59 <tr> 60 <th>メールアドレス(必須)</th> 61 <td><input type="text" name="mail"></td> 62 </tr> 63 <tr> 64 <th>内容(必須)</th> 65 <td><textarea name="content"></textarea></td> 66 </tr> 67</table> 68<input type="hidden" name="token" value="<?php echo h($token); ?>"> 69<button>送信内容確認</button> 70</form> 71</body> 72</html>
php
1<?php 2// 他のサイトでインラインフレーム表示を禁止する(クリックジャッキング対策) 3header('X-FRAME-OPTIONS: SAMEORIGIN'); 4 5// セッション開始 6session_start(); 7 8// HTML特殊文字をエスケープする関数 9function h($str) { 10 return htmlspecialchars($str,ENT_QUOTES,'UTF-8'); 11} 12?> 13<!DOCTYPE html> 14<html lang="ja"> 15<head> 16<meta charset="UTF-8"> 17<title>確認画面</title> 18</head> 19<body> 20 21<h1>お問い合わせ(確認画面)</h1> 22 23<?php 24 25// セッション変数がなければ、空文字を代入 26if(!isset($_SESSION['token'])) { 27 $_SESSION['token'] = ''; 28} 29 30// POSTされたデータを変数に代入(magic_quotes_gpc = On + NULLバイト 対策) 31foreach (array('token','name','ruby','mail','content') as $v) { 32 $$v = filter_input(INPUT_POST, $v, FILTER_DEFAULT, FILTER_FLAG_STRIP_LOW); 33} 34 35// トークンを確認し、確認画面を表示 36if($token !== $_SESSION['token']) { 37 38 echo '<p>お問い合わせの手順に誤りがあります。<br>お手数ですが、最初からやり直してください。</p>'; 39 40} else { 41 42 $error_flag = 0; 43 44 // 必須項目は未入力をチェック 45 if ($name === '') { 46 echo '<p>お名前をご入力してください。</p>'; 47 $error_flag = 1; 48 } elseif (mb_strlen($name) > 50) { 49 echo '<p>お名前は 50 文字以内で入力してください。</p>'; 50 $error_flag = 1; 51 } 52 53 if ($ruby === '') { 54 echo '<p>ふりがなをご入力してください。</p>'; 55 $error_flag = 1; 56 } elseif (mb_strlen($ruby) > 50) { 57 echo '<p>ふりがなは 50 文字以内で入力してください。</p>'; 58 $error_flag = 1; 59 } 60 61 if ($mail === '') { 62 echo '<p>メールアドレスをご入力してください。</p>'; 63 $error_flag = 1; 64 } elseif (mb_strlen($mail) > 100) { 65 echo '<p>メールアドレスは 100 文字以内で入力してください。</p>'; 66 $error_flag = 1; 67 } elseif (!filter_var($mail, FILTER_VALIDATE_EMAIL)) { 68 echo '<p>メールアドレスの形式が正しくありません。</p>'; 69 $error_flag = 1; 70 } 71 72 if ($content === '') { 73 echo '<p>内容をご入力してください。</p>'; 74 $error_flag = 1; 75 } elseif (mb_strlen($content) > 500) { 76 echo '<p>内容は 500 文字以内で入力してください。</p>'; 77 $error_flag = 1; 78 } 79 80 // エラーがある場合は、戻るボタンを表示し、エラーがない場合は、確認画面を表示 81 if ($error_flag === 1) { 82 echo '<button onClick="history.back(); return false;">戻る</button>'; 83 } else { 84 // セッション変数に代入 85 $_SESSION['name'] = $name; 86 $_SESSION['ruby'] = $ruby; 87 $_SESSION['mail'] = $mail; 88 $_SESSION['content'] = $content; 89 90 // 確認用画面の表示 91 ?> 92 93 <form method="post" action="send.php"> 94 <table> 95 <tr> 96 <th>お名前</th> 97 <td><?php echo h($name); ?></td> 98 </tr> 99 <tr> 100 <th>ふりがな</th> 101 <td><?php echo h($ruby); ?></td> 102 </tr> 103 <tr> 104 <th>メールアドレス</th> 105 <td><?php echo h($mail); ?></td> 106 </tr> 107 <tr> 108 <th>内容</th> 109 <td><?php echo nl2br(h($content)); ?></td> 110 </tr> 111 </table> 112 <input type="hidden" name="token" value="<?php echo h($token); ?>"> 113 <button>送信</button> 114 </form> 115 116<?php 117 } 118} 119?> 120</body> 121</html>
php
1<?php 2// 他のサイトでインラインフレーム表示を禁止する(クリックジャッキング対策) 3header('X-FRAME-OPTIONS: SAMEORIGIN'); 4 5// セッション開始 6session_start(); 7 8// mb_send_mail のエンコーディング 9mb_language('ja'); 10 11// 内部文字エンコーディングを設定 12mb_internal_encoding('UTF-8'); 13?> 14<!DOCTYPE html> 15<html lang="ja"> 16<head> 17<meta charset="UTF-8"> 18<title>送信画面</title> 19</head> 20<body> 21 22<h1>お問い合わせ(送信画面)</h1> 23 24<?php 25 26// セッション変数がなければ、空文字を代入 27if(!isset($_SESSION['token'])) { 28 $_SESSION['token'] = ''; 29} 30 31// POST['token']の値をtoken変数に代入 32$token = filter_input(INPUT_POST, 'token', FILTER_DEFAULT, FILTER_FLAG_STRIP_LOW); 33 34// 各セッション変数を各変数に代入 35$name = isset($_SESSION['name']) && is_string($_SESSION['name']) ? $_SESSION['name'] : ''; 36$ruby = isset($_SESSION['ruby']) && is_string($_SESSION['ruby']) ? $_SESSION['ruby'] : ''; 37$mail = isset($_SESSION['mail']) && is_string($_SESSION['mail']) ? $_SESSION['mail'] : ''; 38$content = isset($_SESSION['content']) && is_string($_SESSION['content']) ? $_SESSION['content'] : ''; 39 40// トークンの値が一致しない場合は、エラー文を表示し、一致する場合は送信する 41if($token !== $_SESSION['token']) { 42 43 echo '<p>送信後に再度アクセスされたか、お問い合わせの手順に誤りがあります。<br>お手数ですが、最初の画面からご入力ください。<p>'; 44 45} else { 46 47 /* 運営側へ送信するメールの設定 */ 48 49 // 送信先のメールアドレス 50 $to = 'xxxxxx@xxxxx.xxxx'; 51 // 件名 52 $subject = '【お問い合わせからの送信】○○○○○○○○○○'; 53 // 本文 54 $message = "◆お名前\n$name\n\n◆フリガナ\n$ruby\n\n◆メールアドレス\n$mail\n\n◆内容\n$content"; 55 // オプション 56 $option = '-f'. $to; 57 58 59 /* 問い合わせされた方へ自動返信するメールの設定 */ 60 61 // 件名 62 $auto_subject = '【お問い合わせ】○○○○○○○○○○'; 63 // 送信元のメールアドレス 64 $auto_from = 'From:' . $to; 65 // 本文 66 $auto_message = " 67※このメールは自動返信によるものです。 68 69$name 様 70 71このたびは、お問合せいただき、誠にありがとうございました。 72 73お送りいただきました内容を確認の上、担当者より折り返しご連絡させていただきます。 74 "; 75 76 /* セーフモードがONの場合は、mb_send_mailの第5引数が使えないため、処理を分岐して送信 */ 77 if(ini_get('safe_mode')) { 78 79 /* 運営側と自動返信のメールの送信が完了したら、送信完了の文章を表示する */ 80 if(mb_send_mail($to, $subject, $message, "From:$mail") && mb_send_mail($mail, $auto_subject, $auto_message, $auto_from)) { 81 echo '<p>このたびは、お問合せいただき、誠にありがとうございました。<br>お送りいただきました内容を確認の上、担当者より折り返しご連絡させていただきます。</p>'; 82 } else { 83 echo '<p>大変申し訳ございませんが、メールの送信に失敗しました。<br>お手数ですが最初からやり直してください。</p>'; 84 } 85 86 } else { 87 88 /* 運営側と自動返信のメールの送信が完了したら、送信完了の文章を表示する */ 89 if(mb_send_mail($to, $subject, $message, "From:$mail", $option) && mb_send_mail($mail, $auto_subject, $auto_message, $auto_from, $option)) { 90 echo '<p>このたびは、お問合せいただき、誠にありがとうございました。<br>お送りいただきました内容を確認の上、担当者より折り返しご連絡させていただきます。</p>'; 91 } else { 92 echo '<p>大変申し訳ございませんが、メールの送信に失敗しました。<br>お手数ですが最初からやり直してください。</p>'; 93 } 94 95 } 96 97} 98 99// セッションの破棄 100$_SESSION = array(); 101session_destroy(); 102 103?> 104</body> 105</html>
2017.04.11 追記
徳丸氏による、脆弱性の解説です。
ありがとうございます((_ _ (´ω` )ペコ
回答4件
あなたの回答
tips
プレビュー
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
2017/04/09 10:51
退会済みユーザー
2017/04/09 16:23 編集
2017/04/09 21:42
退会済みユーザー
2017/04/09 23:15