visitService = $this->prophesize(VisitService::class); $this->ipResolver = $this->prophesize(IpApiLocationResolver::class); $this->ipResolver->getApiLimit()->willReturn(10000000000); $command = new ProcessVisitsCommand( $this->visitService->reveal(), $this->ipResolver->reveal(), Translator::factory([]) ); $app = new Application(); $app->add($command); $this->commandTester = new CommandTester($command); } /** * @test */ public function allReturnedVisitsIpsAreProcessed() { $visits = [ (new Visit())->setRemoteAddr('1.2.3.4'), (new Visit())->setRemoteAddr('4.3.2.1'), (new Visit())->setRemoteAddr('12.34.56.78'), ]; $this->visitService->getUnlocatedVisits()->willReturn($visits) ->shouldBeCalledTimes(1); $this->visitService->saveVisit(Argument::any())->shouldBeCalledTimes(count($visits)); $this->ipResolver->resolveIpLocation(Argument::any())->willReturn([]) ->shouldBeCalledTimes(count($visits)); $this->commandTester->execute([ 'command' => 'visit:process', ]); $output = $this->commandTester->getDisplay(); $this->assertContains('Processing IP 1.2.3.0', $output); $this->assertContains('Processing IP 4.3.2.0', $output); $this->assertContains('Processing IP 12.34.56.0', $output); } /** * @test */ public function localhostAndEmptyAddressIsIgnored() { $visits = [ (new Visit())->setRemoteAddr('1.2.3.4'), (new Visit())->setRemoteAddr('4.3.2.1'), (new Visit())->setRemoteAddr('12.34.56.78'), (new Visit())->setRemoteAddr('127.0.0.1'), (new Visit())->setRemoteAddr('127.0.0.1'), (new Visit())->setRemoteAddr(''), (new Visit())->setRemoteAddr(null), ]; $this->visitService->getUnlocatedVisits()->willReturn($visits) ->shouldBeCalledTimes(1); $this->visitService->saveVisit(Argument::any())->shouldBeCalledTimes(count($visits) - 4); $this->ipResolver->resolveIpLocation(Argument::any())->willReturn([]) ->shouldBeCalledTimes(count($visits) - 4); $this->commandTester->execute([ 'command' => 'visit:process', ], ['verbosity' => OutputInterface::VERBOSITY_VERBOSE]); $output = $this->commandTester->getDisplay(); $this->assertContains('Ignored localhost address', $output); $this->assertContains('Ignored visit with no IP address', $output); } /** * @test */ public function sleepsEveryTimeTheApiLimitIsReached() { $visits = [ (new Visit())->setRemoteAddr('1.2.3.4'), (new Visit())->setRemoteAddr('4.3.2.1'), (new Visit())->setRemoteAddr('12.34.56.78'), (new Visit())->setRemoteAddr('1.2.3.4'), (new Visit())->setRemoteAddr('4.3.2.1'), (new Visit())->setRemoteAddr('12.34.56.78'), (new Visit())->setRemoteAddr('1.2.3.4'), (new Visit())->setRemoteAddr('4.3.2.1'), (new Visit())->setRemoteAddr('12.34.56.78'), (new Visit())->setRemoteAddr('4.3.2.1'), ]; $apiLimit = 3; $this->visitService->getUnlocatedVisits()->willReturn($visits); $this->visitService->saveVisit(Argument::any())->will(function () { }); $getApiLimit = $this->ipResolver->getApiLimit()->willReturn($apiLimit); $getApiInterval = $this->ipResolver->getApiInterval()->willReturn(0); $resolveIpLocation = $this->ipResolver->resolveIpLocation(Argument::any())->willReturn([]) ->shouldBeCalledTimes(count($visits)); $this->commandTester->execute([ 'command' => 'visit:process', ]); $getApiLimit->shouldHaveBeenCalledTimes(count($visits)); $getApiInterval->shouldHaveBeenCalledTimes(round(count($visits) / $apiLimit)); $resolveIpLocation->shouldHaveBeenCalledTimes(count($visits)); } }