Skip to content

Instantly share code, notes, and snippets.

@frknikiz
Created February 4, 2016 13:42
Show Gist options
  • Save frknikiz/1ae256d26bca5036c40b to your computer and use it in GitHub Desktop.
Save frknikiz/1ae256d26bca5036c40b to your computer and use it in GitHub Desktop.
Rate Limiter Middleware for Laravel 5.1
<?php
namespace App\Http\Middleware;
use App\Exceptions\ThrottleException;
use Closure;
use Illuminate\Cache\RateLimiter;
use Response;
use RuntimeException;
class ThrottleRequests
{
/**
* The rate limiter instance.
*
* @var \Illuminate\Cache\RateLimiter
*/
protected $limiter;
/**
* Create a new request throttler.
*
* @param \Illuminate\Cache\RateLimiter $limiter
*/
public function __construct(RateLimiter $limiter)
{
$this->limiter = $limiter;
}
/**
* Handle an incoming request.
*
* @param \Illuminate\Http\Request $request
* @param \Closure $next
* @param int $maxAttempts
* @param int $decayMinutes
*
* @return mixed
* @throws ThrottleException
*/
public function handle($request, Closure $next, $maxAttempts = 60, $decayMinutes = 1)
{
$key = $this->resolveRequestSignature($request);
if ($this->limiter->tooManyAttempts($key, $maxAttempts, $decayMinutes)) {
return new Response('Too Many Attempts.', 429, [
'Retry-After' => $this->limiter->availableIn($key),
'X-RateLimit-Limit' => $maxAttempts,
'X-RateLimit-Remaining' => 0,
]);
}
$this->limiter->hit($key, $decayMinutes);
$response = $next($request);
$response->headers->add([
'X-RateLimit-Limit' => $maxAttempts,
'X-RateLimit-Remaining' => $maxAttempts - $this->limiter->attempts($key) + 1,
]);
return $response;
}
/**
* Resolve request signature.
*
* @param \Illuminate\Http\Request $request
* @return string
*/
protected function resolveRequestSignature($request)
{
return $this->fingerprint($request);
}
/**
* Get a unique fingerprint for the request / route / IP address.
*
* @return string
*
* @throws \RuntimeException
*/
public function fingerprint($request)
{
if (! $request->route()) {
throw new RuntimeException('Unable to generate fingerprint. Route unavailable.');
}
return sha1(
implode('|', $request->route()->methods()).
'|'.$request->route()->domain().
'|'.$request->route()->uri().
'|'.$request->ip()
);
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment