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;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 |