文章來源:?[http://www.cnblogs.com/xiaochangwei/p/api.html](http://www.cnblogs.com/xiaochangwei/p/api.html)
?API接口由于需要供第三方服務調用,所以必須暴露到外網,并提供了具體請求地址和請求參數
為了防止被第別有用心之人獲取到真實請求參數后再次發起請求獲取信息,需要采取很多安全機制
1.首先: 需要采用https方式對第三方提供接口,數據的加密傳輸會更安全,即便是被破解,也需要耗費更多時間
2.其次:需要有安全的后臺驗證機制【本文重點】,達到防參數篡改+防二次請求
主要防御措施可以歸納為兩點:
對請求的合法性進行校驗
對請求的數據進行校驗
防止重放攻擊必須要保證請求僅一次有效
需要通過在請求體中攜帶當前請求的唯一標識,并且進行簽名防止被篡改。
所以防止重放攻擊需要建立在防止簽名被串改的基礎之上。
請求參數防篡改
采用https協議可以將傳輸的明文進行加密,但是黑客仍然可以截獲傳輸的數據包,進一步偽造請求進行重放攻擊。如果黑客使用特殊手段讓請求方設備使用了偽造的證書進行通信,那么https加密的內容也將會被解密。
在API接口中我們除了使用https協議進行通信外,還需要有自己的一套加解密機制,對請求參數進行保護,防止被篡改。
過程如下:
客戶端使用約定好的秘鑰對傳輸參數進行加密,得到簽名值signature,并且將簽名值也放入請求參數中,發送請求給服務端
服務端接收客戶端的請求,然后使用約定好的秘鑰對請求的參數(除了signature以外)再次進行簽名,得到簽名值autograph。
服務端對比signature和autograph的值,如果對比一致,認定為合法請求。如果對比不一致,說明參數被篡改,認定為非法請求。
因為黑客不知道簽名的秘鑰,所以即使截取到請求數據,對請求參數進行篡改,但是卻無法對參數進行簽名,無法得到修改后參數的簽名值signature。
簽名的秘鑰我們可以使用很多方案,可以采用對稱加密或者非對稱加密。
防止重放攻擊
基于timestamp的方案
每次HTTP請求,都需要加上timestamp參數,然后把timestamp和其他參數一起進行數字簽名。因為一次正常的HTTP請求,從發出到達服務器一般都不會超過60s,所以服務器收到HTTP請求之后,首先判斷時間戳參數與當前時間相比較,是否超過了60s,如果超過了則認為是非法的請求。
一般情況下,黑客從抓包重放請求耗時遠遠超過了60s,所以此時請求中的timestamp參數已經失效了。
如果黑客修改timestamp參數為當前的時間戳,則signature參數對應的數字簽名就會失效,因為黑客不知道簽名秘鑰,沒有辦法生成新的數字簽名。
但這種方式的漏洞也是顯而易見的,如果在60s之后進行重放攻擊,那就沒辦法了,所以這種方式不能保證請求僅一次有效。
基于nonce的方案
nonce的意思是僅一次有效的隨機字符串,要求每次請求時,該參數要保證不同,所以該參數一般與時間戳有關,我們這里為了方便起見,直接使用時間戳的16進制,實際使用時可以加上客戶端的ip地址,mac地址等信息做個哈希之后,作為nonce參數。
我們將每次請求的nonce參數存儲到一個“集合”中,可以json格式存儲到數據庫或緩存中。
每次處理HTTP請求時,首先判斷該請求的nonce參數是否在該“集合”中,如果存在則認為是非法請求。
nonce參數在首次請求時,已經被存儲到了服務器上的“集合”中,再次發送請求會被識別并拒絕。
nonce參數作為數字簽名的一部分,是無法篡改的,因為黑客不清楚token,所以不能生成新的sign。
這種方式也有很大的問題,那就是存儲nonce參數的“集合”會越來越大,驗證nonce是否存在“集合”中的耗時會越來越長。我們不能讓nonce“集合”無限大,所以需要定期清理該“集合”,但是一旦該“集合”被清理,我們就無法驗證被清理了的nonce參數了。也就是說,假設該“集合”平均1天清理一次的話,我們抓取到的該url,雖然當時無法進行重放攻擊,但是我們還是可以每隔一天進行一次重放攻擊的。而且存儲24小時內,所有請求的“nonce”參數,也是一筆不小的開銷。
基于timestamp和nonce的方案
nonce的一次性可以解決timestamp參數60s的問題,timestamp可以解決nonce參數“集合”越來越大的問題。
防止重放攻擊一般和防止請求參數被串改一起做,請求的Headers數據如下圖所示。
我們在timestamp方案的基礎上,加上nonce參數,因為timstamp參數對于超過60s的請求,都認為非法請求,所以我們只需要存儲60s的nonce參數的“集合”即可。
?
\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*常用接口簽名,防止參數被串改偽造.\[未加密\]\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*?
其中幾個要點
token參數\[雙方私下約定\]
請求參數排序\[sort ksort krsort asort arsort 等\]
請求參數生成關聯字符串\[http\_build\_query,自定義拼接等\]
對字符串單向加密\[md5,sha1等\]
????/\*\*
? ? ?\* 簽名
? ? ?\*
? ? ?\* @param array $params
? ? ?\* @param string $token
? ? ?\* @return void
? ? ?\* @author skyer 405661806@qq.com
? ? ?\*/
? ? public function \_sign($params=array(), $token=''){
? ? ? ? $response = array('code'=>1, '簽名成功');
? ? ? ? try{
? ? ? ? ? ? if(!is\_array($params) || empty($params)){
? ? ? ? ? ? ? ? throw new \\Exception('簽名參數異常');
? ? ? ? ? ? }
? ? ? ? ? ? $params\['token'\] = $token;
? ? ? ? ? ? sort($params);
? ? ? ? ? ? $tmpstr = http\_build\_query($params);
? ? ? ? ? ? $signature = sha1($tmpstr);
? ? ? ? ? ? if(!$signature){
? ? ? ? ? ? ? ? throw new \\Exception('簽名失敗');
? ? ? ? ? ? }
? ? ? ? ? ? $response\['data'\] = $signature;
? ? ? ? }catch(\\Exception $e){
? ? ? ? ? ? $response\['code'\] = 0;
? ? ? ? ? ? $response\['msg'\] = $e->getMessage();
? ? ? ? }
? ? ? ? return $response;
? ? }
? ? /\*\*
? ? ?\* 驗簽
? ? ?\*
? ? ?\* @param \[type\] $params
? ? ?\* @param string $token
? ? ?\* @return void
? ? ?\* @author skyer 405661806@qq.com
? ? ?\*/
? ? public function \_vaild($params, $token=''){
? ? ? ? $response = array('code'=>1, '驗簽成功');
? ? ? ? try{
? ? ? ? ? ? // fileDebug('驗簽參數: '.json\_encode($params) , '\_vaild.php');
? ? ? ? ? ? if(!is\_array($params) || empty($params)){
? ? ? ? ? ? ? ? throw new \\Exception('驗簽參數異常');
? ? ? ? ? ? }
? ? ? ? ? ? $signature = $params\['signature'\];
? ? ? ? ? ? unset($params\['signature'\]);
? ? ? ? ? ? $params\['token'\] = $token;
? ? ? ? ? ? sort($params);
? ? ? ? ? ? $tmpstr = http\_build\_query($params);
? ? ? ? ? ? $vaild = sha1($tmpstr);
? ? ? ? ? ? if($signature != $vaild){
? ? ? ? ? ? ? ? throw new \\Exception('驗簽失敗');
? ? ? ? ? ? }
? ? ? ? }catch(\\Exception $e){
? ? ? ? ? ? $response\['code'\] = 0;
? ? ? ? ? ? $response\['msg'\] = $e->getMessage();
? ? ? ? }
? ? ? ? return $response;
? ? }
- 常見功能
- 第三方授權登錄
- 郵件發送
- 簡易聊天室
- 獲取各國匯率
- PHP獲取服務器硬件指標
- 數據上報之
- web開發
- 開發規范
- 前端
- 踩坑
- 將footer固定在底部
- bootstrap
- Metronic
- 用到的jquery插件
- bootstrap-hover-dropdown
- jquery.slimscroll
- jquery.blockui
- bootstrap-switch
- js.cookie
- moment
- bootstrap-daterangepicker
- morris
- raphael
- jquery.waypoints
- jquery.counterup
- select2
- 取值和設置默認值
- vue
- axios
- 瀏覽器
- 谷歌瀏覽器
- 谷歌插件
- layui
- layui-表格
- layui-表單
- layui-彈窗
- layui-分頁
- 后端
- 操作系統
- linux
- 用戶管理
- 文件管理
- 目錄管理
- 壓縮和解壓縮
- 進程查看
- 端口查看
- 開機自啟動服務
- 定時任務
- shell腳本
- 殺掉運行超過指定時長指定服務的進程
- 獲取服務器使用狀態
- bash-shell連接socket
- 自定義快捷命令
- centos-踩坑
- 防火墻
- 軟件
- yum
- vim
- screen
- window
- 語言
- PHP
- 配置優化
- 框架
- thinkphp5.1+
- think命令行
- laravel6.+
- 維護模式
- 根據環境讀取不同配置
- laravel6.+采坑
- laravel坑位
- 數據庫事務
- 任務調度
- 文件權限問題
- 增強框架
- larvel:elastic-search
- 圖形驗證碼
- laravel獲取ip
- 函數
- strtotime
- 正則匹配
- 類
- 接口類與抽象類
- 類相關的關鍵字 - abstract
- 類相關的關鍵字 - interface
- PHP有關類的調用方式"->"與"::"的區別
- 擴展
- 問題歸納
- json_encode和json_decode
- 字符串的運算
- curl
- 優化php效率
- 數組相加合并與array_merge
- 時區轉換
- 不常用特性
- php反射
- 包管理器-composer
- GuzzleHttp
- Python
- Go
- 數據庫
- Redis
- 安裝
- 本地化-數據備份
- php-redis操作
- Mysql
- mysql-命令集合
- 設置終端可訪問
- 數據庫設計
- 用戶基礎信息表
- 踩坑集合
- mysql-2002
- mysql-2054
- 優化策略
- mysql-密碼驗證插件
- 一些牛逼的sql查詢
- topN
- 無限級分類
- Memcache
- MongoDb
- 安裝mongo-server
- 安裝php-mongodb擴展
- 在laravel中使用mongoDB
- 客戶端軟件
- Hbase
- Elasticsearch
- elastic-search
- restfulApi操作es
- web服務器
- 1.nginx
- 配置語法規則
- 配置詳解
- rewrite規則
- request_filename
- 2.apache
- 功能設計
- 加密解密
- Base64
- 對亞馬遜SKU加密
- 兼職項目中的加解密
- 騰訊外包時的加密
- 接口設計
- 接口限流設計
- 分庫分表
- 遍歷展示文件目錄結構
- 時區換算
- 文件切割
- 解析xml字符串
- 項目
- 博客后臺管理
- 亞馬遜廣告API
- 官方指引文檔
- 開發人員中心
- 應用商店
- 第三方庫
- 申請API郵件記錄
- 亞馬遜MWS
- 付款報告
- 亂碼
- 亞馬遜管理庫存報告
- 報告
- 商品
- 入庫
- 履行
- 出庫
- 財務
- 訂單
- 異步任務處理
- 集群如何同步代碼
- 基本開發流程
- 文檔管理
- showdoc
- 運行環境
- 開發環境
- vagrant
- windows上配置安裝
- vagrant安裝插件緩慢
- 更換ssh默認端口映射
- 設置x-shell密碼登錄
- 使用市場的box-homestead
- homestead-7: Box 'lc/homestead'
- 常見問題
- 虛擬環境reboot
- 突然無法使用
- phpStudy
- wamp
- 壓測性能
- VPN
- vultr
- 凌空圖床
- 寶塔
- 自動化部署
- 版本管理軟件鉤子
- 線上環境-LNMP
- centos7
- nginx
- mysql
- mysql開機自啟
- mysql-更換默認端口
- datetime字段類型默認值
- php
- php擴展安裝
- redis
- swoole
- gd
- BCMath
- igbinary
- zstd
- 包管理器:composer
- 優化性能
- nodejs
- 更新gcc版本
- 版本控制
- git
- 常用命令
- gitlab
- 版本管理規范
- 使用阿里云創建遠程倉庫
- git自動化部署
- svn
- 忽略指定文件
- 拉取代碼
- 自動化運維
- jekins
- 容器
- 集群
- 架構設計
- 設計原則
- 閱讀參考
- 代碼規劃
- 架構實戰
- 服務治理
- 權限控制設計
- 具體設計
- 計劃
- 疑問知識點
- 讀書筆記
- 高性能Mysql
- TCP-IP詳解-卷一:協議
- 思考
- php如何實現并發執行
- 對接調用設計
- 如何在瀏覽器上實現插件
- 如何設計一個app結合業務告警
- mysql的where查詢沒有用到索引
- 為啥in查詢比循環嵌套sql的查詢還要慢
- 使用git來創建屬于自己的composer包
- 翻頁獲取數據的時候又新增了數據
- 安全思路
- 月報
- PHP ?? 和 ?: 的區別
- PHP異步執行
- redis集群的目標是什么
- 大文件數據處理
- 性能瓶頸分析
- 命令行里輸出帶顏色的字體
- 面試問題合集
- 基礎
- 安全
- 算法
- 冒泡排序
- 快速排序
- 二分法查詢數組指定成員
- 字符查找匹配
- 令牌桶
- 漏桶
- 計數器
- 代理
- 協議
- http
- 狀態碼
- tcp
- udp
- Oauth2.0
- 設計模式
- 單例模式
- 適配器模式
- 工廠模式
- 觀察者模式
- 流程化
- 地址欄輸入網址到返回網頁的流程
- 題目收集
- 工具
- rabbitMq
- rabbitMQ用戶管理
- 生產者
- 消費者
- 支持TP5.*的think-queue
- 消息丟失
- 消費者報錯
- rabbitMQ配置優化
- 磁盤滿載導致服務掛掉
- PHP類庫
- rabbitMQ踩坑
- navicat
- vscode
- phpstorm
- 激活碼
- markdown
- PHP自定義類庫
- 工具類
- 領導力
- 任務分配
- 代碼組織
- 不要重復
- 避免污染
- 接口定義規范
- 小業務需求
- 獲取充值面額組成
- 監控服務器CPU和內存
- shell腳本版本