回答編集履歴
1
sample
test
CHANGED
@@ -7,3 +7,119 @@
|
|
7
7
|
- ワンタイムパスを発行する
|
8
8
|
|
9
9
|
- 本文の一致をチェックする
|
10
|
+
|
11
|
+
|
12
|
+
|
13
|
+
#sample;
|
14
|
+
|
15
|
+
とりあえずワンタイムパスを利用した二重投稿防止のサンプルつけときます
|
16
|
+
|
17
|
+
otpカラムがuniqueで二重投稿を防止しています。
|
18
|
+
|
19
|
+
- テーブル構造
|
20
|
+
|
21
|
+
```SQL
|
22
|
+
|
23
|
+
create table XXX(id int not null primary key auto_increment,name varchar(10),comment text,otp varchar(100) not null unique);
|
24
|
+
|
25
|
+
```
|
26
|
+
|
27
|
+
|
28
|
+
|
29
|
+
```PHP
|
30
|
+
|
31
|
+
<form method="POST" action="<?php echo $_SERVER['SCRIPT_NAME']; ?>">
|
32
|
+
|
33
|
+
<input type="text" name="name" placeholder="名前"><br>
|
34
|
+
|
35
|
+
<input type="text" name="comment" placeholder="コメント">
|
36
|
+
|
37
|
+
<input type="hidden" name="otp" value="<?PHP print md5(microtime());?>">
|
38
|
+
|
39
|
+
<input type="submit" value="投稿">
|
40
|
+
|
41
|
+
</form>
|
42
|
+
|
43
|
+
|
44
|
+
|
45
|
+
<?php
|
46
|
+
|
47
|
+
$otp = filter_input(INPUT_POST,"otp");
|
48
|
+
|
49
|
+
$name = filter_input(INPUT_POST,"name");
|
50
|
+
|
51
|
+
$comment = filter_input(INPUT_POST,"comment");
|
52
|
+
|
53
|
+
try {
|
54
|
+
|
55
|
+
$pdo = new PDO($dsn, $user, $password);
|
56
|
+
|
57
|
+
$pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
|
58
|
+
|
59
|
+
$pdo->setAttribute(PDO::ATTR_DEFAULT_FETCH_MODE, PDO::FETCH_ASSOC);
|
60
|
+
|
61
|
+
if(!is_null($otp)){
|
62
|
+
|
63
|
+
$sql = "INSERT INTO XXX (name, comment,otp) VALUES (?,?,?)";
|
64
|
+
|
65
|
+
$stm = $pdo->prepare($sql);
|
66
|
+
|
67
|
+
$data=[$name,$comment,$otp];
|
68
|
+
|
69
|
+
if($stm->execute($data)){
|
70
|
+
|
71
|
+
echo '追加エラーがありました';
|
72
|
+
|
73
|
+
}
|
74
|
+
|
75
|
+
}
|
76
|
+
|
77
|
+
$sql = "SELECT * FROM XXX ORDER BY id DESC LIMIT 10";
|
78
|
+
|
79
|
+
$stm = $pdo->query($sql);
|
80
|
+
|
81
|
+
$result = $stm->fetchAll();
|
82
|
+
|
83
|
+
foreach ($result as $row) {
|
84
|
+
|
85
|
+
echo "<div>";
|
86
|
+
|
87
|
+
echo $row['id'].")";
|
88
|
+
|
89
|
+
echo htmlspecialchars($row['name']);
|
90
|
+
|
91
|
+
echo htmlspecialchars($row['comment']);
|
92
|
+
|
93
|
+
echo "</div>";
|
94
|
+
|
95
|
+
}
|
96
|
+
|
97
|
+
} catch (Exception $e) {
|
98
|
+
|
99
|
+
echo 'エラーがありました。<br>';
|
100
|
+
|
101
|
+
echo $e->getMessage();
|
102
|
+
|
103
|
+
}
|
104
|
+
|
105
|
+
?>
|
106
|
+
|
107
|
+
</body>
|
108
|
+
|
109
|
+
```
|
110
|
+
|
111
|
+
気がついた点
|
112
|
+
|
113
|
+
- 空のidがhiddenで設定されていますが、なにか意味があるのでしょうか?
|
114
|
+
|
115
|
+
- 今回例示の処理でinsertが失敗してもidはインクリメントされます
|
116
|
+
|
117
|
+
- SELECTにWHERE句がなかったのでprepare処理する必要はありません。
|
118
|
+
|
119
|
+
逆にユーザーから送られてきた任意の条件で絞り込むならprepareで処理してください
|
120
|
+
|
121
|
+
- $_SERVER["PHP_SELF"]はセキュリティの関係で現在はあまり使われていません
|
122
|
+
|
123
|
+
- ユーザーから送らてきたデータをHTML上に表示するときにはかならずhtmlspecialcharsなどを通してください
|
124
|
+
|
125
|
+
- コメントなど文書はnl2br()的な処理も必要かもしれません
|