82 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			82 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			PHP
		
	
	
| <?php
 | |
| 
 | |
| /*
 | |
|  * This file is part of the Symfony package.
 | |
|  *
 | |
|  * (c) Fabien Potencier <fabien@symfony.com>
 | |
|  *
 | |
|  * For the full copyright and license information, please view the LICENSE
 | |
|  * file that was distributed with this source code.
 | |
|  */
 | |
| 
 | |
| namespace Symfony\Component\HttpFoundation\RateLimiter;
 | |
| 
 | |
| use Symfony\Component\HttpFoundation\Request;
 | |
| use Symfony\Component\RateLimiter\LimiterInterface;
 | |
| use Symfony\Component\RateLimiter\Policy\NoLimiter;
 | |
| use Symfony\Component\RateLimiter\RateLimit;
 | |
| 
 | |
| /**
 | |
|  * An implementation of PeekableRequestRateLimiterInterface that
 | |
|  * fits most use-cases.
 | |
|  *
 | |
|  * @author Wouter de Jong <wouter@wouterj.nl>
 | |
|  */
 | |
| abstract class AbstractRequestRateLimiter implements PeekableRequestRateLimiterInterface
 | |
| {
 | |
|     public function consume(Request $request): RateLimit
 | |
|     {
 | |
|         return $this->doConsume($request, 1);
 | |
|     }
 | |
| 
 | |
|     public function peek(Request $request): RateLimit
 | |
|     {
 | |
|         return $this->doConsume($request, 0);
 | |
|     }
 | |
| 
 | |
|     private function doConsume(Request $request, int $tokens): RateLimit
 | |
|     {
 | |
|         $limiters = $this->getLimiters($request);
 | |
|         if (0 === \count($limiters)) {
 | |
|             $limiters = [new NoLimiter()];
 | |
|         }
 | |
| 
 | |
|         $minimalRateLimit = null;
 | |
|         foreach ($limiters as $limiter) {
 | |
|             $rateLimit = $limiter->consume($tokens);
 | |
| 
 | |
|             $minimalRateLimit = $minimalRateLimit ? self::getMinimalRateLimit($minimalRateLimit, $rateLimit) : $rateLimit;
 | |
|         }
 | |
| 
 | |
|         return $minimalRateLimit;
 | |
|     }
 | |
| 
 | |
|     public function reset(Request $request): void
 | |
|     {
 | |
|         foreach ($this->getLimiters($request) as $limiter) {
 | |
|             $limiter->reset();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return LimiterInterface[] a set of limiters using keys extracted from the request
 | |
|      */
 | |
|     abstract protected function getLimiters(Request $request): array;
 | |
| 
 | |
|     private static function getMinimalRateLimit(RateLimit $first, RateLimit $second): RateLimit
 | |
|     {
 | |
|         if ($first->isAccepted() !== $second->isAccepted()) {
 | |
|             return $first->isAccepted() ? $second : $first;
 | |
|         }
 | |
| 
 | |
|         $firstRemainingTokens = $first->getRemainingTokens();
 | |
|         $secondRemainingTokens = $second->getRemainingTokens();
 | |
| 
 | |
|         if ($firstRemainingTokens === $secondRemainingTokens) {
 | |
|             return $first->getRetryAfter() < $second->getRetryAfter() ? $second : $first;
 | |
|         }
 | |
| 
 | |
|         return $firstRemainingTokens > $secondRemainingTokens ? $second : $first;
 | |
|     }
 | |
| }
 |