2019-01-30 20:28:07 +03:00
|
|
|
<?php
|
2019-10-05 18:26:10 +03:00
|
|
|
|
2019-01-30 20:28:07 +03:00
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace ShlinkioApiTest\Shlink\Rest\Action;
|
|
|
|
|
|
|
|
use Cake\Chronos\Chronos;
|
|
|
|
use GuzzleHttp\RequestOptions;
|
2023-02-09 22:42:18 +03:00
|
|
|
use PHPUnit\Framework\Attributes\DataProvider;
|
|
|
|
use PHPUnit\Framework\Attributes\Test;
|
2019-08-11 17:30:46 +03:00
|
|
|
use Shlinkio\Shlink\TestUtils\ApiTest\ApiTestCase;
|
2019-02-27 00:56:43 +03:00
|
|
|
|
2023-11-29 14:34:13 +03:00
|
|
|
use function array_map;
|
2019-02-17 22:28:34 +03:00
|
|
|
use function range;
|
2019-11-27 22:48:35 +03:00
|
|
|
use function sprintf;
|
2019-01-30 20:28:07 +03:00
|
|
|
|
2021-01-24 11:25:36 +03:00
|
|
|
class CreateShortUrlTest extends ApiTestCase
|
2019-01-30 20:28:07 +03:00
|
|
|
{
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2019-02-03 13:01:38 +03:00
|
|
|
public function createsNewShortUrlWhenOnlyLongUrlIsProvided(): void
|
2019-01-30 20:28:07 +03:00
|
|
|
{
|
|
|
|
$expectedKeys = ['shortCode', 'shortUrl', 'longUrl', 'dateCreated', 'visitsCount', 'tags'];
|
|
|
|
[$statusCode, $payload] = $this->createShortUrl();
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $statusCode);
|
2019-01-30 20:28:07 +03:00
|
|
|
foreach ($expectedKeys as $key) {
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertArrayHasKey($key, $payload);
|
2019-01-30 20:28:07 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2019-02-03 13:01:38 +03:00
|
|
|
public function createsNewShortUrlWithCustomSlug(): void
|
2019-01-30 20:28:07 +03:00
|
|
|
{
|
|
|
|
[$statusCode, $payload] = $this->createShortUrl(['customSlug' => 'my cool slug']);
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $statusCode);
|
|
|
|
self::assertEquals('my-cool-slug', $payload['shortCode']);
|
2019-01-30 20:28:07 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideConflictingSlugs')]
|
2019-10-02 21:13:25 +03:00
|
|
|
public function failsToCreateShortUrlWithDuplicatedSlug(string $slug, ?string $domain): void
|
|
|
|
{
|
2019-11-27 22:48:35 +03:00
|
|
|
$suffix = $domain === null ? '' : sprintf(' for domain "%s"', $domain);
|
|
|
|
$detail = sprintf('Provided slug "%s" is already in use%s.', $slug, $suffix);
|
|
|
|
|
2019-10-02 21:13:25 +03:00
|
|
|
[$statusCode, $payload] = $this->createShortUrl(['customSlug' => $slug, 'domain' => $domain]);
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_BAD_REQUEST, $statusCode);
|
|
|
|
self::assertEquals(self::STATUS_BAD_REQUEST, $payload['status']);
|
|
|
|
self::assertEquals($detail, $payload['detail']);
|
|
|
|
self::assertEquals('INVALID_SLUG', $payload['type']);
|
|
|
|
self::assertEquals('Invalid custom slug', $payload['title']);
|
|
|
|
self::assertEquals($slug, $payload['customSlug']);
|
2019-11-27 22:48:35 +03:00
|
|
|
|
|
|
|
if ($domain !== null) {
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals($domain, $payload['domain']);
|
2019-11-27 22:48:35 +03:00
|
|
|
} else {
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertArrayNotHasKey('domain', $payload);
|
2019-11-27 22:48:35 +03:00
|
|
|
}
|
2019-10-02 21:13:25 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideDuplicatedSlugApiVersions')]
|
2022-08-13 18:48:55 +03:00
|
|
|
public function expectedTypeIsReturnedForConflictingSlugBasedOnApiVersion(
|
|
|
|
string $version,
|
|
|
|
string $expectedType,
|
|
|
|
): void {
|
|
|
|
[, $payload] = $this->createShortUrl(['customSlug' => 'custom'], version: $version);
|
|
|
|
self::assertEquals($expectedType, $payload['type']);
|
|
|
|
}
|
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideDuplicatedSlugApiVersions(): iterable
|
2022-08-13 18:48:55 +03:00
|
|
|
{
|
|
|
|
yield ['1', 'INVALID_SLUG'];
|
|
|
|
yield ['2', 'INVALID_SLUG'];
|
|
|
|
yield ['3', 'https://shlink.io/api/error/non-unique-slug'];
|
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideTags')]
|
2021-01-31 12:53:18 +03:00
|
|
|
public function createsNewShortUrlWithTags(array $providedTags, array $expectedTags): void
|
2019-01-30 20:28:07 +03:00
|
|
|
{
|
2021-01-31 12:53:18 +03:00
|
|
|
[$statusCode, ['tags' => $tags]] = $this->createShortUrl(['tags' => $providedTags]);
|
2019-01-30 20:28:07 +03:00
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $statusCode);
|
2021-01-31 12:53:18 +03:00
|
|
|
self::assertEquals($expectedTags, $tags);
|
|
|
|
}
|
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideTags(): iterable
|
2021-01-31 12:53:18 +03:00
|
|
|
{
|
|
|
|
yield 'simple tags' => [$simpleTags = ['foo', 'bar', 'baz'], $simpleTags];
|
|
|
|
yield 'tags with spaces' => [['fo o', ' bar', 'b az'], ['fo-o', 'bar', 'b-az']];
|
|
|
|
yield 'tags with special chars' => [['UUU', 'Aäa'], ['uuu', 'aäa']];
|
2019-01-30 20:28:07 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideMaxVisits')]
|
2019-02-03 13:01:38 +03:00
|
|
|
public function createsNewShortUrlWithVisitsLimit(int $maxVisits): void
|
2019-01-30 20:28:07 +03:00
|
|
|
{
|
|
|
|
[$statusCode, ['shortCode' => $shortCode]] = $this->createShortUrl(['maxVisits' => $maxVisits]);
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $statusCode);
|
2019-01-30 20:28:07 +03:00
|
|
|
|
|
|
|
// Last request to the short URL will return a 404, and the rest, a 302
|
|
|
|
for ($i = 0; $i < $maxVisits; $i++) {
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_FOUND, $this->callShortUrl($shortCode)->getStatusCode());
|
2019-01-30 20:28:07 +03:00
|
|
|
}
|
|
|
|
$lastResp = $this->callShortUrl($shortCode);
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_NOT_FOUND, $lastResp->getStatusCode());
|
2019-01-30 20:28:07 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideMaxVisits(): array
|
2019-01-30 20:28:07 +03:00
|
|
|
{
|
2023-11-29 14:34:13 +03:00
|
|
|
return array_map(static fn (int $i) => [$i], range(10, 15));
|
2019-01-30 20:28:07 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2019-02-03 13:01:38 +03:00
|
|
|
public function createsShortUrlWithValidSince(): void
|
2019-01-30 20:28:07 +03:00
|
|
|
{
|
|
|
|
[$statusCode, ['shortCode' => $shortCode]] = $this->createShortUrl([
|
2023-08-19 12:48:04 +03:00
|
|
|
'validSince' => Chronos::now()->addDays(1)->toAtomString(),
|
2019-01-30 20:28:07 +03:00
|
|
|
]);
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $statusCode);
|
2019-01-30 20:28:07 +03:00
|
|
|
|
2019-10-11 10:14:25 +03:00
|
|
|
// Request to the short URL will return a 404 since it's not valid yet
|
2019-01-30 20:28:07 +03:00
|
|
|
$lastResp = $this->callShortUrl($shortCode);
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_NOT_FOUND, $lastResp->getStatusCode());
|
2019-01-30 20:28:07 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2019-02-03 13:01:38 +03:00
|
|
|
public function createsShortUrlWithValidUntil(): void
|
2019-01-30 20:28:07 +03:00
|
|
|
{
|
|
|
|
[$statusCode, ['shortCode' => $shortCode]] = $this->createShortUrl([
|
2023-08-19 12:48:04 +03:00
|
|
|
'validUntil' => Chronos::now()->subDays(1)->toAtomString(),
|
2019-01-30 20:28:07 +03:00
|
|
|
]);
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $statusCode);
|
2019-01-30 20:28:07 +03:00
|
|
|
|
|
|
|
// Request to the short URL will return a 404 since it's no longer valid
|
|
|
|
$lastResp = $this->callShortUrl($shortCode);
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_NOT_FOUND, $lastResp->getStatusCode());
|
2019-01-30 20:28:07 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideMatchingBodies')]
|
2019-02-03 13:01:38 +03:00
|
|
|
public function returnsAnExistingShortUrlWhenRequested(array $body): void
|
|
|
|
{
|
|
|
|
[$firstStatusCode, ['shortCode' => $firstShortCode]] = $this->createShortUrl($body);
|
|
|
|
|
|
|
|
$body['findIfExists'] = true;
|
|
|
|
[$secondStatusCode, ['shortCode' => $secondShortCode]] = $this->createShortUrl($body);
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $firstStatusCode);
|
|
|
|
self::assertEquals(self::STATUS_OK, $secondStatusCode);
|
|
|
|
self::assertEquals($firstShortCode, $secondShortCode);
|
2019-02-03 13:01:38 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideMatchingBodies(): iterable
|
2019-02-03 13:01:38 +03:00
|
|
|
{
|
|
|
|
$longUrl = 'https://www.alejandrocelaya.com';
|
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
yield 'only long URL' => [['longUrl' => $longUrl]];
|
|
|
|
yield 'long URL and tags' => [['longUrl' => $longUrl, 'tags' => ['boo', 'far']]];
|
|
|
|
yield 'long URL and custom slug' => [['longUrl' => $longUrl, 'customSlug' => 'my cool slug']];
|
2023-01-22 13:36:00 +03:00
|
|
|
yield 'several params' => [
|
|
|
|
[
|
|
|
|
'longUrl' => $longUrl,
|
|
|
|
'tags' => ['boo', 'far'],
|
|
|
|
'validSince' => Chronos::now()->toAtomString(),
|
|
|
|
'maxVisits' => 7,
|
2023-01-22 14:15:29 +03:00
|
|
|
],
|
2023-01-22 13:36:00 +03:00
|
|
|
];
|
2019-02-03 13:01:38 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideConflictingSlugs')]
|
2019-10-02 21:13:25 +03:00
|
|
|
public function returnsErrorWhenRequestingReturnExistingButCustomSlugIsInUse(string $slug, ?string $domain): void
|
2019-02-03 13:01:38 +03:00
|
|
|
{
|
|
|
|
$longUrl = 'https://www.alejandrocelaya.com';
|
|
|
|
|
|
|
|
[$firstStatusCode] = $this->createShortUrl(['longUrl' => $longUrl]);
|
|
|
|
[$secondStatusCode] = $this->createShortUrl([
|
|
|
|
'longUrl' => $longUrl,
|
2019-10-02 21:13:25 +03:00
|
|
|
'customSlug' => $slug,
|
2019-02-03 13:01:38 +03:00
|
|
|
'findIfExists' => true,
|
2019-10-02 21:13:25 +03:00
|
|
|
'domain' => $domain,
|
2019-02-03 13:01:38 +03:00
|
|
|
]);
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $firstStatusCode);
|
|
|
|
self::assertEquals(self::STATUS_BAD_REQUEST, $secondStatusCode);
|
2019-02-03 13:01:38 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideConflictingSlugs(): iterable
|
2019-10-02 21:13:25 +03:00
|
|
|
{
|
|
|
|
yield 'without domain' => ['custom', null];
|
|
|
|
yield 'with domain' => ['custom-with-domain', 'some-domain.com'];
|
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2019-02-03 13:01:38 +03:00
|
|
|
public function createsNewShortUrlIfRequestedToFindButThereIsNoMatch(): void
|
|
|
|
{
|
|
|
|
[$firstStatusCode, ['shortCode' => $firstShortCode]] = $this->createShortUrl([
|
|
|
|
'longUrl' => 'https://www.alejandrocelaya.com',
|
|
|
|
]);
|
|
|
|
[$secondStatusCode, ['shortCode' => $secondShortCode]] = $this->createShortUrl([
|
|
|
|
'longUrl' => 'https://www.alejandrocelaya.com/projects',
|
|
|
|
'findIfExists' => true,
|
|
|
|
]);
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $firstStatusCode);
|
|
|
|
self::assertEquals(self::STATUS_OK, $secondStatusCode);
|
|
|
|
self::assertNotEquals($firstShortCode, $secondShortCode);
|
2019-02-03 13:01:38 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideIdn')]
|
2019-11-16 14:38:45 +03:00
|
|
|
public function createsNewShortUrlWithInternationalizedDomainName(string $longUrl): void
|
2019-11-16 12:06:55 +03:00
|
|
|
{
|
2019-11-16 15:37:53 +03:00
|
|
|
[$statusCode, $payload] = $this->createShortUrl(['longUrl' => $longUrl]);
|
2019-11-16 12:06:55 +03:00
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $statusCode);
|
|
|
|
self::assertEquals($payload['longUrl'], $longUrl);
|
2019-11-16 12:06:55 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideIdn(): iterable
|
2019-11-16 14:38:45 +03:00
|
|
|
{
|
2019-11-16 15:37:53 +03:00
|
|
|
yield ['http://tést.shlink.io']; // Redirects to https://shlink.io
|
|
|
|
yield ['http://test.shlink.io']; // Redirects to http://tést.shlink.io
|
|
|
|
yield ['http://téstb.shlink.io']; // Redirects to http://tést.shlink.io
|
2019-11-16 14:38:45 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideInvalidUrls')]
|
2022-08-13 18:48:55 +03:00
|
|
|
public function failsToCreateShortUrlWithInvalidLongUrl(string $url, string $version, string $expectedType): void
|
2019-11-26 23:33:22 +03:00
|
|
|
{
|
2019-11-27 22:48:35 +03:00
|
|
|
$expectedDetail = sprintf('Provided URL %s is invalid. Try with a different one.', $url);
|
|
|
|
|
2022-08-13 18:48:55 +03:00
|
|
|
[$statusCode, $payload] = $this->createShortUrl(['longUrl' => $url, 'validateUrl' => true], version: $version);
|
2019-11-26 23:33:22 +03:00
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_BAD_REQUEST, $statusCode);
|
|
|
|
self::assertEquals(self::STATUS_BAD_REQUEST, $payload['status']);
|
2022-08-13 18:48:55 +03:00
|
|
|
self::assertEquals($expectedType, $payload['type']);
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals($expectedDetail, $payload['detail']);
|
|
|
|
self::assertEquals('Invalid URL', $payload['title']);
|
|
|
|
self::assertEquals($url, $payload['url']);
|
2019-11-26 23:33:22 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideInvalidUrls(): iterable
|
2021-01-30 16:18:44 +03:00
|
|
|
{
|
2023-01-14 17:19:47 +03:00
|
|
|
yield 'API version 2' => ['https://this-has-to-be-invalid.com', '2', 'INVALID_URL'];
|
|
|
|
yield 'API version 3' => ['https://this-has-to-be-invalid.com', '3', 'https://shlink.io/api/error/invalid-url'];
|
2021-01-30 16:18:44 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideInvalidArgumentApiVersions')]
|
2023-01-22 13:36:00 +03:00
|
|
|
public function failsToCreateShortUrlWithoutLongUrl(array $payload, string $version, string $expectedType): void
|
2021-01-30 16:18:44 +03:00
|
|
|
{
|
2022-08-13 18:48:55 +03:00
|
|
|
$resp = $this->callApiWithKey(
|
|
|
|
self::METHOD_POST,
|
|
|
|
sprintf('/rest/v%s/short-urls', $version),
|
2023-01-22 13:36:00 +03:00
|
|
|
[RequestOptions::JSON => $payload],
|
2022-08-13 18:48:55 +03:00
|
|
|
);
|
2021-01-30 16:18:44 +03:00
|
|
|
$payload = $this->getJsonResponsePayload($resp);
|
|
|
|
|
|
|
|
self::assertEquals(self::STATUS_BAD_REQUEST, $resp->getStatusCode());
|
|
|
|
self::assertEquals(self::STATUS_BAD_REQUEST, $payload['status']);
|
2022-08-13 18:48:55 +03:00
|
|
|
self::assertEquals($expectedType, $payload['type']);
|
2021-01-30 16:18:44 +03:00
|
|
|
self::assertEquals('Provided data is not valid', $payload['detail']);
|
|
|
|
self::assertEquals('Invalid data', $payload['title']);
|
|
|
|
}
|
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideInvalidArgumentApiVersions(): iterable
|
2022-08-13 18:48:55 +03:00
|
|
|
{
|
2023-01-22 13:36:00 +03:00
|
|
|
yield 'missing long url v2' => [[], '2', 'INVALID_ARGUMENT'];
|
|
|
|
yield 'missing long url v3' => [[], '3', 'https://shlink.io/api/error/invalid-data'];
|
|
|
|
yield 'empty long url v2' => [['longUrl' => null], '2', 'INVALID_ARGUMENT'];
|
|
|
|
yield 'empty long url v3' => [['longUrl' => ' '], '3', 'https://shlink.io/api/error/invalid-data'];
|
2023-03-25 11:52:47 +03:00
|
|
|
yield 'missing url schema v2' => [['longUrl' => 'foo.com'], '2', 'INVALID_ARGUMENT'];
|
|
|
|
yield 'missing url schema v3' => [['longUrl' => 'foo.com'], '3', 'https://shlink.io/api/error/invalid-data'];
|
2023-01-22 13:36:00 +03:00
|
|
|
yield 'empty device long url v2' => [[
|
|
|
|
'longUrl' => 'foo',
|
|
|
|
'deviceLongUrls' => [
|
|
|
|
'android' => null,
|
|
|
|
],
|
|
|
|
], '2', 'INVALID_ARGUMENT'];
|
|
|
|
yield 'empty device long url v3' => [[
|
|
|
|
'longUrl' => 'foo',
|
|
|
|
'deviceLongUrls' => [
|
|
|
|
'ios' => ' ',
|
|
|
|
],
|
|
|
|
], '3', 'https://shlink.io/api/error/invalid-data'];
|
2022-08-13 18:48:55 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2020-02-08 14:30:47 +03:00
|
|
|
public function defaultDomainIsDroppedIfProvided(): void
|
|
|
|
{
|
|
|
|
[$createStatusCode, ['shortCode' => $shortCode]] = $this->createShortUrl([
|
|
|
|
'longUrl' => 'https://www.alejandrocelaya.com',
|
2023-01-19 11:05:52 +03:00
|
|
|
'domain' => 's.test',
|
2020-02-08 14:30:47 +03:00
|
|
|
]);
|
|
|
|
$getResp = $this->callApiWithKey(self::METHOD_GET, '/short-urls/' . $shortCode);
|
|
|
|
$payload = $this->getJsonResponsePayload($getResp);
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $createStatusCode);
|
|
|
|
self::assertEquals(self::STATUS_OK, $getResp->getStatusCode());
|
|
|
|
self::assertArrayHasKey('domain', $payload);
|
|
|
|
self::assertNull($payload['domain']);
|
2020-02-08 14:30:47 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideDomains')]
|
2021-01-10 11:09:56 +03:00
|
|
|
public function apiKeyDomainIsEnforced(?string $providedDomain): void
|
|
|
|
{
|
|
|
|
[$statusCode, ['domain' => $returnedDomain]] = $this->createShortUrl(
|
|
|
|
['domain' => $providedDomain],
|
|
|
|
'domain_api_key',
|
|
|
|
);
|
|
|
|
|
|
|
|
self::assertEquals(self::STATUS_OK, $statusCode);
|
|
|
|
self::assertEquals('example.com', $returnedDomain);
|
|
|
|
}
|
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideDomains(): iterable
|
2021-01-10 11:09:56 +03:00
|
|
|
{
|
|
|
|
yield 'no domain' => [null];
|
|
|
|
yield 'invalid domain' => ['this-will-be-overwritten.com'];
|
|
|
|
yield 'example domain' => ['example.com'];
|
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2022-01-10 15:04:16 +03:00
|
|
|
public function canCreateShortUrlsWithEmojis(): void
|
|
|
|
{
|
|
|
|
[$statusCode, $payload] = $this->createShortUrl([
|
|
|
|
'longUrl' => 'https://emojipedia.org/fire/',
|
|
|
|
'title' => '🔥🔥🔥',
|
|
|
|
'customSlug' => '🦣🦣🦣',
|
|
|
|
]);
|
|
|
|
self::assertEquals(self::STATUS_OK, $statusCode);
|
|
|
|
self::assertEquals('🔥🔥🔥', $payload['title']);
|
|
|
|
self::assertEquals('🦣🦣🦣', $payload['shortCode']);
|
2023-01-19 11:05:52 +03:00
|
|
|
self::assertEquals('http://s.test/🦣🦣🦣', $payload['shortUrl']);
|
2022-01-10 15:04:16 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2023-01-22 13:36:00 +03:00
|
|
|
public function canCreateShortUrlsWithDeviceLongUrls(): void
|
|
|
|
{
|
|
|
|
[$statusCode, $payload] = $this->createShortUrl([
|
|
|
|
'longUrl' => 'https://github.com/shlinkio/shlink/issues/1557',
|
|
|
|
'deviceLongUrls' => [
|
|
|
|
'ios' => 'https://github.com/shlinkio/shlink/ios',
|
|
|
|
'android' => 'https://github.com/shlinkio/shlink/android',
|
|
|
|
],
|
|
|
|
]);
|
|
|
|
|
|
|
|
self::assertEquals(self::STATUS_OK, $statusCode);
|
|
|
|
self::assertEquals('https://github.com/shlinkio/shlink/ios', $payload['deviceLongUrls']['ios'] ?? null);
|
|
|
|
self::assertEquals('https://github.com/shlinkio/shlink/android', $payload['deviceLongUrls']['android'] ?? null);
|
|
|
|
}
|
|
|
|
|
2019-01-30 20:28:07 +03:00
|
|
|
/**
|
2022-10-24 21:25:06 +03:00
|
|
|
* @return array{int, array}
|
2019-01-30 20:28:07 +03:00
|
|
|
*/
|
2022-08-13 18:48:55 +03:00
|
|
|
private function createShortUrl(array $body = [], string $apiKey = 'valid_api_key', string $version = '2'): array
|
2019-01-30 20:28:07 +03:00
|
|
|
{
|
2019-02-03 13:01:38 +03:00
|
|
|
if (! isset($body['longUrl'])) {
|
|
|
|
$body['longUrl'] = 'https://app.shlink.io';
|
|
|
|
}
|
2022-08-13 18:48:55 +03:00
|
|
|
$resp = $this->callApiWithKey(
|
|
|
|
self::METHOD_POST,
|
|
|
|
sprintf('/rest/v%s/short-urls', $version),
|
|
|
|
[RequestOptions::JSON => $body],
|
|
|
|
$apiKey,
|
|
|
|
);
|
2019-01-30 20:28:07 +03:00
|
|
|
$payload = $this->getJsonResponsePayload($resp);
|
|
|
|
|
|
|
|
return [$resp->getStatusCode(), $payload];
|
|
|
|
}
|
|
|
|
}
|