简介

Laravel 包含了一个开箱即用的,基于 缓存 实现的限流器,提供了一个简单的方法来限制指定时间内的任何操作。

技巧
了解更多关于如何限制 HTTP 请求,请参考 请求频率限制中间件。

缓存配置

通常情况下,限流器使用你默认的缓存驱动,由 php cache 配置文件中的 php default 键定义。你也可以通过在你的应用程序的 php cache 配置文件中定义一个 php limiter 来指定限流器应该使用哪一个缓存来驱动:

  1. 'default' => 'memcached',
  2. 'limiter' => 'redis',

基本用法

可以通过 php Illuminate\Support\Facades\RateLimiter 来操作限流器。限流器提供的最简单的方法是 php attempt 方法,它将一个给定的回调函数执行次数限制在一个给定的秒数内。

当回调函数执行次数超过限制时, php attempt 方法返回 php false ;否则, php attempt 方法将返回回调的结果或 php truephp attempt 方法接受的第一个参数是一个速率限制器 「key」 ,它可以是你选择的任何字符串,代表被限制速率的动作:

  1. use Illuminate\Support\Facades\RateLimiter;
  2. $executed = RateLimiter::attempt(
  3. 'send-message:'.$user->id,
  4. $perMinute = 5,
  5. function() {
  6. // 发送消息...
  7. }
  8. );
  9. if (! $executed) {
  10. return 'Too many messages sent!';
  11. }

手动配置尝试次数

如果您想手动与限流器交互,可以使用多种方法。例如,您可以调用 php tooManyAttempts 方法来确定给定的限流器是否超过了每分钟允许的最大尝试次数:

  1. use Illuminate\Support\Facades\RateLimiter;
  2. if (RateLimiter::tooManyAttempts('send-message:'.$user->id, $perMinute = 5)) {
  3. return 'Too many attempts!';
  4. }

或者,您可以使用 php remaining 方法检索给定密钥的剩余尝试次数。如果给定的密钥还有重试次数,您可以调用 php hit 方法来增加总尝试次数:

  1. use Illuminate\Support\Facades\RateLimiter;
  2. if (RateLimiter::remaining('send-message:'.$user->id, $perMinute = 5)) {
  3. RateLimiter::hit('send-message:'.$user->id);
  4. // 发送消息...
  5. }

确定限流器可用性
当一个键没有更多的尝试次数时,php availableIn 方法返回在尝试可用之前需等待的剩余秒数:

  1. use Illuminate\Support\Facades\RateLimiter;
  2. if (RateLimiter::tooManyAttempts('send-message:'.$user->id, $perMinute = 5)) {
  3. $seconds = RateLimiter::availableIn('send-message:'.$user->id);
  4. return 'You may try again in '.$seconds.' seconds.';
  5. }

清除尝试次数

您可以使用 php clear 方法重置给定速率限制键的尝试次数。例如,当接收者读取给定消息时,您可以重置尝试次数:

  1. use App\Models\Message;
  2. use Illuminate\Support\Facades\RateLimiter;
  3. /**
  4. * 标记消息为已读。
  5. */
  6. public function read(Message $message): Message
  7. {
  8. $message->markAsRead();
  9. RateLimiter::clear('send-message:'.$message->user_id);
  10. return $message;
  11. }

本文章首发在 网站上。