经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » PHP » 查看文章
PHP转Go系列 | ThinkPHP与Gin框架之API接口签名设计实践
来源:cnblogs  作者:Yxh_blogs  时间:2024/7/8 9:54:23  对本文有异议

大家好,我是码农先森。

回想起以前用模版渲染数据的岁月,那时都没有 API 接口开发的概念。PHP 服务端和前端 HTML、CSS、JS 代码混合式开发,也不分前端、后端程序员,大家都是全干工程师。随着前后端分离、移动端开发的兴起,用后端渲染数据的开发方式效率低下,已经不能满足业务对需求快速上线的要求了。于是为了前后端的高效协同开发引入了 API 接口,只要在开发需求之前约定好数据传参,之后便可以开始启动自己的开发任务且互不干涉,最后再进行统一的接口联调。

根据熵增原则,如果任何事情不加以规则来限制,则都会朝着泛滥的方式发展。同样 API 接口开发也会出现这样的情况,由于每个人的开发习惯不同,导致 API 接口的开发格式五花八门,联调过程困难重重。无规矩不成方圆,因此为了规范 API 接口开发的形式,同时也结合我平时的项目开发经验。总结了一些 API 接口开发的实践经验,希望对大家能有所帮助。

话不多说,开整!

这次主要的实践内容是 API 接口签名设计,以下是一些关键的步骤:

  • 给前端分配一个 AppKey,这个 AppKey 需要带在 HTTP Header 头中进行传输。
  • 在前端的传参中需要额外增加 时间戳 timestamp、随机字符串 nonce 参数。
  • 将前端的所有参数排序后拼接成一个字符串,再使用 MD5 加密函数生成 sign 签名字符串。
  • 服务端接收到参数后,先验证 AppKey 是否一致。
  • 再验证前端所传的时间戳参数是否还在有效期。
  • 之后在服务端使用同样的加密算法生成 sign 签名串,再与前端的 sign 签名串比对。
  • 最后判断前端所传的随机字符串是否已被使用,一次请求有效。

接下来开始在 ThinkPHP 和 Gin 框架中进行实现,文中只展示了核心的代码,完整代码的获取方式放在了文章末尾。

我们先熟悉一下项目结构核心的目录,有助于理解文中的内容。一个正常的请求首先要经过路由 route 再到中间件 middleware 最后到控制器 controller,API 接口的签名验证是在中间件 middleware 中实现,作为一个中间层在整个请求链路中起着承上启下的重要作用。

  1. [manongsen@root php_to_go]$ tree -L 2
  2. .
  3. ├── go_sign
  4. ├── app
  5. ├── controller
  6. └── user.go
  7. ├── middleware
  8. └── api_sign.go
  9. ├── config.go
  10. └── route.go
  11. ├── go.mod
  12. ├── go.sum
  13. └── main.go
  14. └── php_sign
  15. ├── app
  16. ├── controller
  17. └── User.php
  18. ├── middleware
  19. └── ApiSign.php
  20. └── middleware.php
  21. ├── composer.json
  22. ├── composer.lock
  23. ├── config
  24. ├── route
  25. └── app.php
  26. ├── think
  27. ├── vendor
  28. └── .env

ThinkPHP

使用 composer 创建基于 ThinkPHP 框架的 php_sign 项目。

  1. [manongsen@root ~]$ pwd
  2. /home/manongsen/workspace/php_to_go/php_sign
  3. [manongsen@root php_sign]$ composer create-project topthink/think php_sign

随机字符串需要用到 Redis 进行存储,所以这里需要安装 Redis 扩展包,便于操作 Redis。

  1. [manongsen@root php_sign]$ composer require predis/predis

在项目 php_sign 下创建 ApiSign 中间件。

  1. [manongsen@root php_sign]$ php think make:middleware ApiSign
  2. Middleware:app\middleware\ApiSign created successfully.

在项目 php_sign 下复制一个 env 配置文件,并且定义好 AppKey。

  1. [manongsen@root php_sign]$ cp .example.env .env

