Swoole 如何对 IP 限制访问频率

在我们开发 API 的过程中,有的时候我们还需要考虑单个用户(IP)访问频率控制,避免被恶意调用。

归根到底也就只有两个步骤:

  • 用户访问要统计次数
  • 执行操作逻辑之前要判断次数频率是否过高,过高则不执行

EasySwoole 中实现 IP 访问频率限制

本文举例的是在 EasySwoole 框架中实现的代码,在Swoole 原生中实现方式是一样的。

只要在对应的回调事件做判断拦截处理即可。

  • 使用 Swoole\Table,存储用户访问情况(也可以使用其他组件、方式存储)
  • 使用定时器,将前一周期的访问情况清空,统计下一周期

实现 IP 访问统计类

如以下 IpList 类,实现了 初始化 Table统计 IP访问次数获取一个周期内次数超过一定值的记录

  1. <?php
  2. /**
  3. * Ip访问次数统计
  4. * User: Siam
  5. * Date: 2019/7/8 0008
  6. * Time: 下午 9:53
  7. */
  8. namespace App;
  9. use EasySwoole\Component\Singleton;
  10. use EasySwoole\Component\TableManager;
  11. use Swoole\Table;
  12. class IpList
  13. {
  14. use Singleton;
  15. /** @var Table */
  16. protected $table;
  17. public function __construct()
  18. {
  19. TableManager::getInstance()->add('ipList', [
  20. 'ip' => [
  21. 'type' => Table::TYPE_STRING,
  22. 'size' => 16
  23. ],
  24. 'count' => [
  25. 'type' => Table::TYPE_INT,
  26. 'size' => 8
  27. ],
  28. 'lastAccessTime' => [
  29. 'type' => Table::TYPE_INT,
  30. 'size' => 8
  31. ]
  32. ], 1024 * 128);
  33. $this->table = TableManager::getInstance()->get('ipList');
  34. }
  35. public function access(string $ip): int
  36. {
  37. $key = substr(md5($ip), 8, 16);
  38. $info = $this->table->get($key);
  39. if ($info) {
  40. $this->table->set($key, [
  41. 'lastAccessTime' => time(),
  42. 'count' => $info['count'] + 1,
  43. ]);
  44. return $info['count'] + 1;
  45. } else {
  46. $this->table->set($key, [
  47. 'ip' => $ip,
  48. 'lastAccessTime' => time(),
  49. 'count' => 1,
  50. ]);
  51. return 1;
  52. }
  53. }
  54. public function clear()
  55. {
  56. foreach ($this->table as $key => $item) {
  57. $this->table->del($key);
  58. }
  59. }
  60. public function accessList($count = 10): array
  61. {
  62. $ret = [];
  63. foreach ($this->table as $key => $item) {
  64. if ($item['count'] >= $count) {
  65. $ret[] = $item;
  66. }
  67. }
  68. return $ret;
  69. }
  70. }

初始化 IP 统计类 和访问统计定时器

封装完 IP统计 的操作之后,我们就可以在 EasySwooleEvent.php 中的 mainServerCreate 回调事件中初始化 IpList 类和定时器,注册 IP 统计自定义进程

  1. <?php
  2. use App\IpList;
  3. use EasySwoole\Component\Process\AbstractProcess;
  4. use EasySwoole\Component\Process\Manager;
  5. public static function mainServerCreate(EventRegister $register)
  6. {
  7. // 开启 IP 限流
  8. IpList::getInstance();
  9. $class = new class('IpAccessCount') extends AbstractProcess
  10. {
  11. protected function run($arg)
  12. {
  13. $this->addTick(10 * 1000, function () {
  14. /**
  15. * 正常用户不会有一秒超过 6 次的api请求
  16. * 做列表记录并清空
  17. */
  18. $list = IpList::getInstance()->accessList(30);
  19. // var_dump($list);
  20. IpList::getInstance()->clear();
  21. });
  22. }
  23. };
  24. // 注册 IP 限流自定义进程
  25. $processConfig = new \EasySwoole\Component\Process\Config();
  26. $processConfig->setProcessName('IP_LIST');// 设置进程名称
  27. $processConfig->setProcessGroup('IP_LIST');// 设置进程组名称
  28. $processConfig->setArg([]);// 传参
  29. $processConfig->setRedirectStdinStdout(false);// 是否重定向标准io
  30. $processConfig->setPipeType(\EasySwoole\Component\Process\Config::PIPE_TYPE_SOCK_DGRAM);// 设置管道类型
  31. $processConfig->setEnableCoroutine(true);// 是否自动开启协程
  32. $processConfig->setMaxExitWaitTime(3);// 最大退出等待时间
  33. Manager::getInstance()->addProcess(new $class($processConfig));
  34. }

实现对 IP 访问的限制

EasySwooleEvent.php 中的 mainServerCreate 回调事件中 接着我们在 EasySwooleEvent.php 中的 initialize 回调事件中注入 HTTP_GLOBAL_ON_REQUEST 全局事件,判断和统计 IP 的访问

  1. <?php
  2. use EasySwoole\Component\Di;
  3. use EasySwoole\Http\Request;
  4. use EasySwoole\Http\Response;
  5. use App\IpList;
  6. public static function initialize()
  7. {
  8. // TODO: Implement initialize() method.
  9. date_default_timezone_set('Asia/Shanghai');
  10. public static function initialize()
  11. {
  12. date_default_timezone_set('Asia/Shanghai');
  13. Di::getInstance()->set('HTTP_GLOBAL_ON_REQUEST', function (Request $request, Response $response) {
  14. $fd = $request->getSwooleRequest()->fd;
  15. $ip = ServerManager::getInstance()->getSwooleServer()->getClientInfo($fd)['remote_ip'];
  16. // 如果当前周期的访问频率已经超过设置的值,则拦截
  17. // 测试的时候可以将 30 改小,比如 3
  18. if (IpList::getInstance()->access($ip) > 3) {
  19. /**
  20. * 直接强制关闭连接
  21. */
  22. ServerManager::getInstance()->getSwooleServer()->close($fd);
  23. // 调试输出 可以做逻辑处理
  24. echo '被拦截' . PHP_EOL;
  25. return false;
  26. }
  27. // 调试输出 可以做逻辑处理
  28. echo '正常访问' . PHP_EOL;
  29. return true;
  30. });
  31. }
  32. }

以上就实现了对同一 IP 访问频率的限制操作。具体还可以根据自身需求进行扩展,如对具体的某个接口再进行限流。

EasySwoole 提供了一个基于 Atomic 计数器的限流器组件。可以直接使用,使用教程请移步查看限流器文档