如何防止PHP中进行SQL注入?
如果将用户输入未经修改地插入到SQL查询中,则该应用程序容易受到SQL注入的攻击,如以下示例所示:
$unsafe_variable = $_POST['user_input'];
mysql_query("INSERT INTO `table` (`column`) VALUES ('$unsafe_variable')");
这是因为用户可以输入类似的内容value'); DROP TABLE table;--
,并且查询变为:
INSERT INTO `table` (`column`) VALUES('value'); DROP TABLE table;--')
如何防止这种情况发生?
1 答案1
**使用准备好的语句和参数化查询。**这些是独立于任何参数发送到数据库服务器并由数据库服务器解析的SQL语句。这样,攻击者就不可能注入恶意SQL。
您基本上有两种选择可以实现此目的:
- 使用PDO(对于任何受支持的数据库驱动程序):
$stmt = $pdo->prepare('SELECT * FROM employees WHERE name = :name'); $stmt->execute([ 'name' => $name ]); foreach ($stmt as $row) { // Do something with $row }
- 使用MySQLi(对于MySQL):
$stmt = $dbConnection->prepare('SELECT * FROM employees WHERE name = ?'); $stmt->bind_param('s', $name); // 's' specifies the variable type => 'string' $stmt->execute(); $result = $stmt->get_result(); while ($row = $result->fetch_assoc()) { // Do something with $row }
如果你连接到MySQL之外的数据库,有一个特定的驱动程序,第二个选项,你可以参考一下(例如,pg_prepare()
和pg_execute()
PostgreSQL的)。PDO是通用选项。
2 正确设置连接
请注意,PDO
用于访问MySQL数据库时**,默认情况下不使用**_真实的_预处理语句。要解决此问题,您必须禁用对准备好的语句的仿真。使用PDO创建连接的示例如下:
$dbConnection = new PDO('mysql:dbname=dbtest;host=127.0.0.1;charset=utf8', 'user', 'password');
$dbConnection->setAttribute(PDO::ATTR_EMULATE_PREPARES, false);
$dbConnection->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
在上面的示例中,错误模式不是严格必需的,但建议添加它。这样,Fatal Error
当出现问题时脚本不会以a停止。并且它为开发人员提供了解决catch
任何throw
n为PDOException
s的错误的机会。
但是,第一行是强制性的,setAttribute()
它告诉PDO禁用模拟的预备语句并使用_实际的_预备语句。这可以确保在将语句和值发送到MySQL服务器之前,不会对PHP进行解析(这使可能的攻击者没有机会注入恶意SQL)。
尽管可以charset
在构造函数的选项中设置,但是必须注意,PHP的“较旧”版本(在5.3.6之前)静默忽略了DSN中的charset参数。
3 说明
传递给您的SQL语句prepare
由数据库服务器解析和编译。通过指定参数(例如上例中的a ?
或命名参数:name
),您可以告诉数据库引擎要在何处进行过滤。然后,当您调用时execute
,准备好的语句将与您指定的参数值组合在一起。
这里重要的是参数值与已编译的语句组合,而不是与SQL字符串组合。SQL注入通过在创建要发送到数据库的SQL时欺骗脚本使其包含恶意字符串来起作用。因此,通过将实际的SQL与参数分开发送,可以减少因意外获得最终结果的风险。
使用预处理语句发送的任何参数都将被视为字符串(尽管数据库引擎可能会进行一些优化,因此参数最终也可能以数字结尾)。在上面的示例中,如果$name
变量包含'Sarah'; DELETE FROM employees
结果,则只是搜索字符串"'Sarah'; DELETE FROM employees"
,并且最终不会得到空表。
使用准备好的语句的另一个好处是,如果您在同一会话中多次执行同一条语句,则该语句仅被解析和编译一次,从而使您获得了一些速度上的提高。
哦,既然您询问了如何进行插入,这是一个示例(使用PDO):
$preparedStatement = $db->prepare('INSERT INTO table (column) VALUES (:column)');
$preparedStatement->execute([ 'column' => $unsafeValue ]);
4 准备好的语句可以用于动态查询吗?
尽管您仍可以对查询参数使用准备好的语句,但是无法对动态查询本身的结构进行参数化,并且无法对某些查询功能进行参数化。
对于这些特定方案,最好的办法是使用白名单过滤器来限制可能的值。
// Value whitelist
// $dir can only be 'DESC', otherwise it will be 'ASC'
if (empty($dir) || $dir !== 'DESC') {
$dir = 'ASC';
}
5 答案2
如果您使用的是最新版本的PHP,则mysql_real_escape_string
以下概述的选项将不再可用(尽管mysqli::escape_string
是现代等效项)。如今,该mysql_real_escape_string
选项仅对旧版本PHP上的遗留代码有意义。
您有两个选择-在中转义特殊字符unsafe_variable
,或使用参数化查询。两者都可以保护您免受SQL注入的侵害。参数化查询被认为是更好的做法,但是在使用它之前,需要在PHP中更改为更新的MySQL扩展。
我们将介绍先转义的较低影响字符串。
//Connect
$unsafe_variable = $_POST["user-input"];
$safe_variable = mysql_real_escape_string($unsafe_variable);
mysql_query("INSERT INTO table (column) VALUES ('" . $safe_variable . "')");
//Disconnect
另请参见,mysql_real_escape_string
函数的详细信息。
要使用参数化查询,您需要使用MySQLi而不是MySQL函数。要重写您的示例,我们将需要以下内容。
<?php
$mysqli = new mysqli("server", "username", "password", "database_name");
// TODO - Check that connection was successful.
$unsafe_variable = $_POST["user-input"];
$stmt = $mysqli->prepare("INSERT INTO table (column) VALUES (?)");
// TODO check that $stmt creation succeeded
// "s" means the database expects a string
$stmt->bind_param("s", $unsafe_variable);
$stmt->execute();
$stmt->close();
$mysqli->close();
?>
您将要阅读的关键功能将是mysqli::prepare
。
另外,正如其他人所建议的那样,您可能会发现有用/更容易地使用PDO之类的东西来增强抽象层。
请注意,您所询问的案例是一个相当简单的案例,更复杂的案例可能需要更复杂的方法。尤其是:
- 如果您想根据用户输入更改SQL的结构,则参数化查询将无济于事,并且不能满足要求的转义
mysql_real_escape_string
。在这种情况下,最好将用户的输入通过白名单,以确保仅允许“安全”值通过。 - 如果您在某种情况下使用用户输入中的整数并采用该
mysql_real_escape_string
方法,则将遭受以下多项式描述的多项式问题。这种情况比较棘手,因为整数不会被引号引起来,因此可以通过验证用户输入仅包含数字来进行处理。 - 可能还有其他我不知道的情况。您可能会发现这是一些有用的资源,可解决您可能遇到的一些更细微的问题。
要想成为php领域的高手,必须了解和弄懂php常见问题。要我们来进行php高手修炼吧。或者你有更好的问题可以留言。