API 接口签名的验证是放在框架的中间件中进行实现的,其中时间戳的有效时间设置的是 2 秒,有些朋友会有疑惑为什么是 2 秒?3 秒、5 秒不行吗?这里的有效时间是基于网络通信的延时考虑的,根据普遍情况延时大概是 2 秒。如果你的服务延时比较长,也可以设置长一些,并没有一个定量的值,话说到这里也提醒一下如果你的接口延时超过 2 秒,大概率需要优化一下代码了。此外,还有一个随机字符串参数,这个参数的目的是为了防止接口被重放,如果做过爬虫的朋友可能对这个会深有感触,这也是防范爬虫的一种手段。

  1. <?php
  2. declare (strict_types = 1);
  3. namespace app\middleware;
  4. use think\facade\Env;
  5. use think\facade\Cache;
  6. class ApiSign
  7. {
  8. /**
  9. * 处理请求
  10. *
  11. * @param \think\Request $request
  12. * @param \Closure $next
  13. * @return Response
  14. */
  15. public function handle($request, \Closure $next)
  16. {
  17. /*********************** 验证AppKey参数 ******************/
  18. $headers = $request->header();
  19. if (!isset($headers["app-key"])) {
  20. return json(["code" => 400, "msg" => "秘钥参数缺失"]);
  21. }
  22. $reqAppKey = $headers["app-key"];
  23. $vfyAppKey = Env::get("APP_KEY");
  24. if ($reqAppKey != $vfyAppKey) {
  25. return json(["code" => 400, "msg" => "签名秘钥无效"]);
  26. }
  27. /*********************** 验证时间戳参数 *******************/
  28. $params = $request->param();
  29. if (!isset($params["timestamp"])) {
  30. return json(["code" => 400, "msg" => "时间参数缺失"]);
  31. }
  32. $timestamp = $params["timestamp"];
  33. $nowTime = time();
  34. if (($nowTime-$timestamp) > 2) {
  35. return json(["code" => 400, "msg" => "时间参数过期"]);
  36. }
  37. /*********************** 验证签名串参数 *******************/
  38. if (!isset($params["sign"])) {
  39. return json(["code" => 400, "msg" => "签名参数缺失"]);
  40. }
  41. $reqSign = $params["sign"];
  42. unset($params["sign"]);
  43. // 将参数进行排序
  44. ksort($params);
  45. $paramStr = http_build_query($params);
  46. // md5 加密处理
  47. $vfySign = md5($paramStr . "&app_key={$vfyAppKey}");
  48. // 比较签名参数
  49. if ($reqSign != $vfySign) {
  50. return json(["code" => 400, "msg" => "签名验证失败"]);
  51. }
  52. /*********************** 验证随机串参数 *******************/
  53. if (!isset($params["nonce_str"])) {
  54. return json(["code" => 400, "msg" => "随机串参数缺失"]);
  55. }
  56. $nonceStr = $params["nonce_str"];
  57. // 判断 nonce_str 随机字符串是否被使用
  58. $redis = Cache::store('redis')->handler();
  59. $flag = $redis->exists($nonceStr);
  60. if ($flag) {
  61. return json(["code" => 400, "msg" => "随机串参数无效"]);
  62. }
  63. // 存储 nonce_str 随机字符串
  64. $redis->set($nonceStr, $timestamp, 2);
  65. return $next($request);
  66. }
  67. }

