2016-07-30 15:12:56 +03:00
|
|
|
<?php
|
2019-10-05 18:26:10 +03:00
|
|
|
|
2017-10-12 11:13:20 +03:00
|
|
|
declare(strict_types=1);
|
|
|
|
|
2018-09-16 19:36:02 +03:00
|
|
|
namespace ShlinkioTest\Shlink\CLI\Command\ShortUrl;
|
2016-07-30 15:12:56 +03:00
|
|
|
|
2018-11-17 20:06:06 +03:00
|
|
|
use PHPUnit\Framework\Assert;
|
2017-03-24 22:34:18 +03:00
|
|
|
use PHPUnit\Framework\TestCase;
|
2016-07-30 15:12:56 +03:00
|
|
|
use Prophecy\Argument;
|
|
|
|
use Prophecy\Prophecy\ObjectProphecy;
|
2021-12-09 11:45:15 +03:00
|
|
|
use Shlinkio\Shlink\CLI\Command\ShortUrl\CreateShortUrlCommand;
|
2019-10-02 21:22:42 +03:00
|
|
|
use Shlinkio\Shlink\CLI\Util\ExitCodes;
|
2018-09-12 21:32:58 +03:00
|
|
|
use Shlinkio\Shlink\Core\Entity\ShortUrl;
|
2016-07-30 15:12:56 +03:00
|
|
|
use Shlinkio\Shlink\Core\Exception\InvalidUrlException;
|
2019-10-02 21:22:42 +03:00
|
|
|
use Shlinkio\Shlink\Core\Exception\NonUniqueSlugException;
|
2020-09-24 22:54:03 +03:00
|
|
|
use Shlinkio\Shlink\Core\Model\ShortUrlMeta;
|
2016-07-30 15:12:56 +03:00
|
|
|
use Shlinkio\Shlink\Core\Service\UrlShortener;
|
2021-02-02 00:55:52 +03:00
|
|
|
use Shlinkio\Shlink\Core\ShortUrl\Helper\ShortUrlStringifierInterface;
|
2021-04-08 14:42:56 +03:00
|
|
|
use ShlinkioTest\Shlink\CLI\CliTestUtilsTrait;
|
2016-07-30 15:12:56 +03:00
|
|
|
use Symfony\Component\Console\Tester\CommandTester;
|
|
|
|
|
2021-12-09 11:45:15 +03:00
|
|
|
class CreateShortUrlCommandTest extends TestCase
|
2016-07-30 15:12:56 +03:00
|
|
|
{
|
2021-04-08 14:42:56 +03:00
|
|
|
use CliTestUtilsTrait;
|
2020-11-02 13:50:19 +03:00
|
|
|
|
2021-12-09 12:24:58 +03:00
|
|
|
private const DEFAULT_DOMAIN = 'default.com';
|
|
|
|
|
2019-12-30 00:27:00 +03:00
|
|
|
private CommandTester $commandTester;
|
|
|
|
private ObjectProphecy $urlShortener;
|
2021-02-02 00:55:52 +03:00
|
|
|
private ObjectProphecy $stringifier;
|
2016-07-30 15:12:56 +03:00
|
|
|
|
2019-02-16 12:53:45 +03:00
|
|
|
public function setUp(): void
|
2016-07-30 15:12:56 +03:00
|
|
|
{
|
|
|
|
$this->urlShortener = $this->prophesize(UrlShortener::class);
|
2021-02-02 00:55:52 +03:00
|
|
|
$this->stringifier = $this->prophesize(ShortUrlStringifierInterface::class);
|
|
|
|
$this->stringifier->stringify(Argument::type(ShortUrl::class))->willReturn('');
|
|
|
|
|
2021-12-09 12:24:58 +03:00
|
|
|
$command = new CreateShortUrlCommand(
|
|
|
|
$this->urlShortener->reveal(),
|
|
|
|
$this->stringifier->reveal(),
|
|
|
|
5,
|
|
|
|
self::DEFAULT_DOMAIN,
|
|
|
|
);
|
2021-04-08 14:42:56 +03:00
|
|
|
$this->commandTester = $this->testerForCommand($command);
|
2016-07-30 15:12:56 +03:00
|
|
|
}
|
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
/** @test */
|
2019-10-02 21:22:42 +03:00
|
|
|
public function properShortCodeIsCreatedIfLongUrlIsCorrect(): void
|
2016-07-30 15:12:56 +03:00
|
|
|
{
|
2021-01-30 16:18:44 +03:00
|
|
|
$shortUrl = ShortUrl::createEmpty();
|
2020-11-06 22:05:57 +03:00
|
|
|
$urlToShortCode = $this->urlShortener->shorten(Argument::cetera())->willReturn($shortUrl);
|
2021-02-02 00:55:52 +03:00
|
|
|
$stringify = $this->stringifier->stringify($shortUrl)->willReturn('stringified_short_url');
|
2016-07-30 15:12:56 +03:00
|
|
|
|
|
|
|
$this->commandTester->execute([
|
2017-10-12 11:13:20 +03:00
|
|
|
'longUrl' => 'http://domain.com/foo/bar',
|
2021-01-30 13:25:20 +03:00
|
|
|
'--max-visits' => '3',
|
2016-07-30 15:12:56 +03:00
|
|
|
]);
|
|
|
|
$output = $this->commandTester->getDisplay();
|
2018-11-17 20:06:06 +03:00
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(ExitCodes::EXIT_SUCCESS, $this->commandTester->getStatusCode());
|
2021-02-02 00:55:52 +03:00
|
|
|
self::assertStringContainsString('stringified_short_url', $output);
|
2018-11-17 20:06:06 +03:00
|
|
|
$urlToShortCode->shouldHaveBeenCalledOnce();
|
2021-02-02 00:55:52 +03:00
|
|
|
$stringify->shouldHaveBeenCalledOnce();
|
2016-07-30 15:12:56 +03:00
|
|
|
}
|
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
/** @test */
|
2019-10-02 21:22:42 +03:00
|
|
|
public function exceptionWhileParsingLongUrlOutputsError(): void
|
2016-07-30 15:12:56 +03:00
|
|
|
{
|
2019-11-28 20:42:27 +03:00
|
|
|
$url = 'http://domain.com/invalid';
|
2020-11-06 22:05:57 +03:00
|
|
|
$this->urlShortener->shorten(Argument::cetera())->willThrow(InvalidUrlException::fromUrl($url))
|
2018-11-11 15:18:21 +03:00
|
|
|
->shouldBeCalledOnce();
|
2016-07-30 15:12:56 +03:00
|
|
|
|
2019-11-28 20:42:27 +03:00
|
|
|
$this->commandTester->execute(['longUrl' => $url]);
|
2016-07-30 15:12:56 +03:00
|
|
|
$output = $this->commandTester->getDisplay();
|
2019-10-02 21:22:42 +03:00
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(ExitCodes::EXIT_FAILURE, $this->commandTester->getStatusCode());
|
|
|
|
self::assertStringContainsString('Provided URL http://domain.com/invalid is invalid.', $output);
|
2019-10-02 21:22:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @test */
|
|
|
|
public function providingNonUniqueSlugOutputsError(): void
|
|
|
|
{
|
2020-11-06 22:05:57 +03:00
|
|
|
$urlToShortCode = $this->urlShortener->shorten(Argument::cetera())->willThrow(
|
2020-01-01 22:48:31 +03:00
|
|
|
NonUniqueSlugException::fromSlug('my-slug'),
|
2016-07-30 15:12:56 +03:00
|
|
|
);
|
2019-10-02 21:22:42 +03:00
|
|
|
|
2021-01-30 13:25:20 +03:00
|
|
|
$this->commandTester->execute(['longUrl' => 'http://domain.com/invalid', '--custom-slug' => 'my-slug']);
|
2019-10-02 21:22:42 +03:00
|
|
|
$output = $this->commandTester->getDisplay();
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(ExitCodes::EXIT_FAILURE, $this->commandTester->getStatusCode());
|
|
|
|
self::assertStringContainsString('Provided slug "my-slug" is already in use', $output);
|
2019-10-02 21:22:42 +03:00
|
|
|
$urlToShortCode->shouldHaveBeenCalledOnce();
|
2016-07-30 15:12:56 +03:00
|
|
|
}
|
2018-11-17 20:06:06 +03:00
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
/** @test */
|
2019-10-02 21:22:42 +03:00
|
|
|
public function properlyProcessesProvidedTags(): void
|
2018-11-17 20:06:06 +03:00
|
|
|
{
|
2021-01-30 16:18:44 +03:00
|
|
|
$shortUrl = ShortUrl::createEmpty();
|
2020-11-06 22:05:57 +03:00
|
|
|
$urlToShortCode = $this->urlShortener->shorten(
|
2021-01-30 21:17:12 +03:00
|
|
|
Argument::that(function (ShortUrlMeta $meta) {
|
|
|
|
$tags = $meta->getTags();
|
2018-11-17 20:06:06 +03:00
|
|
|
Assert::assertEquals(['foo', 'bar', 'baz', 'boo', 'zar'], $tags);
|
2021-01-30 21:17:12 +03:00
|
|
|
return true;
|
2018-11-17 20:06:06 +03:00
|
|
|
}),
|
2019-10-11 10:14:25 +03:00
|
|
|
)->willReturn($shortUrl);
|
2021-02-02 00:55:52 +03:00
|
|
|
$stringify = $this->stringifier->stringify($shortUrl)->willReturn('stringified_short_url');
|
2018-11-17 20:06:06 +03:00
|
|
|
|
|
|
|
$this->commandTester->execute([
|
|
|
|
'longUrl' => 'http://domain.com/foo/bar',
|
2019-02-03 14:09:56 +03:00
|
|
|
'--tags' => ['foo,bar', 'baz', 'boo,zar,baz'],
|
2018-11-17 20:06:06 +03:00
|
|
|
]);
|
|
|
|
$output = $this->commandTester->getDisplay();
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(ExitCodes::EXIT_SUCCESS, $this->commandTester->getStatusCode());
|
2021-02-02 00:55:52 +03:00
|
|
|
self::assertStringContainsString('stringified_short_url', $output);
|
2018-11-17 20:06:06 +03:00
|
|
|
$urlToShortCode->shouldHaveBeenCalledOnce();
|
2021-02-02 00:55:52 +03:00
|
|
|
$stringify->shouldHaveBeenCalledOnce();
|
2018-11-17 20:06:06 +03:00
|
|
|
}
|
2020-09-24 22:54:03 +03:00
|
|
|
|
2021-12-09 12:24:58 +03:00
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @dataProvider provideDomains
|
|
|
|
*/
|
|
|
|
public function properlyProcessesProvidedDomain(array $input, ?string $expectedDomain): void
|
|
|
|
{
|
|
|
|
$shorten = $this->urlShortener->shorten(
|
|
|
|
Argument::that(function (ShortUrlMeta $meta) use ($expectedDomain) {
|
|
|
|
Assert::assertEquals($expectedDomain, $meta->getDomain());
|
|
|
|
return true;
|
|
|
|
}),
|
|
|
|
)->willReturn(ShortUrl::createEmpty());
|
|
|
|
|
|
|
|
$input['longUrl'] = 'http://domain.com/foo/bar';
|
|
|
|
$this->commandTester->execute($input);
|
|
|
|
|
|
|
|
self::assertEquals(ExitCodes::EXIT_SUCCESS, $this->commandTester->getStatusCode());
|
|
|
|
$shorten->shouldHaveBeenCalledOnce();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideDomains(): iterable
|
|
|
|
{
|
|
|
|
yield 'no domain' => [[], null];
|
|
|
|
yield 'non-default domain foo' => [['--domain' => 'foo.com'], 'foo.com'];
|
|
|
|
yield 'non-default domain bar' => [['-d' => 'bar.com'], 'bar.com'];
|
|
|
|
yield 'default domain' => [['--domain' => self::DEFAULT_DOMAIN], null];
|
|
|
|
}
|
|
|
|
|
2020-09-24 22:54:03 +03:00
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @dataProvider provideFlags
|
|
|
|
*/
|
|
|
|
public function urlValidationHasExpectedValueBasedOnProvidedTags(array $options, ?bool $expectedValidateUrl): void
|
|
|
|
{
|
2021-01-30 16:18:44 +03:00
|
|
|
$shortUrl = ShortUrl::createEmpty();
|
2020-11-06 22:05:57 +03:00
|
|
|
$urlToShortCode = $this->urlShortener->shorten(
|
2020-09-24 22:54:03 +03:00
|
|
|
Argument::that(function (ShortUrlMeta $meta) use ($expectedValidateUrl) {
|
|
|
|
Assert::assertEquals($expectedValidateUrl, $meta->doValidateUrl());
|
|
|
|
return $meta;
|
|
|
|
}),
|
|
|
|
)->willReturn($shortUrl);
|
|
|
|
|
|
|
|
$options['longUrl'] = 'http://domain.com/foo/bar';
|
|
|
|
$this->commandTester->execute($options);
|
|
|
|
|
|
|
|
$urlToShortCode->shouldHaveBeenCalledOnce();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideFlags(): iterable
|
|
|
|
{
|
|
|
|
yield 'no flags' => [[], null];
|
|
|
|
yield 'no-validate-url only' => [['--no-validate-url' => true], false];
|
|
|
|
yield 'validate-url' => [['--validate-url' => true], true];
|
|
|
|
yield 'both flags' => [['--validate-url' => true, '--no-validate-url' => true], false];
|
|
|
|
}
|
2016-07-30 15:12:56 +03:00
|
|
|
}
|