0

0

使用表单密钥保护您的表单

WBOY

WBOY

发布时间:2023-09-07 21:49:14

|

1496人浏览过

|

来源于php中文网

原创

安全是一个热门话题。确保您的网站安全对于任何 web 应用程序都极其重要。事实上,我 70% 的时间都花在保护应用程序上。我们必须保护的最重要的事情之一是表单。今天,我们将回顾一种防止表单上的 xss(跨站脚本)和跨站请求伪造的方法。

为什么?

POST 数据可以从一个网站发送到另一个网站。为什么这样不好?一个简单的场景...

登录到您网站的用户在其会话期间访问另一个网站。该网站将能够将 POST 数据发送到您的网站 - 例如,使用 AJAX。由于用户已登录您的网站,因此其他网站也能够将发布数据发送到只有登录后才能访问的安全表单。

我们还必须保护我们的页面免受使用 cURL 的攻击

我们如何解决这个问题?

带有表单键!我们将向每个表单添加一个特殊的哈希值(表单密钥),以确保数据仅在从您的网站发送时才会被处理。提交表单后,我们的 PHP 脚本将根据我们在会话中设置的表单密钥验证提交的表单密钥。

我们必须做什么:

  1. 为每个表单添加表单键。
  2. 将表单密钥存储在会话中。
  3. 提交表单后验证表单密钥。

第 1 步:简单的表单

首先,我们需要一个简单的表单来进行演示。我们必须保护的最重要的表单之一是登录表单。登录表单容易受到暴力攻击。创建一个新文件,并将其保存为 Web 根目录中的 index.php。在正文中添加以下代码:

	
	
	
		
		Securing forms with form keys
	
	
		

现在我们有了一个带有登录表单的简单 XHTML 页面。如果您想在网站上使用表单键,您可以将上面的脚本替换为您自己的登录页面。现在,让我们继续真正的行动。

第 2 步:创建类

我们将为表单键创建一个 PHP 类。因为每个页面只能包含一个表单键,所以我们可以为我们的类创建一个单例,以确保我们的类被正确使用。因为创建单例是一个更高级的 OOP 主题,所以我们将跳过这一部分。创建一个名为 formkey.class.php 的新文件并将其放置在您的 Web 根目录中。现在我们必须考虑我们需要的功能。首先,我们需要一个函数来生成表单密钥,以便我们可以将其放入表单中。在您的 PHP 文件中放置以下代码:


在上面,您看到一个包含三个部分的类:两个变量和一个函数。我们将该函数设置为私有,因为该函数将仅由我们稍后将创建的输出函数使用。在这两个变量中,我们将存储表单键。它们也是私有的,因为它们只能由我们类内的函数使用。

现在,我们必须想办法生成表单密钥。因为我们的表单密钥必须是唯一的(否则我们没有任何安全性),所以我们使用用户 IP 地址的组合将密钥绑定到用户,使用 mt_rand() 使其唯一,并使用 uniqid() 函数使其更加独特。我们还使用 md5() 加密此信息以创建唯一的哈希值,然后将其插入到我们的页面中。因为我们使用了 md5(),所以用户无法看到我们用来生成密钥的内容。整个功能:

//Function to generate the form key
private function generateKey()
{
	//Get the IP-address of the user
	$ip = $_SERVER['REMOTE_ADDR'];
	
	//We use mt_rand() instead of rand() because it is better for generating random numbers.
	//We use 'true' to get a longer string.
	//See http://www.php.net/mt_rand for a precise description of the function and more examples.
	$uniqid = uniqid(mt_rand(), true);
	
	//Return the hash
	return md5($ip . $uniqid);
}

将上面的代码插入到您的 formkey.class.php 文件中。用新函数替换该函数。

第 3 步:将表单密钥插入表单

对于这一步,我们创建一个新函数,使用表单键输出隐藏的 HTML 字段。该函数由三个步骤组成:

  1. 使用我们的generateKey() 函数生成表单密钥。
  2. 将表单密钥存储在 $formKey 变量和会话中。
  3. 输出 HTML 字段。

我们将函数命名为 outputKey() 并将其公开,因为我们必须在类之外使用它。我们的函数将调用私有函数generateKey()来生成新的表单密钥并将其保存在本地会话中。最后,我们创建 XHTML 代码。现在在我们的 PHP 类中添加以下代码:

