ThrottlesExceptions.php
TLDR
The provided file, ThrottlesExceptions.php
, is a part of the Illuminate\Queue\Middleware
namespace in the Laravel framework. It contains a class named ThrottlesExceptions
. This class is responsible for handling rate limiting behavior for queued jobs.
Methods
__construct($maxAttempts = 10, $decaySeconds = 600)
This method is the constructor of the ThrottlesExceptions
class. It initializes the maximum number of attempts allowed before rate limiting applies ($maxAttempts
) and the number of seconds until the maximum attempts are reset ($decaySeconds
).
handle($job, $next)
This method is responsible for processing the job. It checks if the number of attempts for the job has reached the maximum limit. If it has, the job is released with a delay until the next retry. If not, the job is executed and the rate limiter is cleared. If an exception occurs during the job execution, the rate limiter is hit and the job is released with a delay determined by the $retryAfterMinutes
property.
when(callable $callback)
This method allows specifying a callback that determines if rate limiting behavior should apply. it sets the $whenCallback
property.
withPrefix(string $prefix)
This method sets the prefix of the rate limiter key. It sets the $prefix
property.
backoff($backoff)
This method sets the number of minutes a job should be delayed when it is released before it has reached its maximum exceptions. It sets the $retryAfterMinutes
property.
by($key)
This method sets the value that the rate limiter should be keyed by. It sets the $key
property.
byJob()
This method indicates that the throttle key should use the job's UUID. It sets the $byJob
property to true
.
Classes
There are no classes mentioned in the provided file.
<?php
namespace Illuminate\Queue\Middleware;
use Illuminate\Cache\RateLimiter;
use Illuminate\Container\Container;
use Throwable;
class ThrottlesExceptions
{
/**
* The developer specified key that the rate limiter should use.
*
* @var string
*/
protected $key;
/**
* Indicates whether the throttle key should use the job's UUID.
*
* @var bool
*/
protected $byJob = false;
/**
* The maximum number of attempts allowed before rate limiting applies.
*
* @var int
*/
protected $maxAttempts;
/**
* The number of seconds until the maximum attempts are reset.
*
* @var int
*/
protected $decaySeconds;
/**
* The number of minutes to wait before retrying the job after an exception.
*
* @var int
*/
protected $retryAfterMinutes = 0;
/**
* The callback that determines if rate limiting should apply.
*
* @var callable
*/
protected $whenCallback;
/**
* The prefix of the rate limiter key.
*
* @var string
*/
protected $prefix = 'laravel_throttles_exceptions:';
/**
* The rate limiter instance.
*
* @var \Illuminate\Cache\RateLimiter
*/
protected $limiter;
/**
* Create a new middleware instance.
*
* @param int $maxAttempts
* @param int $decaySeconds
* @return void
*/
public function __construct($maxAttempts = 10, $decaySeconds = 600)
{
$this->maxAttempts = $maxAttempts;
$this->decaySeconds = $decaySeconds;
}
/**
* Process the job.
*
* @param mixed $job
* @param callable $next
* @return mixed
*/
public function handle($job, $next)
{
$this->limiter = Container::getInstance()->make(RateLimiter::class);
if ($this->limiter->tooManyAttempts($jobKey = $this->getKey($job), $this->maxAttempts)) {
return $job->release($this->getTimeUntilNextRetry($jobKey));
}
try {
$next($job);
$this->limiter->clear($jobKey);
} catch (Throwable $throwable) {
if ($this->whenCallback && ! call_user_func($this->whenCallback, $throwable)) {
throw $throwable;
}
$this->limiter->hit($jobKey, $this->decaySeconds);
return $job->release($this->retryAfterMinutes * 60);
}
}
/**
* Specify a callback that should determine if rate limiting behavior should apply.
*
* @param callable $callback
* @return $this
*/
public function when(callable $callback)
{
$this->whenCallback = $callback;
return $this;
}
/**
* Set the prefix of the rate limiter key.
*
* @param string $prefix
* @return $this
*/
public function withPrefix(string $prefix)
{
$this->prefix = $prefix;
return $this;
}
/**
* Specify the number of minutes a job should be delayed when it is released (before it has reached its max exceptions).
*
* @param int $backoff
* @return $this
*/
public function backoff($backoff)
{
$this->retryAfterMinutes = $backoff;
return $this;
}
/**
* Get the cache key associated for the rate limiter.
*
* @param mixed $job
* @return string
*/
protected function getKey($job)
{
if ($this->key) {
return $this->prefix.$this->key;
} elseif ($this->byJob) {
return $this->prefix.$job->job->uuid();
}
return $this->prefix.md5(get_class($job));
}
/**
* Set the value that the rate limiter should be keyed by.
*
* @param string $key
* @return $this
*/
public function by($key)
{
$this->key = $key;
return $this;
}
/**
* Indicate that the throttle key should use the job's UUID.
*
* @return $this
*/
public function byJob()
{
$this->byJob = true;
return $this;
}
/**
* Get the number of seconds that should elapse before the job is retried.
*
* @param string $key
* @return int
*/
protected function getTimeUntilNextRetry($key)
{
return $this->limiter->availableIn($key) + 3;
}
}