广告位
如何防止 PHP 中的 SQL 注入?
作者: 分类:编程技术浏览(649)2021-7-5

问题:

如果用户输入未经修改就插入到 查询中,那么应用程序就容易受到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;--')

可以做些什么来防止这种情况发生?

答案:

无论您使用哪种数据库,避免 SQL 注入攻击的正确方法是正确编码不受信任的输入,以便数据保持数据并且永远不会被SQL 解析器解释为命令。有多种方法可以安全地对数据进行编码。如果您不完全了解细节,则应始终使用准备好的语句和参数化查询。这些是独立于任何参数发送到数据库服务器并由其解析的 SQL 语句。这样攻击者就不可能注入恶意 SQL。

您基本上有两种选择来实现这一目标:

  1. 使用PDO(对于任何受支持的数据库驱动程序):
     $stmt = $pdo->prepare('SELECT * FROM employees WHERE name = :name');
    
     $stmt->execute([ 'name' => $name ]);
    
     foreach ($stmt as $row) {
         // Do something with $row
     }
    
  2. 使用i(用于 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 是通用选项。


正确设置连接

请注意,当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在出现问题时停止。它让开发人员有机会发现catch任何thrown 为PDOExceptions 的错误。

然而,第一行是强制性的setAttribute()它告诉 PDO 禁用模拟准备好的语句并使用真正的准备好的语句。这可以确保语句和值在将其发送到 MySQL 服务器之前不被 解析(使可能的攻击者没有机会注入恶意 SQL)。

尽管您可以charset在构造函数的选项中设置 ,但重要的是要注意“旧”版本的 PHP(5.3.6 之前)会默默地忽略DSN 中的 charset 参数


解释

您传递给的 SQL 语句prepare由数据库服务器解析和编译。通过指定参数(如上例中的?参数或命名参数:name),您可以告诉数据库引擎您要过滤的位置。然后,当您调用 时execute,准备好的语句将与您指定的参数值组合在一起。

这里重要的是参数值与编译语句组合,而不是 SQL 字符串。SQL 注入的工作原理是在脚本创建要发送到数据库的 SQL 时诱使脚本包含恶意字符串。因此,通过将实际 SQL 与参数分开发送,您可以限制以您不想要的方式结束的风险。

您在使用准备好的语句时发送的任何参数都将被视为字符串(尽管数据库引擎可能会进行一些优化,因此参数当然也可能以数字结尾)。在上面的示例中,如果$name变量包含'Sarah'; DELETE FROM employees结果将只是搜索 string "'Sarah'; DELETE FROM employees",并且您最终不会得到一个空 table

使用准备好的语句的另一个好处是,如果您在同一个会话中多次执行相同的语句,它只会被解析和编译一次,从而提高速度。

哦,既然你问了如何为插入做这件事,这里有一个例子(使用 PDO):

$preparedStatement = $db->prepare('INSERT INTO table (column) VALUES (:column)');

$preparedStatement->execute([ 'column' => $unsafeValue ]);

准备好的语句可以用于动态查询吗?

虽然您仍然可以为查询参数使用准备好的语句,但动态查询本身的结构不能被参数化,并且某些查询特征不能被参数化。

对于这些特定场景,最好的办法是使用限制可能值的白名单过滤器。

// Value whitelist
// $dir can only be 'DESC', otherwise it will be 'ASC'
if (empty($dir) || $dir !== 'DESC') {
   $dir = 'ASC';
}
图片压缩在线工具 tools online