urlValidator = $this->prophesize(UrlValidatorInterface::class); $this->em = $this->prophesize(EntityManagerInterface::class); $conn = $this->prophesize(Connection::class); $conn->isTransactionActive()->willReturn(false); $this->em->getConnection()->willReturn($conn->reveal()); $this->em->flush()->willReturn(null); $this->em->commit()->willReturn(null); $this->em->beginTransaction()->willReturn(null); $this->em->persist(Argument::any())->will(function ($arguments): void { /** @var ShortUrl $shortUrl */ [$shortUrl] = $arguments; $shortUrl->setId('10'); }); $repo = $this->prophesize(ShortUrlRepository::class); $repo->shortCodeIsInUse(Argument::cetera())->willReturn(false); $this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal()); $this->setUrlShortener(false); } private function setUrlShortener(bool $urlValidationEnabled): void { $this->urlShortener = new UrlShortener( $this->urlValidator->reveal(), $this->em->reveal(), new UrlShortenerOptions(['validate_url' => $urlValidationEnabled]), ); } /** @test */ public function urlIsProperlyShortened(): void { $shortUrl = $this->urlShortener->urlToShortCode( new Uri('http://foobar.com/12345/hello?foo=bar'), [], ShortUrlMeta::createEmpty(), ); $this->assertEquals('http://foobar.com/12345/hello?foo=bar', $shortUrl->getLongUrl()); } /** @test */ public function shortCodeIsRegeneratedIfAlreadyInUse(): void { $callIndex = 0; $expectedCalls = 3; $repo = $this->prophesize(ShortUrlRepository::class); $shortCodeIsInUse = $repo->shortCodeIsInUse(Argument::cetera())->will( function () use (&$callIndex, $expectedCalls) { $callIndex++; return $callIndex < $expectedCalls; }, ); $repo->findBy(Argument::cetera())->willReturn([]); $getRepo = $this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal()); $shortUrl = $this->urlShortener->urlToShortCode( new Uri('http://foobar.com/12345/hello?foo=bar'), [], ShortUrlMeta::createEmpty(), ); $this->assertEquals('http://foobar.com/12345/hello?foo=bar', $shortUrl->getLongUrl()); $getRepo->shouldBeCalledTimes($expectedCalls); $shortCodeIsInUse->shouldBeCalledTimes($expectedCalls); } /** @test */ public function transactionIsRolledBackAndExceptionRethrownWhenExceptionIsThrown(): void { $conn = $this->prophesize(Connection::class); $conn->isTransactionActive()->willReturn(true); $this->em->getConnection()->willReturn($conn->reveal()); $this->em->rollback()->shouldBeCalledOnce(); $this->em->close()->shouldBeCalledOnce(); $this->em->flush()->willThrow(new ORMException()); $this->expectException(ORMException::class); $this->urlShortener->urlToShortCode( new Uri('http://foobar.com/12345/hello?foo=bar'), [], ShortUrlMeta::createEmpty(), ); } /** @test */ public function validatorIsCalledWhenUrlValidationIsEnabled(): void { $this->setUrlShortener(true); $validateUrl = $this->urlValidator->validateUrl('http://foobar.com/12345/hello?foo=bar')->will( function (): void { }, ); $this->urlShortener->urlToShortCode( new Uri('http://foobar.com/12345/hello?foo=bar'), [], ShortUrlMeta::createEmpty(), ); $validateUrl->shouldHaveBeenCalledOnce(); } /** @test */ public function exceptionIsThrownWhenNonUniqueSlugIsProvided(): void { $repo = $this->prophesize(ShortUrlRepository::class); $shortCodeIsInUse = $repo->shortCodeIsInUse('custom-slug', null)->willReturn(true); $repo->findBy(Argument::cetera())->willReturn([]); $getRepo = $this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal()); $shortCodeIsInUse->shouldBeCalledOnce(); $getRepo->shouldBeCalled(); $this->expectException(NonUniqueSlugException::class); $this->urlShortener->urlToShortCode( new Uri('http://foobar.com/12345/hello?foo=bar'), [], ShortUrlMeta::fromRawData(['customSlug' => 'custom-slug']), ); } /** * @test * @dataProvider provideExistingShortUrls */ public function existingShortUrlIsReturnedWhenRequested( string $url, array $tags, ShortUrlMeta $meta, ShortUrl $expected ): void { $repo = $this->prophesize(ShortUrlRepository::class); $findExisting = $repo->findBy(Argument::any())->willReturn([$expected]); $getRepo = $this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal()); $result = $this->urlShortener->urlToShortCode(new Uri($url), $tags, $meta); $findExisting->shouldHaveBeenCalledOnce(); $getRepo->shouldHaveBeenCalledOnce(); $this->em->persist(Argument::cetera())->shouldNotHaveBeenCalled(); $this->assertSame($expected, $result); } public function provideExistingShortUrls(): iterable { $url = 'http://foo.com'; yield [$url, [], ShortUrlMeta::fromRawData(['findIfExists' => true]), new ShortUrl($url)]; yield [$url, [], ShortUrlMeta::fromRawData( ['findIfExists' => true, 'customSlug' => 'foo'], ), new ShortUrl($url)]; yield [ $url, ['foo', 'bar'], ShortUrlMeta::fromRawData(['findIfExists' => true]), (new ShortUrl($url))->setTags(new ArrayCollection([new Tag('bar'), new Tag('foo')])), ]; yield [ $url, [], ShortUrlMeta::fromRawData(['findIfExists' => true, 'maxVisits' => 3]), new ShortUrl($url, ShortUrlMeta::fromRawData(['maxVisits' => 3])), ]; yield [ $url, [], ShortUrlMeta::fromRawData(['findIfExists' => true, 'validSince' => Chronos::parse('2017-01-01')]), new ShortUrl($url, ShortUrlMeta::fromRawData(['validSince' => Chronos::parse('2017-01-01')])), ]; yield [ $url, [], ShortUrlMeta::fromRawData(['findIfExists' => true, 'validUntil' => Chronos::parse('2017-01-01')]), new ShortUrl($url, ShortUrlMeta::fromRawData(['validUntil' => Chronos::parse('2017-01-01')])), ]; yield [ $url, [], ShortUrlMeta::fromRawData(['findIfExists' => true, 'domain' => 'example.com']), new ShortUrl($url, ShortUrlMeta::fromRawData(['domain' => 'example.com'])), ]; yield [ $url, ['baz', 'foo', 'bar'], ShortUrlMeta::fromRawData([ 'findIfExists' => true, 'validUntil' => Chronos::parse('2017-01-01'), 'maxVisits' => 4, ]), (new ShortUrl($url, ShortUrlMeta::fromRawData([ 'validUntil' => Chronos::parse('2017-01-01'), 'maxVisits' => 4, ])))->setTags(new ArrayCollection([new Tag('foo'), new Tag('bar'), new Tag('baz')])), ]; } /** @test */ public function properExistingShortUrlIsReturnedWhenMultipleMatch(): void { $url = 'http://foo.com'; $tags = ['baz', 'foo', 'bar']; $meta = ShortUrlMeta::fromRawData([ 'findIfExists' => true, 'validUntil' => Chronos::parse('2017-01-01'), 'maxVisits' => 4, ]); $tagsCollection = new ArrayCollection(array_map(fn (string $tag) => new Tag($tag), $tags)); $expected = (new ShortUrl($url, $meta))->setTags($tagsCollection); $repo = $this->prophesize(ShortUrlRepository::class); $findExisting = $repo->findBy(Argument::any())->willReturn([ new ShortUrl($url), new ShortUrl($url, $meta), $expected, (new ShortUrl($url))->setTags($tagsCollection), ]); $getRepo = $this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal()); $result = $this->urlShortener->urlToShortCode(new Uri($url), $tags, $meta); $this->assertSame($expected, $result); $findExisting->shouldHaveBeenCalledOnce(); $getRepo->shouldHaveBeenCalledOnce(); } }