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() { $shortUrl = new ShortUrl(''); $visits = [ new Visit($shortUrl, new Visitor('', '', '1.2.3.4')), new Visit($shortUrl, new Visitor('', '', '4.3.2.1')), new Visit($shortUrl, new Visitor('', '', '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() { $shortUrl = new ShortUrl(''); $visits = [ new Visit($shortUrl, new Visitor('', '', '1.2.3.4')), new Visit($shortUrl, new Visitor('', '', '4.3.2.1')), new Visit($shortUrl, new Visitor('', '', '12.34.56.78')), new Visit($shortUrl, new Visitor('', '', '127.0.0.1')), new Visit($shortUrl, new Visitor('', '', '127.0.0.1')), new Visit($shortUrl, new Visitor('', '', '')), new Visit($shortUrl, new Visitor('', '', 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() { $shortUrl = new ShortUrl(''); $visits = [ new Visit($shortUrl, new Visitor('', '', '1.2.3.4')), new Visit($shortUrl, new Visitor('', '', '4.3.2.1')), new Visit($shortUrl, new Visitor('', '', '12.34.56.78')), new Visit($shortUrl, new Visitor('', '', '1.2.3.4')), new Visit($shortUrl, new Visitor('', '', '4.3.2.1')), new Visit($shortUrl, new Visitor('', '', '12.34.56.78')), new Visit($shortUrl, new Visitor('', '', '1.2.3.4')), new Visit($shortUrl, new Visitor('', '', '4.3.2.1')), new Visit($shortUrl, new Visitor('', '', '12.34.56.78')), new Visit($shortUrl, new Visitor('', '', '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)); } }