2019-01-27 14:35:00 +03:00
|
|
|
<?php
|
2019-10-05 18:26:10 +03:00
|
|
|
|
2019-01-27 14:35:00 +03:00
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace ShlinkioApiTest\Shlink\Rest\Action;
|
|
|
|
|
2019-12-17 01:03:32 +03:00
|
|
|
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-01-27 14:35:00 +03:00
|
|
|
|
2019-12-17 01:03:32 +03:00
|
|
|
use function count;
|
|
|
|
|
2019-01-27 14:35:00 +03:00
|
|
|
class ListShortUrlsTest extends ApiTestCase
|
|
|
|
{
|
2021-02-04 23:27:16 +03:00
|
|
|
private const SHORT_URL_SHLINK_WITH_TITLE = [
|
2019-12-17 01:03:32 +03:00
|
|
|
'shortCode' => 'abc123',
|
2023-01-19 11:05:52 +03:00
|
|
|
'shortUrl' => 'http://s.test/abc123',
|
2019-12-17 01:03:32 +03:00
|
|
|
'longUrl' => 'https://shlink.io',
|
|
|
|
'dateCreated' => '2018-05-01T00:00:00+00:00',
|
2022-12-16 12:53:44 +03:00
|
|
|
'visitsSummary' => [
|
|
|
|
'total' => 3,
|
|
|
|
'nonBots' => 3,
|
|
|
|
'bots' => 0,
|
|
|
|
],
|
2019-12-17 01:03:32 +03:00
|
|
|
'tags' => ['foo'],
|
|
|
|
'meta' => [
|
|
|
|
'validSince' => null,
|
|
|
|
'validUntil' => null,
|
2022-12-11 15:38:11 +03:00
|
|
|
'maxVisits' => 2,
|
2019-12-17 01:03:32 +03:00
|
|
|
],
|
2020-02-02 10:57:04 +03:00
|
|
|
'domain' => null,
|
2021-02-04 23:27:16 +03:00
|
|
|
'title' => 'My cool title',
|
2021-05-22 08:35:47 +03:00
|
|
|
'crawlable' => true,
|
2021-10-02 11:31:23 +03:00
|
|
|
'forwardQuery' => true,
|
2019-12-17 01:03:32 +03:00
|
|
|
];
|
2020-02-02 11:51:17 +03:00
|
|
|
private const SHORT_URL_DOCS = [
|
|
|
|
'shortCode' => 'ghi789',
|
2023-01-19 11:05:52 +03:00
|
|
|
'shortUrl' => 'http://s.test/ghi789',
|
2020-02-02 11:51:17 +03:00
|
|
|
'longUrl' => 'https://shlink.io/documentation/',
|
|
|
|
'dateCreated' => '2018-05-01T00:00:00+00:00',
|
2022-12-16 12:53:44 +03:00
|
|
|
'visitsSummary' => [
|
|
|
|
'total' => 2,
|
|
|
|
'nonBots' => 2,
|
|
|
|
'bots' => 0,
|
|
|
|
],
|
2020-02-02 11:51:17 +03:00
|
|
|
'tags' => [],
|
|
|
|
'meta' => [
|
|
|
|
'validSince' => null,
|
2022-12-11 15:38:11 +03:00
|
|
|
'validUntil' => '2020-05-01T00:00:00+00:00',
|
2020-02-02 11:51:17 +03:00
|
|
|
'maxVisits' => null,
|
|
|
|
],
|
|
|
|
'domain' => null,
|
2021-02-02 22:51:28 +03:00
|
|
|
'title' => null,
|
2021-05-22 08:35:47 +03:00
|
|
|
'crawlable' => false,
|
2021-10-02 11:31:23 +03:00
|
|
|
'forwardQuery' => true,
|
2020-02-02 11:51:17 +03:00
|
|
|
];
|
2019-12-17 01:03:32 +03:00
|
|
|
private const SHORT_URL_CUSTOM_SLUG_AND_DOMAIN = [
|
|
|
|
'shortCode' => 'custom-with-domain',
|
|
|
|
'shortUrl' => 'http://some-domain.com/custom-with-domain',
|
|
|
|
'longUrl' => 'https://google.com',
|
|
|
|
'dateCreated' => '2018-10-20T00:00:00+00:00',
|
2022-12-16 12:53:44 +03:00
|
|
|
'visitsSummary' => [
|
|
|
|
'total' => 0,
|
|
|
|
'nonBots' => 0,
|
|
|
|
'bots' => 0,
|
|
|
|
],
|
2019-12-17 01:03:32 +03:00
|
|
|
'tags' => [],
|
|
|
|
'meta' => [
|
|
|
|
'validSince' => null,
|
|
|
|
'validUntil' => null,
|
|
|
|
'maxVisits' => null,
|
|
|
|
],
|
2020-02-02 10:57:04 +03:00
|
|
|
'domain' => 'some-domain.com',
|
2021-02-02 22:51:28 +03:00
|
|
|
'title' => null,
|
2021-05-22 08:35:47 +03:00
|
|
|
'crawlable' => false,
|
2021-10-02 11:31:23 +03:00
|
|
|
'forwardQuery' => true,
|
2019-12-17 01:03:32 +03:00
|
|
|
];
|
|
|
|
private const SHORT_URL_META = [
|
|
|
|
'shortCode' => 'def456',
|
2023-01-19 11:05:52 +03:00
|
|
|
'shortUrl' => 'http://s.test/def456',
|
2019-12-17 01:03:32 +03:00
|
|
|
'longUrl' =>
|
|
|
|
'https://blog.alejandrocelaya.com/2017/12/09'
|
|
|
|
. '/acmailer-7-0-the-most-important-release-in-a-long-time/',
|
2019-12-18 17:26:23 +03:00
|
|
|
'dateCreated' => '2019-01-01T00:00:10+00:00',
|
2022-12-16 12:53:44 +03:00
|
|
|
'visitsSummary' => [
|
|
|
|
'total' => 2,
|
|
|
|
'nonBots' => 1,
|
|
|
|
'bots' => 1,
|
|
|
|
],
|
2019-12-17 01:03:32 +03:00
|
|
|
'tags' => ['bar', 'foo'],
|
|
|
|
'meta' => [
|
|
|
|
'validSince' => '2020-05-01T00:00:00+00:00',
|
|
|
|
'validUntil' => null,
|
|
|
|
'maxVisits' => null,
|
|
|
|
],
|
2020-02-02 10:57:04 +03:00
|
|
|
'domain' => null,
|
2021-02-02 22:51:28 +03:00
|
|
|
'title' => null,
|
2021-05-22 08:35:47 +03:00
|
|
|
'crawlable' => false,
|
2021-10-02 11:31:23 +03:00
|
|
|
'forwardQuery' => true,
|
2019-12-17 01:03:32 +03:00
|
|
|
];
|
|
|
|
private const SHORT_URL_CUSTOM_SLUG = [
|
|
|
|
'shortCode' => 'custom',
|
2023-01-19 11:05:52 +03:00
|
|
|
'shortUrl' => 'http://s.test/custom',
|
2019-12-17 01:03:32 +03:00
|
|
|
'longUrl' => 'https://shlink.io',
|
2019-12-18 17:26:23 +03:00
|
|
|
'dateCreated' => '2019-01-01T00:00:20+00:00',
|
2022-12-16 12:53:44 +03:00
|
|
|
'visitsSummary' => [
|
|
|
|
'total' => 0,
|
|
|
|
'nonBots' => 0,
|
|
|
|
'bots' => 0,
|
|
|
|
],
|
2019-12-17 01:03:32 +03:00
|
|
|
'tags' => [],
|
|
|
|
'meta' => [
|
|
|
|
'validSince' => null,
|
|
|
|
'validUntil' => null,
|
|
|
|
'maxVisits' => 2,
|
|
|
|
],
|
2020-02-02 10:57:04 +03:00
|
|
|
'domain' => null,
|
2021-02-02 22:51:28 +03:00
|
|
|
'title' => null,
|
2023-02-10 22:26:18 +03:00
|
|
|
'crawlable' => true,
|
2021-10-02 11:31:23 +03:00
|
|
|
'forwardQuery' => false,
|
2019-12-17 01:03:32 +03:00
|
|
|
];
|
|
|
|
private const SHORT_URL_CUSTOM_DOMAIN = [
|
|
|
|
'shortCode' => 'ghi789',
|
|
|
|
'shortUrl' => 'http://example.com/ghi789',
|
|
|
|
'longUrl' =>
|
|
|
|
'https://blog.alejandrocelaya.com/2019/04/27'
|
|
|
|
. '/considerations-to-properly-use-open-source-software-projects/',
|
2019-12-18 17:26:23 +03:00
|
|
|
'dateCreated' => '2019-01-01T00:00:30+00:00',
|
2022-12-16 12:53:44 +03:00
|
|
|
'visitsSummary' => [
|
|
|
|
'total' => 0,
|
|
|
|
'nonBots' => 0,
|
|
|
|
'bots' => 0,
|
|
|
|
],
|
2021-01-10 11:54:19 +03:00
|
|
|
'tags' => ['foo'],
|
2019-12-17 01:03:32 +03:00
|
|
|
'meta' => [
|
|
|
|
'validSince' => null,
|
|
|
|
'validUntil' => null,
|
|
|
|
'maxVisits' => null,
|
|
|
|
],
|
2020-02-02 10:57:04 +03:00
|
|
|
'domain' => 'example.com',
|
2021-02-02 22:51:28 +03:00
|
|
|
'title' => null,
|
2021-05-22 08:35:47 +03:00
|
|
|
'crawlable' => false,
|
2021-10-02 11:31:23 +03:00
|
|
|
'forwardQuery' => true,
|
2019-12-17 01:03:32 +03:00
|
|
|
];
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideFilteredLists')]
|
2021-01-10 10:49:31 +03:00
|
|
|
public function shortUrlsAreProperlyListed(array $query, array $expectedShortUrls, string $apiKey): void
|
2019-01-27 14:35:00 +03:00
|
|
|
{
|
2021-01-10 10:49:31 +03:00
|
|
|
$resp = $this->callApiWithKey(self::METHOD_GET, '/short-urls', [RequestOptions::QUERY => $query], $apiKey);
|
2019-01-27 14:35:00 +03:00
|
|
|
$respPayload = $this->getJsonResponsePayload($resp);
|
|
|
|
|
2020-10-04 01:35:14 +03:00
|
|
|
self::assertEquals(self::STATUS_OK, $resp->getStatusCode());
|
|
|
|
self::assertEquals([
|
2019-01-27 14:35:00 +03:00
|
|
|
'shortUrls' => [
|
2019-12-17 01:03:32 +03:00
|
|
|
'data' => $expectedShortUrls,
|
|
|
|
'pagination' => $this->buildPagination(count($expectedShortUrls)),
|
2019-01-27 14:35:00 +03:00
|
|
|
],
|
|
|
|
], $respPayload);
|
|
|
|
}
|
2019-12-17 01:03:32 +03:00
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideFilteredLists(): iterable
|
2019-12-17 01:03:32 +03:00
|
|
|
{
|
|
|
|
yield [[], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG,
|
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG_AND_DOMAIN,
|
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
|
|
|
self::SHORT_URL_DOCS,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'valid_api_key'];
|
2022-12-11 15:38:11 +03:00
|
|
|
yield [['excludePastValidUntil' => 'true'], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG,
|
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG_AND_DOMAIN,
|
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
2022-12-11 15:38:11 +03:00
|
|
|
], 'valid_api_key'];
|
|
|
|
yield [['excludeMaxVisitsReached' => 'true'], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG,
|
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG_AND_DOMAIN,
|
|
|
|
self::SHORT_URL_DOCS,
|
2022-12-11 15:38:11 +03:00
|
|
|
], 'valid_api_key'];
|
2019-12-17 01:03:32 +03:00
|
|
|
yield [['orderBy' => 'shortCode'], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG_AND_DOMAIN,
|
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_DOCS,
|
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'valid_api_key'];
|
2020-09-21 23:03:43 +03:00
|
|
|
yield [['orderBy' => 'shortCode-DESC'], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_DOCS,
|
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG_AND_DOMAIN,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG,
|
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'valid_api_key'];
|
2021-02-05 00:07:54 +03:00
|
|
|
yield [['orderBy' => 'title-DESC'], [
|
2024-03-29 11:35:54 +03:00
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG,
|
|
|
|
self::SHORT_URL_DOCS,
|
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG_AND_DOMAIN,
|
2021-02-05 00:07:54 +03:00
|
|
|
], 'valid_api_key'];
|
2019-12-17 01:03:32 +03:00
|
|
|
yield [['startDate' => Chronos::parse('2018-12-01')->toAtomString()], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_CUSTOM_SLUG,
|
|
|
|
self::SHORT_URL_META,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'valid_api_key'];
|
2019-12-17 01:03:32 +03:00
|
|
|
yield [['endDate' => Chronos::parse('2018-12-01')->toAtomString()], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_SLUG_AND_DOMAIN,
|
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
|
|
|
self::SHORT_URL_DOCS,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'valid_api_key'];
|
2019-12-17 01:03:32 +03:00
|
|
|
yield [['tags' => ['foo']], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'valid_api_key'];
|
2019-12-17 01:03:32 +03:00
|
|
|
yield [['tags' => ['bar']], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_META,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'valid_api_key'];
|
2022-01-04 16:34:31 +03:00
|
|
|
yield [['tags' => ['foo', 'bar']], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
2022-01-04 16:34:31 +03:00
|
|
|
], 'valid_api_key'];
|
|
|
|
yield [['tags' => ['foo', 'bar'], 'tagsMode' => 'any'], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
2022-01-04 16:34:31 +03:00
|
|
|
], 'valid_api_key'];
|
|
|
|
yield [['tags' => ['foo', 'bar'], 'tagsMode' => 'all'], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_META,
|
2022-01-04 16:34:31 +03:00
|
|
|
], 'valid_api_key'];
|
|
|
|
yield [['tags' => ['foo', 'bar', 'baz']], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
2022-01-04 16:34:31 +03:00
|
|
|
], 'valid_api_key'];
|
|
|
|
yield [['tags' => ['foo', 'bar', 'baz'], 'tagsMode' => 'all'], [], 'valid_api_key'];
|
2019-12-17 01:03:32 +03:00
|
|
|
yield [['tags' => ['foo'], 'endDate' => Chronos::parse('2018-12-01')->toAtomString()], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'valid_api_key'];
|
2019-12-17 01:03:32 +03:00
|
|
|
yield [['searchTerm' => 'alejandro'], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
|
|
|
self::SHORT_URL_META,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'valid_api_key'];
|
2021-02-04 23:27:16 +03:00
|
|
|
yield [['searchTerm' => 'cool'], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
2021-02-04 23:27:16 +03:00
|
|
|
], 'valid_api_key'];
|
2020-02-02 22:07:19 +03:00
|
|
|
yield [['searchTerm' => 'example.com'], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'valid_api_key'];
|
|
|
|
yield [[], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_SLUG,
|
|
|
|
self::SHORT_URL_META,
|
|
|
|
self::SHORT_URL_SHLINK_WITH_TITLE,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'author_api_key'];
|
|
|
|
yield [[], [
|
2024-02-27 20:46:49 +03:00
|
|
|
self::SHORT_URL_CUSTOM_DOMAIN,
|
2021-01-10 10:49:31 +03:00
|
|
|
], 'domain_api_key'];
|
2019-12-17 01:03:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
private function buildPagination(int $itemsCount): array
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
'currentPage' => 1,
|
|
|
|
'pagesCount' => 1,
|
|
|
|
'itemsPerPage' => 10,
|
|
|
|
'itemsInCurrentPage' => $itemsCount,
|
|
|
|
'totalItems' => $itemsCount,
|
|
|
|
];
|
|
|
|
}
|
2022-01-04 16:42:31 +03:00
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideInvalidFiltering')]
|
2022-01-09 13:28:32 +03:00
|
|
|
public function errorIsReturnedWhenProvidingInvalidValues(array $query, array $expectedInvalidElements): void
|
2022-01-04 16:42:31 +03:00
|
|
|
{
|
|
|
|
$resp = $this->callApiWithKey(self::METHOD_GET, '/short-urls', [RequestOptions::QUERY => $query]);
|
|
|
|
$respPayload = $this->getJsonResponsePayload($resp);
|
|
|
|
|
|
|
|
self::assertEquals(400, $resp->getStatusCode());
|
|
|
|
self::assertEquals([
|
2022-01-09 13:28:32 +03:00
|
|
|
'invalidElements' => $expectedInvalidElements,
|
2022-01-04 16:42:31 +03:00
|
|
|
'title' => 'Invalid data',
|
2024-02-12 22:29:40 +03:00
|
|
|
'type' => 'https://shlink.io/api/error/invalid-data',
|
2022-01-04 16:42:31 +03:00
|
|
|
'status' => 400,
|
|
|
|
'detail' => 'Provided data is not valid',
|
|
|
|
], $respPayload);
|
|
|
|
}
|
2022-01-09 13:28:32 +03:00
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideInvalidFiltering(): iterable
|
2022-01-09 13:28:32 +03:00
|
|
|
{
|
|
|
|
yield [['tagsMode' => 'invalid'], ['tagsMode']];
|
|
|
|
yield [['orderBy' => 'invalid'], ['orderBy']];
|
|
|
|
yield [['orderBy' => 'invalid', 'tagsMode' => 'invalid'], ['tagsMode', 'orderBy']];
|
|
|
|
}
|
2019-01-27 14:35:00 +03:00
|
|
|
}
|