easyswoole - 基于swoole扩展实现的一款高性能php框架

Swoole 如何對 IP 限制訪問頻率

在我們開發 API 的過程中,有的時候我們還需要考慮單個用戶(IP)訪問頻率控制,避免被惡意調用。

歸根到底也就只有兩個步驟:

  • 用戶訪問要統計次數
  • 執行操作邏輯之前要判斷次數頻率是否過高,過高則不執行

EasySwoole 中實現 IP 訪問頻率限制

本文舉例的是在 EasySwoole 框架中實現的代碼,在Swoole 原生中實現方式是一樣的。

只要在對應的回調事件做判斷攔截處理即可。

  • 使用 Swoole\Table,存儲用戶訪問情況(也可以使用其他組件、方式存儲)
  • 使用定時器,將前一周期的訪問情況清空,統計下一周期

實現 IP 訪問統計類

如以下 IpList 類,實現了 初始化 Table統計 IP訪問次數獲取一個周期內次數超過一定值的記錄

<?php
/**
 * Ip訪問次數統計
 * User: Siam
 * Date: 2019/7/8 0008
 * Time: 下午 9:53
 */

namespace App;

use EasySwoole\Component\Singleton;
use EasySwoole\Component\TableManager;
use Swoole\Table;

class IpList
{
    use Singleton;

    /** @var Table */
    protected $table;

    public function __construct()
    {
        TableManager::getInstance()->add('ipList', [
            'ip' => [
                'type' => Table::TYPE_STRING,
                'size' => 16
            ],
            'count' => [
                'type' => Table::TYPE_INT,
                'size' => 8
            ],
            'lastAccessTime' => [
                'type' => Table::TYPE_INT,
                'size' => 8
            ]
        ], 1024 * 128);
        $this->table = TableManager::getInstance()->get('ipList');
    }

    public function access(string $ip): int
    {
        $key = substr(md5($ip), 8, 16);
        $info = $this->table->get($key);

        if ($info) {
            $this->table->set($key, [
                'lastAccessTime' => time(),
                'count' => $info['count'] + 1,
            ]);
            return $info['count'] + 1;
        } else {
            $this->table->set($key, [
                'ip' => $ip,
                'lastAccessTime' => time(),
                'count' => 1,
            ]);
            return 1;
        }
    }

    public function clear()
    {
        foreach ($this->table as $key => $item) {
            $this->table->del($key);
        }
    }

    public function accessList($count = 10): array
    {
        $ret = [];
        foreach ($this->table as $key => $item) {
            if ($item['count'] >= $count) {
                $ret[] = $item;
            }
        }
        return $ret;
    }
}

初始化 IP 統計類 和訪問統計定時器

封裝完 IP統計 的操作之后,我們就可以在 EasySwooleEvent.php 中的 mainServerCreate 回調事件中初始化 IpList 類和定時器,注冊 IP 統計自定義進程

<?php
use App\IpList;
use EasySwoole\Component\Process\AbstractProcess;
use EasySwoole\Component\Process\Manager;

public static function mainServerCreate(EventRegister $register)
{
    // 開啟 IP 限流
    IpList::getInstance();
    $class = new class('IpAccessCount') extends AbstractProcess
    {
        protected function run($arg)
        {
            $this->addTick(10 * 1000, function () {
                /**
                 * 正常用戶不會有一秒超過 6 次的api請求
                 * 做列表記錄并清空
                 */
                $list = IpList::getInstance()->accessList(30);
                // var_dump($list);
                IpList::getInstance()->clear();
            });
        }
    };

    // 注冊 IP 限流自定義進程
    $processConfig = new \EasySwoole\Component\Process\Config();
    $processConfig->setProcessName('IP_LIST');// 設置進程名稱
    $processConfig->setProcessGroup('IP_LIST');// 設置進程組名稱
    $processConfig->setArg([]);// 傳參
    $processConfig->setRedirectStdinStdout(false);// 是否重定向標準io
    $processConfig->setPipeType(\EasySwoole\Component\Process\Config::PIPE_TYPE_SOCK_DGRAM);// 設置管道類型
    $processConfig->setEnableCoroutine(true);// 是否自動開啟協程
    $processConfig->setMaxExitWaitTime(3);// 最大退出等待時間
    Manager::getInstance()->addProcess(new $class($processConfig));
}

實現對 IP 訪問的限制

EasySwooleEvent.php 中的 mainServerCreate 回調事件中 接著我們在 EasySwooleEvent.php 中的 initialize 回調事件中注入 HTTP_GLOBAL_ON_REQUEST 全局事件,判斷和統計 IP 的訪問

<?php

use EasySwoole\Component\Di;
use EasySwoole\Http\Request;
use EasySwoole\Http\Response;
use App\IpList;

public static function initialize()
{
    date_default_timezone_set('Asia/Shanghai');

    Di::getInstance()->set('HTTP_GLOBAL_ON_REQUEST', function (Request $request, Response $response) {
        $fd = $request->getSwooleRequest()->fd;
        $ip = ServerManager::getInstance()->getSwooleServer()->getClientInfo($fd)['remote_ip'];

        // 如果當前周期的訪問頻率已經超過設置的值,則攔截
        // 測試的時候可以將 30 改小,比如 3
        if (IpList::getInstance()->access($ip) > 3) {
            /**
             * 直接強制關閉連接
             */
            ServerManager::getInstance()->getSwooleServer()->close($fd);
            // 調試輸出 可以做邏輯處理
            echo '被攔截' . PHP_EOL;
            return false;
        }
        // 調試輸出 可以做邏輯處理
        echo '正常訪問' . PHP_EOL;
        return true;
    });
}

以上就實現了對同一 IP 訪問頻率的限制操作。具體還可以根據自身需求進行擴展,如對具體的某個接口再進行限流。

