RateLimiter.php 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. <?php
  2. declare(strict_types=1);
  3. // +----------------------------------------------------------------------
  4. // | CatchAdmin [Just Like ~ ]
  5. // +----------------------------------------------------------------------
  6. // | Copyright (c) 2017~2020 http://catchadmin.com All rights reserved.
  7. // +----------------------------------------------------------------------
  8. // | Licensed ( https://github.com/yanwenwu/catch-admin/blob/master/LICENSE.txt )
  9. // +----------------------------------------------------------------------
  10. // | Author: JaguarJack [ njphper@gmail.com ]
  11. // +----------------------------------------------------------------------
  12. namespace catcher\library\rate;
  13. use catcher\exceptions\FailedException;
  14. class RateLimiter
  15. {
  16. use Redis;
  17. protected $key;
  18. /**
  19. * 令牌容量
  20. *
  21. * @var int
  22. */
  23. protected $capacity = 5;
  24. /**
  25. * 每次添加 token 的数量
  26. *
  27. * @var int
  28. */
  29. protected $eachTokens = 5;
  30. /**
  31. * 添加 token 的时间
  32. *
  33. * @var string
  34. */
  35. protected $addTokenTimeKey = '_add_token';
  36. /**
  37. * 添加 token 的时间间隔 /s
  38. *
  39. * @var int
  40. */
  41. protected $interval = 5;
  42. /**
  43. * RateLimiter constructor.
  44. * @param $key
  45. */
  46. public function __construct($key)
  47. {
  48. $this->key = $key;
  49. }
  50. /**
  51. * 处理
  52. *
  53. * @time 2020年07月02日
  54. * @return void
  55. */
  56. public function overflow()
  57. {
  58. // 添加 token
  59. if ($this->canAddToken()) {
  60. $this->addTokens();
  61. }
  62. if (!$this->tokens()) {
  63. throw new FailedException('访问限制');
  64. }
  65. // 每次请求拿走一个 token
  66. $this->removeToken();
  67. }
  68. /**
  69. *
  70. *
  71. * @time 2020年07月02日
  72. d * @return void
  73. */
  74. protected function addTokens()
  75. {
  76. $leftTokens = $this->capacity - $this->tokens();
  77. $tokens = array_fill(0, $leftTokens < $this->eachTokens ? $leftTokens : $this->eachTokens, 1);
  78. $this->getRedis()->lPush($this->key, ...$tokens);
  79. $this->rememberAddTokenTime();
  80. }
  81. /**
  82. * 拿走一个 token
  83. *
  84. * @time 2020年07月02日
  85. * @return void
  86. */
  87. protected function removeToken()
  88. {
  89. $this->getRedis()->rPop($this->key);
  90. }
  91. /**
  92. * 设置令牌桶数量
  93. *
  94. * @time 2020年07月02日
  95. * @param $capacity
  96. * @return $this
  97. */
  98. public function setCapacity($capacity)
  99. {
  100. $this->capacity = $capacity;
  101. return $this;
  102. }
  103. /**
  104. * 剩余的 token 数量
  105. *
  106. * @time 2020年07月02日
  107. * @return bool|int
  108. */
  109. protected function tokens()
  110. {
  111. return $this->getRedis()->lLen($this->key);
  112. }
  113. /**
  114. * 设置时间间隔
  115. *
  116. * @time 2020年07月02日
  117. * @param $seconds
  118. * @return $this
  119. */
  120. public function setInterval($seconds)
  121. {
  122. $this->interval = $seconds;
  123. return $this;
  124. }
  125. /**
  126. * 是否可以添加 token
  127. *
  128. * @time 2020年07月02日
  129. * @return bool
  130. */
  131. protected function canAddToken()
  132. {
  133. $currentTime = \time();
  134. $lastAddTokenTime = $this->getRedis()->get($this->key. $this->addTokenTimeKey);
  135. // 如果是满的 则不添加
  136. if ($this->tokens() == $this->capacity) {
  137. return false;
  138. }
  139. return ($currentTime - $lastAddTokenTime) > $this->interval;
  140. }
  141. /**
  142. * 记录添加 token 的时间
  143. *
  144. * @time 2020年07月02日
  145. * @return void
  146. */
  147. protected function rememberAddTokenTime()
  148. {
  149. $this->getRedis()->set($this->key. $this->addTokenTimeKey, time());
  150. }
  151. }