## 一,创建middelware和controller
~~~
liuhongdi@lhdpc:/data/php/admapi$ php think make:middleware CheckSign
Middleware:app\middleware\CheckSign created successfully.
~~~
~~~
liuhongdi@lhdpc:/data/php/admapi$ php think make:controller Pay
Controller:app\controller\Pay created successfully.
~~~
说明:刘宏缔的架构森林是一个专注架构的博客,地址:[https://www.cnblogs.com/architectforest](https://www.cnblogs.com/architectforest)
对应的源码可以访问这里获取: [https://github.com/liuhongdi/
](https://github.com/liuhongdi/) 或: [https://gitee.com/liuhongdi](https://gitee.com/liuhongdi)
## 二,编写php代码:
1,middleware/CheckSign.php
~~~
<?php
declare (strict_types = 1);
namespace app\middleware;
use app\result\Result;
use app\lib\util\sign;
class CheckSign
{
//第三方对接口的访问不需要检查
private $notCheck = ["/pay/alipayreturn","/pay/alipayrotify"];
/**
* 处理请求
*
* @param \think\Request $request
* @param \Closure $next
* @return Response
*/
public function handle($request, \Closure $next)
{
$uri = $request->server("REQUEST_URI");
//如果是不需检查的uri
if (in_array($uri ,$this->notCheck)) {
return $next($request);
} else {
$sign = new sign();
$param = $request->request();
unset($param['s']);
//检查参数的sign
$res = $sign->verifySign($param);
if ($res['code'] == 0){
return $next($request);
} else {
return Result::Error(1,$res['msg']);
}
}
}
}
~~~
2,lib/util/sign.php
~~~
<?php
namespace app\lib\util;
class sign {
private $appId = "lc20220118";
private $appKey = "u665698fzur5e2t85tyu142";
//创建sign
public function makeSign($data) {
ksort($data);
$string = $this->toUrlParams($data);
$string = $string . "&key=" . $this->appKey;
$string = md5($string);
$result = strtolower($string);
return $result;
}
//检验sign是否正确
public function verifySign($data) {
//check sign
if (!isset($data['sign']) || !$data['sign']) {
return ['code'=>1,'msg'=>'发送的数据签名不存在'];
}
//check sign
if (!isset($data['appid']) || !$data['appid']) {
return ['code'=>1,'msg'=>'发送的应用参数1不存在'];
}
if ($data['appid'] != $this->appId) {
return ['code'=>1,'msg'=>'发送的应用参数1错误'];
}
//check sign
if (!isset($data['nonce']) || !$data['nonce']) {
return ['code'=>1,'msg'=>'发送的应用参数2不存在'];
}
//check timestamp
if (!isset($data['timestamp']) || !$data['timestamp']) {
return ['code'=>1,'msg'=>'发送的数据参数不合法'];
}
// 验证请求, 10分钟失效
if (time() - $data['timestamp'] > 600) {
return ['code'=>1,'msg'=>'验证超时, 请重新发送请求'];
}
$clientSign = $data['sign'];
unset($data['sign']);
$serverSign = $this->makeSign($data);
if ($clientSign == $serverSign) {
return ['code'=>0,'msg'=>'验证通过'];
} else {
return ['code'=>1,'msg'=>'请求不合法'];
}
}
//生成url字符串
private function toUrlParams($values){
$buff = "";
foreach ($values as $k => $v)
{
//&& $v != ""
if($k != "sign" && !is_array($v)){
$buff .= $k . "=" . $v . "&";
}
}
$buff = trim($buff, "&");
return $buff;
}
}
~~~
3,controller/Pay.php
~~~
<?php
declare (strict_types = 1);
namespace app\controller;
use think\Request;
class Pay
{
/**
* return
*
* @return \think\Response
*/
public function alipayReturn()
{
echo "alipayReturn";
exit;
}
/**
* notify
*
* @return \think\Response
*/
public function alipayNotify()
{
echo "alipayNotify";
exit;
}
}
~~~
## 三,编写vue代码:
说明:vue代码仅供演示,因为js中无法安全保存appkey,
所以web/wap等不需要做验证:
~~~
<template>
<div style="text-align: left;">
id:{{article.id}}<br/>
title:{{article.title}}<br/>
content:{{article.content}}<br/>
</div>
</template>
<script>
import {apiArticleOne} from "../api/api";
import {ElMessage} from "element-plus";
import {ref} from "vue";
import md5 from 'js-md5';
export default {
name: "Article",
setup() {
const article = ref([]);
//得到sign
const getSign = (param,key) => {
let str = "";
for(var idx in param) {
let one = idx+"="+param[idx];
if (str == "") {
str = one;
} else {
str = str+"&"+one;
}
}
str = str+"&key="+key;
//console.log("before md5:"+str);
let md5str = md5(str);
return md5str;
}
//得到参数对象
const getParam = (id) => {
let appid = "lc20220118";
let appkey = "u665698fzur5e2t85tyu1421";
let timestamp = parseInt((new Date()).getTime()/1000);
let nonce = Math.floor(Math.random()*8999)+1000;
let param = {
appid:appid,
id:id,
nonce:nonce,
timestamp:timestamp,
}
let sign = getSign(param,appkey);
param.sign = sign;
return param;
}
//查询得到一篇文章的内容:
const getArticle = () => {
var param = getParam(1);
apiArticleOne(param).then(res => {
//成功
if (res.code == 0) {
article.value = res.data;
} else {
ElMessage.error("获取表单令牌失败:"+res.msg);
}
}).catch((error) => {
console.log(error)
})
}
getArticle();
return {
article,
}
}
}
</script>
<style scoped>
</style>
~~~
## 四,测试效果:
1,成功返回数据时:
![](https://img.kancloud.cn/55/69/5569b1113b0ef981c6e593d3be37e977_1198x472.png)
2,appkey不正确时:
![](https://img.kancloud.cn/81/cf/81cf1920f5250d29e9a199ecf2b05f85_988x312.png)
3,直接访问接口时报错:
![](https://img.kancloud.cn/8c/ef/8cef16ebb2d5969d3fe7a9043acea3a1_1014x274.png)
4,访问不做check的接口:
![](https://img.kancloud.cn/ed/73/ed7358f0229e0af814b0af3cfe09b11c_902x162.png)
## 五,查看php和thinkphp的版本:
php:
- thinkphp6执行流程(一)
- php中use关键字用法详解
- Thinkphp6使用腾讯云发送短信步骤
- 路由配置
- Thinkphp6,static静态资源访问路径问题
- ThinkPHP6.0+ 使用Redis 原始用法
- smarty在thinkphp6.0中的最佳实践
- Thinkphp6.0 搜索器使用方法
- 从已有安装包(vendor)恢复 composer.json
- tp6with的用法,表间关联查询
- thinkphp6.x多对多如何添加中间表限制条件
- thinkphp6 安装JWT
- 缓存类型
- 请求信息和HTTP头信息
- 模型事件用法
- 助手函数汇总
- tp6集成Alipay 手机和电脑端支付的方法
- thinkphp6使用jwt
- 6.0session cookie cache
- tp6笔记
- TP6(thinkphp6)队列与延时队列
- thinkphp6 command(自定义指令)
- command(自定义指令)
- 本地文件上传
- 缓存
- 响应
- 公共函数配置
- 七牛云+文件上传
- thinkphp6:访问多个redis数据源(thinkphp6.0.5 / php 7.4.9)
- 富文本编辑器wangEditor3
- IP黑名单
- 增删改查 +文件上传
- workerman 定时器操作控制器的方法
- 上传文件到阿里云oss
- 短信或者邮箱验证码防刷代码
- thinkphp6:访问redis6(thinkphp 6.0.9/php 8.0.14)
- 实现关联多个id以逗号分开查询数据
- thinkphp6实现邮箱注册功能的细节和代码(点击链接激活方式)
- 用mpdf生成pdf文件(php 8.1.1 / thinkphp v6.0.10LTS )
- 生成带logo的二维码(php 8.1.1 / thinkphp v6.0.10LTS )
- mysql数据库使用事务(php 8.1.1 / thinkphp v6.0.10LTS)
- 一,创建过滤IP的中间件
- 源码解析请求流程
- 验证码生成
- 权限管理
- 自定义异常类
- 事件监听event-listene
- 安装与使用think-addons
- 事件与多应用
- Workerman 基本使用
- 查询用户列表按拼音字母排序
- 扩展包合集
- 查询用户数据,但是可以通过输入用户昵称来搜索用户同时还要统计用户的文章和粉丝数
- 根据图片的minetype类型获取文件真实拓展名思路
- 到处excel
- 用imagemagick库生成缩略图
- 生成zip压缩包并下载
- API 多版本控制
- 用redis+lua做限流(php 8.1.1 / thinkphp v6.0.10LTS )
- 【thinkphp6源码分析三】 APP类之父, 容器Container类
- thinkphp6表单重复提交解决办法
- 小程序授权
- 最简单的thinkphp6导出Excel
- 根据访问设备不同访问不同模块
- 服务系统
- 前置/后置中间件
- 给接口api做签名验证(php 8.1.1 / thinkphp v6.0.10LTS )
- 6实现邮箱注册功能的细节和代码(点击链接激活方式)
- 使用前后端分离的验证码(thinkphp 6.0.9/php 8.0.14/vue 3.2.26)
- 前后端分离:用jwt+middleware做用户登录验证(php 8.1.1 / thinkphp v6.0.10LTS )
- vue前后端分离多图上传
- thinkphp 分组、页面跳转与ajax
- thinkphp6 常用方法文档
- 手册里没有的一些用法
- Swagger 3 API 注释
- PHP 秒级定时任务
- thinkphp6集成gatewayWorker(workerman)实现实时监听
- thinkphp6按月新增数据表
- 使用redis 实现消息队列
- api接口 统一结果返回处理类
- 使用swoole+thinkphp6.0+redis 结合开发的登录模块
- 给接口api做签名验证
- ThinkPHP6.0 + UniApp 实现小程序的 微信登录
- ThinkPHP6.0 + Vue + ElementUI + axios 的环境安装到实现 CURD 操作!
- 异常$e
- 参数请求验证自定义和异常错误自定义