123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232 |
- <?php
- namespace Illuminate\Tests\Integration\Queue;
- use Illuminate\Bus\Dispatcher;
- use Illuminate\Bus\Queueable;
- use Illuminate\Cache\RateLimiter;
- use Illuminate\Cache\RateLimiting\Limit;
- use Illuminate\Contracts\Cache\Repository as Cache;
- use Illuminate\Contracts\Queue\Job;
- use Illuminate\Queue\CallQueuedHandler;
- use Illuminate\Queue\InteractsWithQueue;
- use Illuminate\Queue\Middleware\RateLimited;
- use Mockery as m;
- use Orchestra\Testbench\TestCase;
- class RateLimitedTest extends TestCase
- {
- protected function tearDown(): void
- {
- parent::tearDown();
- m::close();
- }
- public function testUnlimitedJobsAreExecuted()
- {
- $rateLimiter = $this->app->make(RateLimiter::class);
- $rateLimiter->for('test', function ($job) {
- return Limit::none();
- });
- $this->assertJobRanSuccessfully(RateLimitedTestJob::class);
- $this->assertJobRanSuccessfully(RateLimitedTestJob::class);
- }
- public function testRateLimitedJobsAreNotExecutedOnLimitReached2()
- {
- $cache = m::mock(Cache::class);
- $cache->shouldReceive('get')->andReturn(0, 1, null);
- $cache->shouldReceive('add')->andReturn(true, true);
- $cache->shouldReceive('increment')->andReturn(1);
- $cache->shouldReceive('has')->andReturn(true);
- $rateLimiter = new RateLimiter($cache);
- $this->app->instance(RateLimiter::class, $rateLimiter);
- $rateLimiter = $this->app->make(RateLimiter::class);
- $rateLimiter->for('test', function ($job) {
- return Limit::perHour(1);
- });
- $this->assertJobRanSuccessfully(RateLimitedTestJob::class);
- // Assert Job was released and released with a delay greater than 0
- RateLimitedTestJob::$handled = false;
- $instance = new CallQueuedHandler(new Dispatcher($this->app), $this->app);
- $job = m::mock(Job::class);
- $job->shouldReceive('hasFailed')->once()->andReturn(false);
- $job->shouldReceive('release')->once()->withArgs(function ($delay) {
- return $delay >= 0;
- });
- $job->shouldReceive('isReleased')->andReturn(true);
- $job->shouldReceive('isDeletedOrReleased')->once()->andReturn(true);
- $instance->call($job, [
- 'command' => serialize($command = new RateLimitedTestJob),
- ]);
- $this->assertFalse(RateLimitedTestJob::$handled);
- }
- public function testRateLimitedJobsAreNotExecutedOnLimitReached()
- {
- $rateLimiter = $this->app->make(RateLimiter::class);
- $rateLimiter->for('test', function ($job) {
- return Limit::perHour(1);
- });
- $this->assertJobRanSuccessfully(RateLimitedTestJob::class);
- $this->assertJobWasReleased(RateLimitedTestJob::class);
- }
- public function testRateLimitedJobsCanBeSkippedOnLimitReached()
- {
- $rateLimiter = $this->app->make(RateLimiter::class);
- $rateLimiter->for('test', function ($job) {
- return Limit::perHour(1);
- });
- $this->assertJobRanSuccessfully(RateLimitedDontReleaseTestJob::class);
- $this->assertJobWasSkipped(RateLimitedDontReleaseTestJob::class);
- }
- public function testJobsCanHaveConditionalRateLimits()
- {
- $rateLimiter = $this->app->make(RateLimiter::class);
- $rateLimiter->for('test', function ($job) {
- if ($job->isAdmin()) {
- return Limit::none();
- }
- return Limit::perHour(1);
- });
- $this->assertJobRanSuccessfully(AdminTestJob::class);
- $this->assertJobRanSuccessfully(AdminTestJob::class);
- $this->assertJobRanSuccessfully(NonAdminTestJob::class);
- $this->assertJobWasReleased(NonAdminTestJob::class);
- }
- public function testMiddlewareSerialization()
- {
- $rateLimited = new RateLimited('limiterName');
- $rateLimited->shouldRelease = false;
- $restoredRateLimited = unserialize(serialize($rateLimited));
- $fetch = (function (string $name) {
- return $this->{$name};
- })->bindTo($restoredRateLimited, RateLimited::class);
- $this->assertFalse($restoredRateLimited->shouldRelease);
- $this->assertSame('limiterName', $fetch('limiterName'));
- $this->assertInstanceOf(RateLimiter::class, $fetch('limiter'));
- }
- protected function assertJobRanSuccessfully($class)
- {
- $class::$handled = false;
- $instance = new CallQueuedHandler(new Dispatcher($this->app), $this->app);
- $job = m::mock(Job::class);
- $job->shouldReceive('hasFailed')->once()->andReturn(false);
- $job->shouldReceive('isReleased')->andReturn(false);
- $job->shouldReceive('isDeletedOrReleased')->once()->andReturn(false);
- $job->shouldReceive('delete')->once();
- $instance->call($job, [
- 'command' => serialize($command = new $class),
- ]);
- $this->assertTrue($class::$handled);
- }
- protected function assertJobWasReleased($class)
- {
- $class::$handled = false;
- $instance = new CallQueuedHandler(new Dispatcher($this->app), $this->app);
- $job = m::mock(Job::class);
- $job->shouldReceive('hasFailed')->once()->andReturn(false);
- $job->shouldReceive('release')->once();
- $job->shouldReceive('isReleased')->andReturn(true);
- $job->shouldReceive('isDeletedOrReleased')->once()->andReturn(true);
- $instance->call($job, [
- 'command' => serialize($command = new $class),
- ]);
- $this->assertFalse($class::$handled);
- }
- protected function assertJobWasSkipped($class)
- {
- $class::$handled = false;
- $instance = new CallQueuedHandler(new Dispatcher($this->app), $this->app);
- $job = m::mock(Job::class);
- $job->shouldReceive('hasFailed')->once()->andReturn(false);
- $job->shouldReceive('isReleased')->andReturn(false);
- $job->shouldReceive('isDeletedOrReleased')->once()->andReturn(false);
- $job->shouldReceive('delete')->once();
- $instance->call($job, [
- 'command' => serialize($command = new $class),
- ]);
- $this->assertFalse($class::$handled);
- }
- }
- class RateLimitedTestJob
- {
- use InteractsWithQueue, Queueable;
- public static $handled = false;
- public function handle()
- {
- static::$handled = true;
- }
- public function middleware()
- {
- return [new RateLimited('test')];
- }
- }
- class AdminTestJob extends RateLimitedTestJob
- {
- public function isAdmin()
- {
- return true;
- }
- }
- class NonAdminTestJob extends RateLimitedTestJob
- {
- public function isAdmin()
- {
- return false;
- }
- }
- class RateLimitedDontReleaseTestJob extends RateLimitedTestJob
- {
- public function middleware()
- {
- return [(new RateLimited('test'))->dontRelease()];
- }
- }
|