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

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
ajax教程
ajax教程

php中文网为大家带来ajax教程合集,Ajax是一种用于创建快速动态网页的技术。通过在后台与服务器进行少量数据交换,Ajax可以使网页实现异步更新。这意味着可以在不重新加载整个网页的情况下,对网页的某部分进行更新。php中文网还为大家带来ajax的相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

159

2023.06.14

ajax中文乱码解决方法
ajax中文乱码解决方法

ajax中文乱码解决方法有设置请求头部的字符编码、在服务器端设置响应头部的字符编码和使用encodeURIComponent对中文进行编码。本专题为大家提供ajax中文乱码相关的文章、下载、课程内容,供大家免费下载体验。

160

2023.08.31

ajax传递中文乱码怎么办
ajax传递中文乱码怎么办

ajax传递中文乱码的解决办法:1、设置统一的编码方式;2、服务器端编码;3、客户端解码;4、设置HTTP响应头;5、使用JSON格式。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

117

2023.11.15

ajax网站有哪些
ajax网站有哪些

使用ajax的网站有谷歌、维基百科、脸书、纽约时报、亚马逊、stackoverflow、twitter、hacker news、shopify和basecamp等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

235

2024.09.24

curl_exec
curl_exec

curl_exec函数是PHP cURL函数列表中的一种,它的功能是执行一个cURL会话。给大家总结了一下php curl_exec函数的一些用法实例,这个函数应该在初始化一个cURL会话并且全部的选项都被设置后被调用。他的返回值成功时返回TRUE, 或者在失败时返回FALSE。

440

2023.06.14

linux常见下载安装工具
linux常见下载安装工具

linux常见下载安装工具有APT、YUM、DNF、Snapcraft、Flatpak、AppImage、Wget、Curl等。想了解更多linux常见下载安装工具相关内容,可以阅读本专题下面的文章。

178

2023.10.30

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

297

2023.10.25

java入门学习合集
java入门学习合集

本专题整合了java入门学习指南、初学者项目实战、入门到精通等等内容,阅读专题下面的文章了解更多详细学习方法。

1

2026.01.29

热门下载

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

精品课程

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

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