EasySwoole 提供了一個基于 Atomic 計數器的限流器組件。可以直接使用,使用教程請移步查看限流器文檔

主站蜘蛛池模板: 丝杆升降机-蜗轮丝杆升降机-电动推杆-德州市金宇机械有限公司 | 盘扣租赁|盘扣架租赁|盘扣脚手架|盘扣脚手架租赁|盘扣式脚手架|盘扣式脚手架租赁-北京亚欧盟盘扣租赁有限公司 | 医药冷库设计建造-食品保鲜冷库安装-物流冷库工程造价-开冉制冷 医盟网-全国首家医疗信息化行业门户网站 | 真空机器人维修_晶圆机械手保养_半导体机械臂维修_面板机器人保养_AMHS改造-广州市广科智能技术有限公司 | 呼吸家官网|肺功能检测仪生产厂家|国产肺功能仪知名品牌|肺功能检测仪|肺功能测试仪|婴幼儿肺功能仪|弥散残气肺功能仪|肺功能测试系统|广州红象医疗科技有限公司|便携式肺功能仪|大肺功能仪|呼吸康复一体机|儿童肺功能仪|肺活量计|医用简易肺功能仪|呼吸康复系统|肺功能仪|弥散肺功能仪(大肺)|便携式肺功能检测仪|肺康复|呼吸肌力测定肺功能仪|肺功能测定仪|呼吸神经肌肉刺激仪|便携式肺功能 | 碳化硅脱硫喷嘴,碳化硅烧嘴套,碳化硅耐磨衬套-潍坊致达特种陶瓷有限公司 | 台车炉厂家_台车式退火炉_台车式回火炉—安徽大新工业炉有限公司 | 土工膜_土工布_复合土工膜_山东土工膜生产厂家_山东路易达新材料有限公司 | 取样冷却器-射水抽气器-锅炉炉水取样冷却器-连灵动 | 液力耦合器,摩擦型液力耦合器生产厂家-河南省华升矿机有限公司 | 全自动红外测油仪|全自动COD分析仪|BOD5分析仪|昂林官网 | 兰舍硅藻泥 -- 深圳硅藻泥|深圳兰舍硅藻泥|深圳硅藻泥品牌|深圳硅藻泥价格|深圳硅藻泥厂家|深圳硅藻泥施工| | 景德镇晶达新材料有限公司| 心理咨询室设备_音乐放松椅_心理测评系统_情绪宣泄设备厂家 | 全自动码垛机械手,码垛机器人,拆包机,缠绕机,开箱封箱装箱机厂家-山东昊宇自动化设备有限公司 | 上海慧泰仪器制造有限公司_一体型马弗炉-可控真空干燥箱-强光稳定性试验箱 | 唐山森林光线影视文化有限公司 | 直膨式空调机组_风冷恒温恒湿_转轮式热回收_屋顶式空调机组_德州瑞尼森环保科技有限公司 | 铝合金百叶窗_西安百叶窗厂家-西安市未央区通达建材物资部 | 开拓者喷雾设备有限公司专业生产喷嘴,喷雾设备,清洗,喷涂,降温,除尘,润滑等喷雾系统的方案解决商 | 无尘布_乳胶手套_防静电手环_口罩-苏州迈思德超净科技有限公司 | 磐石在线-磐石市综合信息门户网www.pszx.com - Powered by Discuz! | 净化塔_喷淋塔_脱硫塔_河北宝飞华创玻璃钢制造有限公司-河北宝飞华创玻璃钢制造有限公司 | 延吉新闻网 - 未来之选·就是延吉 [YanJinews.com] | 捏炼机_密炼机_炼胶机_平板硫化机-青岛光越橡胶机械制造有限公司 | 网带式等温正火生产线_燃气式铝合金加热炉_燃气式烘干窑炉-湖州中科炉业科技有限公司 | 移动石料破碎机-颚式锤式反击式破碎机设备厂家_山东.青州富康机械 | 扭矩测试仪_拉力测试仪_扭力扳手测试仪_测试台-上海铸衡电子科技有限公司 | 上海消防器材|水雾喷头|水幕喷头|螺旋喷头|雾化喷头|泡沫喷头 - 上海舜丹消防设备有限公司 | 厦门空压机-厦门空压机出租-厦门钻机出租-厦门岩立盛机电设备有限公司 | 饲料车_散装饲料车_畜禽运输车_散装饲料运输车_饲料车厂家_铝合金运猪车-程力专用汽车股份有限公司 | 商标注册查询_商标注册代理公司_专利申请_版权登记-源智知识产权 | 山东啤酒箱塑料提手_注塑产品加工_手提绳厂家-淄博浩晨包装制品有限公司 | 型煤锅炉进煤机|型煤链条炉排 |重型板链除渣机 |丹东刮板输送机|丹东脱硫除尘器-铧洋机械 | 指挥调度|调度系统|应急指挥调度|应急指挥|可视化调度|多媒体指挥调度|融合通信|综合调度|应急指挥系统|IP调度系统-北京瑞光极远数码科技有限公司 | 久久黄色一级视频_视频一区精品自拍_理论片免费ā片在线观看_亚洲色视频在线播放网站_香港经典a毛片免费观看_亚州三级久久电影 | 景德镇薪如陶瓷有限公司| 和部长一起去出差旅全程,构建和谐上下级关系-蜜桃视频 | 小耳朵电源_安防监控电源|小耳朵官网|电源适配器|摄像机电源|开关电源|小耳朵监控电源 | 江苏广分检测技术有限公司、电力安全工具检测、苏州绝缘工具检测、昆山电力安全工具检测-广分检测技术(苏州)有限公司 | 中原起重-河南省中原起重机有限公司【官网】 |