欠落しているmysqli接続リソース/オブジェクトに加えて、スクリプトには他のいくつかの問題があります:
- SQLインジェクションが発生しやすい
- http://docs.phpに示されているように、mysql接続をテストしていません。 .net / mysqli.quickstart.connections
- 一般に、スクリプトにはエラー処理がありません。 mysqli_*関数/メソッドのいずれかが失敗する可能性があります。例えば。
mysqli_num_rows
に関する警告mysqli_query
の戻り値をチェックしないことに関連しています 。 - 関数test_input()は何もテストせず、値を変更します。また、メールアドレスはhtmlspecialchars()などとは何の関係もありません。その関数をドロップするだけです。
- メールアドレスの検証は、明らかななしでは非常に複雑に見えます メリット。
- SELECT / INSERTコンボを使用して、メールアドレスが2回挿入されるのを防ぐ代わりに、一意のインデックス そのフィールドで、mysqlサーバーは確実に重複を防ぎます。
例えば
<?php
define('MYSQL_ER_DUP_KEY', 1022); // see https://dev.mysql.com/doc/refman/5.6/en/error-messages-server.html#error_er_dup_key
$errors = array();
if($_POST) // might be superfluous
{
// simplified email validation
// improve if needed
$email = filter_input(INPUT_POST, 'email', FILTER_VALIDATE_EMAIL);
if ( !$email ) {
// removed html/style from error message, better do that when printing the error
$errors['email1'] = "A valid email address is required";
}
// you only need the database connection after the email address is validated
$mysqli = new mysqli('localhost', 'root', '','ecommerce');
// see http://docs.php.net/mysqli.quickstart.connections
if ($mysqli->connect_errno) {
trigger_error("Failed to connect to MySQL: (" . $mysqli->connect_errno . ") " . $mysqli->connect_error, E_USER_ERROR);
}
// not checking if this email address is already in the database
// instead create a unique index for that field
// see https://dev.mysql.com/doc/refman/5.6/en/constraint-primary-key.html
// - otherwise you'd at least have to lock the table to avoid race conditions -
// sql injections: see http://docs.php.net/security.database.sql-injection
// to prevent sql injections you either have to make sure string literals are
// properly encoded/escaped or use preparead statements+parameters
$stmt = $mysqli->prepare('INSERT INTO subscriptions (email) VALUES (?)');
if ( !$stmt ) {
trigger_error("prepare statement failed (" . $mysqli->errno . ") " . $mysqli->error, E_USER_ERROR);
}
else if ( !$stmt->bind_param('s', $email) ) {
trigger_error("bind_param failed (" . $stmt->errno . ") " . $stmt->error, E_USER_ERROR);
}
else if ( !$stmt->execute() ) {
// email has a unique index, inserting an email address a second time
// results in a ER_DUP_KEY error
if ( MYSQL_ER_DUP_KEY==$stmt->errno ) {
$errors['email2'] = "email address already in subsription list";
}
else { // otherwise it's "really" an error
trigger_error("execute failed (" . $stmt->errno . ") " . $stmt->error, E_USER_ERROR);
}
}
else {
[... inserted ...]
}
}