visitService = $this->prophesize(VisitService::class); $this->ipResolver = $this->prophesize(IpApiLocationResolver::class); $this->locker = $this->prophesize(Lock\Factory::class); $this->lock = $this->prophesize(Lock\LockInterface::class); $this->lock->acquire()->willReturn(true); $this->lock->release()->will(function () { }); $this->locker->createLock(Argument::type('string'))->willReturn($this->lock->reveal()); $command = new ProcessVisitsCommand( $this->visitService->reveal(), $this->ipResolver->reveal(), $this->locker->reveal() ); $app = new Application(); $app->add($command); $this->commandTester = new CommandTester($command); } /** * @test */ public function allPendingVisitsAreProcessed() { $visit = new Visit(new ShortUrl(''), new Visitor('', '', '1.2.3.4')); $location = new VisitLocation([]); $locateVisits = $this->visitService->locateVisits(Argument::cetera())->will( function (array $args) use ($visit, $location) { $firstCallback = array_shift($args); $firstCallback($visit); $secondCallback = array_shift($args); $secondCallback($location, $visit); } ); $resolveIpLocation = $this->ipResolver->resolveIpLocation(Argument::any())->willReturn([]); $this->commandTester->execute([ 'command' => 'visit:process', ]); $output = $this->commandTester->getDisplay(); $this->assertStringContainsString('Processing IP 1.2.3.0', $output); $locateVisits->shouldHaveBeenCalledOnce(); $resolveIpLocation->shouldHaveBeenCalledOnce(); } /** * @test * @dataProvider provideIgnoredAddresses */ public function localhostAndEmptyAddressesAreIgnored(?string $address, string $message) { $visit = new Visit(new ShortUrl(''), new Visitor('', '', $address)); $location = new VisitLocation([]); $locateVisits = $this->visitService->locateVisits(Argument::cetera())->will( function (array $args) use ($visit, $location) { $firstCallback = array_shift($args); $firstCallback($visit); $secondCallback = array_shift($args); $secondCallback($location, $visit); } ); $resolveIpLocation = $this->ipResolver->resolveIpLocation(Argument::any())->willReturn([]); try { $this->commandTester->execute([ 'command' => 'visit:process', ], ['verbosity' => OutputInterface::VERBOSITY_VERBOSE]); } catch (Throwable $e) { $output = $this->commandTester->getDisplay(); $this->assertInstanceOf(IpCannotBeLocatedException::class, $e); $this->assertStringContainsString($message, $output); $locateVisits->shouldHaveBeenCalledOnce(); $resolveIpLocation->shouldNotHaveBeenCalled(); } } public function provideIgnoredAddresses(): array { return [ ['', 'Ignored visit with no IP address'], [null, 'Ignored visit with no IP address'], [IpAddress::LOCALHOST, 'Ignored localhost address'], ]; } /** * @test */ public function errorWhileLocatingIpIsDisplayed() { $visit = new Visit(new ShortUrl(''), new Visitor('', '', '1.2.3.4')); $location = new VisitLocation([]); $locateVisits = $this->visitService->locateVisits(Argument::cetera())->will( function (array $args) use ($visit, $location) { $firstCallback = array_shift($args); $firstCallback($visit); $secondCallback = array_shift($args); $secondCallback($location, $visit); } ); $resolveIpLocation = $this->ipResolver->resolveIpLocation(Argument::any())->willThrow(WrongIpException::class); try { $this->commandTester->execute([ 'command' => 'visit:process', ], ['verbosity' => OutputInterface::VERBOSITY_VERBOSE]); } catch (Throwable $e) { $output = $this->commandTester->getDisplay(); $this->assertInstanceOf(IpCannotBeLocatedException::class, $e); $this->assertStringContainsString('An error occurred while locating IP. Skipped', $output); $locateVisits->shouldHaveBeenCalledOnce(); $resolveIpLocation->shouldHaveBeenCalledOnce(); } } /** * @test */ public function noActionIsPerformedIfLockIsAcquired() { $this->lock->acquire()->willReturn(false); $locateVisits = $this->visitService->locateVisits(Argument::cetera())->will(function () { }); $resolveIpLocation = $this->ipResolver->resolveIpLocation(Argument::any())->willReturn([]); $this->commandTester->execute([ 'command' => 'visit:process', ], ['verbosity' => OutputInterface::VERBOSITY_VERBOSE]); $output = $this->commandTester->getDisplay(); $this->assertStringContainsString( sprintf('There is already an instance of the "%s" command', ProcessVisitsCommand::NAME), $output ); $locateVisits->shouldNotHaveBeenCalled(); $resolveIpLocation->shouldNotHaveBeenCalled(); } }