//Function to output the form key
public function outputKey()
{
	//Generate the key and store it inside the class
	$this->formKey = $this->generateKey();
	//Store the form key in the session
	$_SESSION['form_key'] = $this->formKey;
	
	//Output the form key
	echo "";
}

现在,我们将把表单密钥添加到我们的登录表单中以确保其安全。我们必须将该类包含在 index.php 文件中。我们还必须启动会话,因为我们的类使用会话来存储生成的密钥。为此,我们在 doctype 和 head 标记上方添加以下代码:


上面的代码非常不言自明。我们启动会话(因为我们存储表单密钥)并加载 PHP 类文件。之后,我们使用new formKey()启动该类,这将创建我们的类并将其存储在$formKey中。现在我们只需编辑表单,使其包含表单键:

outputKey(); ?>
input type="password" name="password" id="password" />

仅此而已!因为我们创建了函数 outputKey(),所以我们只需将它包含在表单中即可。我们可以在每个表单中使用表单键,只需添加 outputKey(); ?> 现在只需查看网页的源代码,您就可以看到表单上附加了一个表单密钥。剩下的唯一步骤是验证请求。

第 4 步:验证

我们不会验证整个表单;只有表单键。验证表单是基本的 PHP 操作,并且可以在网络上找到教程。让我们验证表单密钥。因为我们的“generateKey”函数会覆盖会话值,所以我们向 PHP 类添加一个构造函数。创建(或构造)我们的类时将调用构造函数。在我们创建新密钥之前,构造函数会将前一个密钥存储在类中;所以我们将始终拥有以前的表单密钥来验证我们的表单。如果我们不这样做,我们将无法验证表单密钥。将以下 PHP 函数添加到您的类中:

//The constructor stores the form key (if one exists) in our class variable.
function __construct()
{
	//We need the previous key so we store it
	if(isset($_SESSION['form_key']))
	{
		$this->old_formKey = $_SESSION['form_key'];
	}
}

构造函数应始终命名为__construct()。当调用构造函数时,我们检查是否设置了会话,如果是,我们将其本地存储在 old_formKey 变量中。

现在我们可以验证表单密钥了。我们在类中创建一个基本函数来验证表单密钥。这个函数也应该是公共的,因为我们将在类之外使用它。该函数将根据表单键的存储值验证表单键的 POST 值。将此函数添加到 PHP 类中:

//Function that validated the form key POST data
public function validate()
{
	//We use the old formKey and not the new generated version
	if($_POST['form_key'] == $this->old_formKey)
	{
		//The key is valid, return true.
		return true;
	}
	else
	{
		//The key is invalid, return false.
		return false;
	}
}

index.php中,我们使用刚刚在类中创建的函数来验证表单密钥。当然,我们仅在 POST 请求后进行验证。在 $formKey = new formKey(); 后添加以下代码

$error = 'No error';

//Is request?
if($_SERVER['REQUEST_METHOD'] == 'post')
{
	//Validate the form key
	if(!isset($_POST['form_key']) || !$formKey->validate())
	{
		//Form key is invalid, show an error
		$error = 'Form key error!';
	}
	else
	{
		//Do the rest of your validation here
		$error = 'No form key error!';
	}
}

我们创建了一个变量$error来存储我们的错误消息。如果已发送 POST 请求,我们将使用 $formKey->validate() 验证表单密钥。如果返回 false,则表单键无效,并且我们会显示错误。请注意,我们仅验证表单密钥 - 您需要自己验证表单的其余部分。

在 HTML 中,您可以放置​​以下代码来显示错误消息:

	

这将回显 $error 变量(如果已设置)。

使用表单密钥保护您的表单

如果您启动服务器并转到index.php,您将看到我们的表单和消息“无错误”。当您提交表单时,您将看到消息“无表单键错误”,因为它是有效的 POST 请求。现在尝试重新加载页面并在浏览器请求再次发送 POST 数据时接受。您将看到我们的脚本触发了一条错误消息:“表单键错误!”现在,您的表单可以免受来自其他网站的输入和页面重新加载错误的影响!刷新后也会显示该错误,因为我们提交表单后生成了新的表单密钥。这很好,因为现在用户不会意外地将表单发布两次。

完整代码

以下是完整的 PHP 和 HTML 代码:

index.php

validate())
	{
		//Form key is invalid, show an error
		$error = 'Form key error!';
	}
	else
	{
		//Do the rest of your validation here
		$error = 'No form key error!';
	}
}
?>
	



	
	Securing forms with form keys


	
outputKey(); ?>