启动 php_sign 服务。

  1. [manongsen@root php_sign]$ php think run
  2. ThinkPHP Development server is started On <http://0.0.0.0:8000/>
  3. You can exit with `CTRL-C`
  4. Document root is: /home/manongsen/workspace/php_to_go/php_sign/public
  5. [Wed Jul 3 22:02:16 2024] PHP 8.3.4 Development Server (http://0.0.0.0:8000) started

使用 Postman 工具进行测试验证,通过构造正确的参数,便可以成功的返回数据。

Gin

通过 go mod 初始化 go_sign 项目。

  1. [manongsen@root ~]$ pwd
  2. /home/manongsen/workspace/php_to_go/go_sign
  3. [manongsen@root go_sign]$ go mod init go_sign

安装 Gin 框架库,这里与 ThinkPHP 不一样的是 Gin 框架是以第三库的形式在 gin_sign 项目中进行引用的。

  1. [manongsen@root go_sign]$ go get github.com/gin-gonic/gin

安装 Redis 操作库,与在 ThinkPHP 框架中一样也要使用到 Redis。

  1. [manongsen@root go_sign]$ go get github.com/go-redis/redis

这是在 Gin 框架中利用中间件来进行 API 接口签名验证,从代码量上来看就比 PHP 要多了。其中还需要自行合并 GET 和 POST 参数,方便在中间件中统一进行签名处理。对参数的拼接也没有类似 http_build_query 的方法,总体上来说在 Go 中进行签名验证需要繁琐不少。

  1. package middleware
  2. import (
  3. "bytes"
  4. "crypto/md5"
  5. "encoding/json"
  6. "fmt"
  7. "go_sign/app"
  8. "io/ioutil"
  9. "net/http"
  10. "sort"
  11. "strconv"
  12. "strings"
  13. "time"
  14. "github.com/gin-gonic/gin"
  15. )
  16. func ApiSign() gin.HandlerFunc {
  17. return func(c *gin.Context) {
  18. /*************************** 验证AppKey参数 **************************/
  19. reqAppKey := c.Request.Header.Get("app-key")
  20. if len(reqAppKey) == 0 {
  21. c.JSON(http.StatusOK, gin.H{"code": 400, "msg": "秘钥参数缺失"})
  22. c.Abort()
  23. return
  24. }
  25. vfyAppKey := app.APP_KEY
  26. if reqAppKey != vfyAppKey {
  27. c.JSON(http.StatusOK, gin.H{"code": 400, "msg": "秘钥参数无效"})
  28. c.Abort()
  29. return
  30. }
  31. // 获取请求参数
  32. params := mergeParams(c)
  33. /*************************** 验证时间戳参数 **************************/
  34. if _, ok := params["timestamp"]; !ok {
  35. c.JSON(http.StatusOK, gin.H{"code": 400, "msg": "时间参数无效"})
  36. c.Abort()
  37. return
  38. }
  39. timestampStr := fmt.Sprintf("%v", params["timestamp"])
  40. timestampInt, err := strconv.ParseInt(timestampStr, 0, 64)
  41. if err != nil {
  42. c.JSON(http.StatusOK, gin.H{"code": 400, "msg": "时间参数无效"})
  43. c.Abort()
  44. return
  45. }
  46. nowTime := time.Now().Unix()
  47. if nowTime-timestampInt > 2 {
  48. c.JSON(http.StatusOK, gin.H{"code": 400, "msg": "时间参数过期"})
  49. c.Abort()
  50. return
  51. }
  52. /*************************** 验证签名串参数 **************************/
  53. if _, ok := params["sign"]; !ok {
  54. c.JSON(http.StatusOK, gin.H{"code": 400, "msg": "签名参数无效"})
  55. c.Abort()
  56. return
  57. }
  58. reqSign := fmt.Sprintf("%v", params["sign"])
  59. // 针对 dataMap 进行排序
  60. dataMap := params
  61. keys := make([]string, len(dataMap))
  62. i := 0
  63. for k := range dataMap {
  64. keys[i] = k
  65. i++
  66. }
  67. sort.Strings(keys)
  68. var buf bytes.Buffer
  69. for _, k := range keys {
  70. if k != "sign" && !strings.HasPrefix(k, "reserved") {
  71. buf.WriteString(k)
  72. buf.WriteString("=")
  73. buf.WriteString(fmt.Sprintf("%v", dataMap[k]))
  74. buf.WriteString("&")
  75. }
  76. }
  77. bufStr := buf.String()
  78. dataStr := bufStr + "app_key=" + app.APP_KEY
  79. // 进行 md5 加密处理
  80. data := []byte(dataStr)
  81. has := md5.Sum(data)
  82. vfySign := fmt.Sprintf("%x", has) // 将[]byte转成16进制
  83. if reqSign != vfySign {
  84. c.JSON(http.StatusOK, gin.H{"code": 400, "msg": "签名验证失败"})
  85. c.Abort()
  86. return
  87. }
  88. /*************************** 验证随机串参数 **************************/
  89. if _, ok := params["nonce_str"]; !ok {
  90. c.JSON(http.StatusOK, gin.H{"code": 400, "msg": "随机串参数缺失"})
  91. c.Abort()
  92. return
  93. }
  94. nonceStr := fmt.Sprintf("%v", params["nonce_str"])
  95. // 判断是否存在 nonce_str 随机字符串
  96. flag, _ := app.RedisConn.Exists(nonceStr).Result()
  97. if flag > 0 {
  98. c.JSON(http.StatusOK, gin.H{"code": 400, "msg": "随机串参数无效"})
  99. c.Abort()
  100. return
  101. }
  102. // 存储nonce_str随机字符串
  103. app.RedisConn.Set(nonceStr, timestampInt, time.Second*2).Result()
  104. c.Next()
  105. }
  106. }
  107. // 将 GET 和 POST 的参数合并到同一 Map
  108. func mergeParams(c *gin.Context) map[string]interface{} {
  109. var (
  110. dataMap = make(map[string]interface{})
  111. queryMap = make(map[string]interface{})
  112. postMap = make(map[string]interface{})
  113. )
  114. contentType := c.ContentType()
  115. for k := range c.Request.URL.Query() {
  116. queryMap[k] = c.Query(k)
  117. }
  118. if contentType == "application/json" {
  119. if c.Request != nil && c.Request.Body != nil {
  120. bodyBytes, _ := ioutil.ReadAll(c.Request.Body)
  121. if len(bodyBytes) > 0 {
  122. if err := json.NewDecoder(bytes.NewBuffer(bodyBytes)).Decode(&postMap); err != nil {
  123. return nil
  124. }
  125. c.Request.Body = ioutil.NopCloser(bytes.NewBuffer(bodyBytes))
  126. }
  127. }
  128. } else if contentType == "multipart/form-data" {
  129. for k, v := range c.Request.PostForm {
  130. if len(v) > 1 {
  131. postMap[k] = v
  132. } else if len(v) == 1 {
  133. postMap[k] = v[0]
  134. }
  135. }
  136. } else {
  137. for k, v := range c.Request.PostForm {
  138. if len(v) > 1 {
  139. postMap[k] = v
  140. } else if len(v) == 1 {
  141. postMap[k] = v[0]
  142. }
  143. }
  144. }
  145. // 优先级:以post优先级最高,会覆盖get参数
  146. for k, v := range queryMap {
  147. dataMap[k] = v
  148. }
  149. for k, v := range postMap {
  150. dataMap[k] = v
  151. }
  152. return dataMap
  153. }

启动 gin_sin 服务。

  1. [manongsen@root go_sign]$ go run main.go
  2. [GIN-debug] [WARNING] Creating an Engine instance with the Logger and Recovery middleware already attached.
  3. [GIN-debug] [WARNING] Running in "debug" mode. Switch to "release" mode in production.
  4. - using env: export GIN_MODE=release
  5. - using code: gin.SetMode(gin.ReleaseMode)
  6. [GIN-debug] GET /user/info --> go_sign/app/controller.UserInfo (4 handlers)
  7. [GIN-debug] [WARNING] You trusted all proxies, this is NOT safe. We recommend you to set a value.
  8. Please check https://pkg.go.dev/github.com/gin-gonic/gin#readme-don-t-trust-all-proxies for details.
  9. [GIN-debug] Listening and serving HTTP on :8001

同样也使用 Postman 工具进行测试验证,通过构造正确的参数,便可以成功的返回数据。

结语

数据安全一直是个热门的话题,API 接口在数据的传输上扮演着至关重要的角色。为了 API 接口的安全性、健壮性,完整性,往往需要将网络上的数据进行签名加密传输。同时为了防止 API 接口被重放爬虫伪造等类似恶意攻击的手段,还要在接口设计时增加有效时间、随机字符串、签名串等参数,来保障数据的安全性。这一次的 API 接口签名设计实践,大家也可以手动尝试实验一下,希望对大家的日常工作能有所帮助。最后感兴趣的朋友可以在微信公众号内回复「4867」获取完整的实践代码。


欢迎关注、分享、点赞、收藏、在看,我是微信公众号「码农先森」作者。

原文链接:https://www.cnblogs.com/yxhblogs/p/18282751

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号