fomrkey.class.php

old_formKey = $_SESSION['form_key'];
		}
	}

	//Function to generate the form key
	private function generateKey()
	{
		//Get the IP-address of the user
		$ip = $_SERVER['REMOTE_ADDR'];
		
		//We use mt_rand() instead of rand() because it is better for generating random numbers.
		//We use 'true' to get a longer string.
		//See http://www.php.net/mt_rand for a precise description of the function and more examples.
		$uniqid = uniqid(mt_rand(), true);
		
		//Return the hash
		return md5($ip . $uniqid);
	}

	
	//Function to output the form key
	public function outputKey()
	{
		//Generate the key and store it inside the class
		$this->formKey = $this->generateKey();
		//Store the form key in the session
		$_SESSION['form_key'] = $this->formKey;
		
		//Output the form key
		echo "";
	}

	
	//Function that validated the form key POST data
	public function validate()
	{
		//We use the old formKey and not the new generated version
		if($_POST['form_key'] == $this->old_formKey)
		{
			//The key is valid, return true.
			return true;
		}
		else
		{
			//The key is invalid, return false.
			return false;
		}
	}
}
?>

结论

将此代码添加到您网站上的每个重要表单中将显着提高表单的安全性。它甚至会停止刷新问题,正如我们在步骤 4 中看到的那样。由于表单密钥仅对一个请求有效,因此不可能进行双重发布。

这是我的第一个教程,希望您喜欢它并使用它来提高您的安全性!请通过评论让我知道您的想法。有更好的方法吗?让我们知道。

进一步阅读

  • WordPress 还使用表单键(将其命名为 Nonce):Wordpress Nonce
  • 编写安全 PHP 应用程序的七个习惯
  • 在 Twitter 上关注我们,或订阅 NETTUTS RSS Feed 以获取更多日常 Web 开发教程和文章。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
php文件怎么打开
php文件怎么打开

打开php文件步骤:1、选择文本编辑器;2、在选择的文本编辑器中,创建一个新的文件,并将其保存为.php文件;3、在创建的PHP文件中,编写PHP代码;4、要在本地计算机上运行PHP文件,需要设置一个服务器环境;5、安装服务器环境后,需要将PHP文件放入服务器目录中;6、一旦将PHP文件放入服务器目录中,就可以通过浏览器来运行它。

1992

2023.09.01

php怎么取出数组的前几个元素
php怎么取出数组的前几个元素

取出php数组的前几个元素的方法有使用array_slice()函数、使用array_splice()函数、使用循环遍历、使用array_slice()函数和array_values()函数等。本专题为大家提供php数组相关的文章、下载、课程内容,供大家免费下载体验。

1309

2023.10.11

php反序列化失败怎么办
php反序列化失败怎么办

php反序列化失败的解决办法检查序列化数据。检查类定义、检查错误日志、更新PHP版本和应用安全措施等。本专题为大家提供php反序列化相关的文章、下载、课程内容,供大家免费下载体验。

1216

2023.10.11

php怎么连接mssql数据库
php怎么连接mssql数据库

连接方法:1、通过mssql_系列函数;2、通过sqlsrv_系列函数;3、通过odbc方式连接;4、通过PDO方式;5、通过COM方式连接。想了解php怎么连接mssql数据库的详细内容,可以访问下面的文章。

948

2023.10.23

php连接mssql数据库的方法
php连接mssql数据库的方法

php连接mssql数据库的方法有使用PHP的MSSQL扩展、使用PDO等。想了解更多php连接mssql数据库相关内容,可以阅读本专题下面的文章。

1400

2023.10.23

html怎么上传
html怎么上传

html通过使用HTML表单、JavaScript和PHP上传。更多关于html的问题详细请看本专题下面的文章。php中文网欢迎大家前来学习。

1229

2023.11.03

PHP出现乱码怎么解决
PHP出现乱码怎么解决

PHP出现乱码可以通过修改PHP文件头部的字符编码设置、检查PHP文件的编码格式、检查数据库连接设置和检查HTML页面的字符编码设置来解决。更多关于php乱码的问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1440

2023.11.09

php文件怎么在手机上打开
php文件怎么在手机上打开

php文件在手机上打开需要在手机上搭建一个能够运行php的服务器环境,并将php文件上传到服务器上。再在手机上的浏览器中输入服务器的IP地址或域名,加上php文件的路径,即可打开php文件并查看其内容。更多关于php相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1303

2023.11.13

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号