mirror of
https://github.com/shlinkio/shlink.git
synced 2025-02-18 16:29:56 +03:00
Merge branch 'develop' into feature/phpunit-10
This commit is contained in:
commit
3c12a55872
6 changed files with 82 additions and 23 deletions
|
@ -19,6 +19,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com), and this
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
* [#1698](https://github.com/shlinkio/shlink/issues/1698) Fixed error 500 in `robots.txt`.
|
* [#1698](https://github.com/shlinkio/shlink/issues/1698) Fixed error 500 in `robots.txt`.
|
||||||
|
* [#1688](https://github.com/shlinkio/shlink/issues/1688) Fixed huge performance degradation on `/tags/stats` endpoint.
|
||||||
|
|
||||||
|
|
||||||
## [3.5.1] - 2023-02-04
|
## [3.5.1] - 2023-02-04
|
||||||
|
|
|
@ -5,14 +5,16 @@ declare(strict_types=1);
|
||||||
use Monolog\Level;
|
use Monolog\Level;
|
||||||
use Shlinkio\Shlink\Common\Logger\LoggerType;
|
use Shlinkio\Shlink\Common\Logger\LoggerType;
|
||||||
|
|
||||||
$isSwoole = extension_loaded('openswoole');
|
use function Shlinkio\Shlink\Config\runningInOpenswoole;
|
||||||
|
|
||||||
|
$logToStream = runningInOpenswoole();
|
||||||
|
|
||||||
return [
|
return [
|
||||||
|
|
||||||
'logger' => [
|
'logger' => [
|
||||||
'Shlink' => [
|
'Shlink' => [
|
||||||
// For swoole, send logs as stream
|
// For openswoole, send logs as stream
|
||||||
'type' => $isSwoole ? LoggerType::STREAM->value : LoggerType::FILE->value,
|
'type' => $logToStream ? LoggerType::STREAM->value : LoggerType::FILE->value,
|
||||||
'level' => Level::Debug->value,
|
'level' => Level::Debug->value,
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
set -ex
|
set -ex
|
||||||
|
|
||||||
curl https://packages.microsoft.com/keys/microsoft.asc | apt-key add -
|
curl https://packages.microsoft.com/keys/microsoft.asc | apt-key add -
|
||||||
curl https://packages.microsoft.com/config/ubuntu/20.04/prod.list > /etc/apt/sources.list.d/mssql-release.list
|
curl https://packages.microsoft.com/config/ubuntu/22.04/prod.list > /etc/apt/sources.list.d/mssql-release.list
|
||||||
apt-get update
|
apt-get update
|
||||||
ACCEPT_EULA=Y apt-get install msodbcsql17
|
ACCEPT_EULA=Y apt-get install msodbcsql18
|
||||||
apt-get install unixodbc-dev
|
apt-get install unixodbc-dev
|
||||||
|
|
27
data/migrations/Version20230211171904.php
Normal file
27
data/migrations/Version20230211171904.php
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace ShlinkMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Platforms\MySQLPlatform;
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20230211171904 extends AbstractMigration
|
||||||
|
{
|
||||||
|
private const INDEX_NAME = 'IDX_visits_potential_bot';
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$visits = $schema->getTable('visits');
|
||||||
|
$this->skipIf($visits->hasIndex(self::INDEX_NAME));
|
||||||
|
|
||||||
|
$visits->addIndex(['short_url_id', 'potential_bot'], self::INDEX_NAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isTransactional(): bool
|
||||||
|
{
|
||||||
|
return ! ($this->connection->getDatabasePlatform() instanceof MySQLPlatform);
|
||||||
|
}
|
||||||
|
}
|
|
@ -4,6 +4,7 @@ declare(strict_types=1);
|
||||||
|
|
||||||
namespace Shlinkio\Shlink\Core\Tag\Repository;
|
namespace Shlinkio\Shlink\Core\Tag\Repository;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Query\QueryBuilder as NativeQueryBuilder;
|
||||||
use Doctrine\ORM\Query\ResultSetMappingBuilder;
|
use Doctrine\ORM\Query\ResultSetMappingBuilder;
|
||||||
use Happyr\DoctrineSpecification\Repository\EntitySpecificationRepository;
|
use Happyr\DoctrineSpecification\Repository\EntitySpecificationRepository;
|
||||||
use Happyr\DoctrineSpecification\Spec;
|
use Happyr\DoctrineSpecification\Spec;
|
||||||
|
@ -45,7 +46,6 @@ class TagRepository extends EntitySpecificationRepository implements TagReposito
|
||||||
$orderDir = $filtering?->orderBy?->direction;
|
$orderDir = $filtering?->orderBy?->direction;
|
||||||
$orderMainQuery = $orderField !== null && OrderableField::isAggregateField($orderField);
|
$orderMainQuery = $orderField !== null && OrderableField::isAggregateField($orderField);
|
||||||
|
|
||||||
$conn = $this->getEntityManager()->getConnection();
|
|
||||||
$subQb = $this->createQueryBuilder('t');
|
$subQb = $this->createQueryBuilder('t');
|
||||||
$subQb->select('t.id', 't.name');
|
$subQb->select('t.id', 't.name');
|
||||||
|
|
||||||
|
@ -53,15 +53,51 @@ class TagRepository extends EntitySpecificationRepository implements TagReposito
|
||||||
$subQb->orderBy('t.name', $orderDir ?? 'ASC')
|
$subQb->orderBy('t.name', $orderDir ?? 'ASC')
|
||||||
->setMaxResults($filtering?->limit ?? PHP_INT_MAX)
|
->setMaxResults($filtering?->limit ?? PHP_INT_MAX)
|
||||||
->setFirstResult($filtering?->offset ?? 0);
|
->setFirstResult($filtering?->offset ?? 0);
|
||||||
|
// TODO Check if applying limit/offset ot visits sub-queries is needed with large amounts of tags
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$conn = $this->getEntityManager()->getConnection();
|
||||||
|
$buildVisitsSubQuery = static function (bool $excludeBots, string $aggregateAlias) use ($conn) {
|
||||||
|
$visitsSubQuery = $conn->createQueryBuilder();
|
||||||
|
$commonJoinCondition = $visitsSubQuery->expr()->eq('v.short_url_id', 's.id');
|
||||||
|
$visitsJoin = ! $excludeBots
|
||||||
|
? $commonJoinCondition
|
||||||
|
: $visitsSubQuery->expr()->and(
|
||||||
|
$commonJoinCondition,
|
||||||
|
$visitsSubQuery->expr()->eq('v.potential_bot', $conn->quote('0')),
|
||||||
|
);
|
||||||
|
|
||||||
|
return $visitsSubQuery
|
||||||
|
->select('st.tag_id AS tag_id', 'COUNT(DISTINCT v.id) AS ' . $aggregateAlias)
|
||||||
|
->from('visits', 'v')
|
||||||
|
->join('v', 'short_urls', 's', $visitsJoin) // @phpstan-ignore-line
|
||||||
|
->join('s', 'short_urls_in_tags', 'st', $visitsSubQuery->expr()->eq('st.short_url_id', 's.id'))
|
||||||
|
->groupBy('st.tag_id');
|
||||||
|
};
|
||||||
|
$allVisitsSubQuery = $buildVisitsSubQuery(false, 'visits');
|
||||||
|
$nonBotVisitsSubQuery = $buildVisitsSubQuery(true, 'non_bot_visits');
|
||||||
|
|
||||||
$searchTerm = $filtering?->searchTerm;
|
$searchTerm = $filtering?->searchTerm;
|
||||||
if ($searchTerm !== null) {
|
if ($searchTerm !== null) {
|
||||||
$subQb->andWhere($subQb->expr()->like('t.name', $conn->quote('%' . $searchTerm . '%')));
|
$subQb->andWhere($subQb->expr()->like('t.name', $conn->quote('%' . $searchTerm . '%')));
|
||||||
|
// TODO Check if applying this to all sub-queries makes it faster or slower
|
||||||
}
|
}
|
||||||
|
|
||||||
$apiKey = $filtering?->apiKey;
|
$apiKey = $filtering?->apiKey;
|
||||||
|
$applyApiKeyToNativeQuery = static fn (?ApiKey $apiKey, NativeQueryBuilder $nativeQueryBuilder) =>
|
||||||
|
$apiKey?->mapRoles(static fn (Role $role, array $meta) => match ($role) {
|
||||||
|
Role::DOMAIN_SPECIFIC => $nativeQueryBuilder->andWhere(
|
||||||
|
$nativeQueryBuilder->expr()->eq('s.domain_id', $conn->quote(Role::domainIdFromMeta($meta))),
|
||||||
|
),
|
||||||
|
Role::AUTHORED_SHORT_URLS => $nativeQueryBuilder->andWhere(
|
||||||
|
$nativeQueryBuilder->expr()->eq('s.author_api_key_id', $conn->quote($apiKey->getId())),
|
||||||
|
),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Apply API key specification to all sub-queries
|
||||||
$this->applySpecification($subQb, new WithInlinedApiKeySpecsEnsuringJoin($apiKey), 't');
|
$this->applySpecification($subQb, new WithInlinedApiKeySpecsEnsuringJoin($apiKey), 't');
|
||||||
|
$applyApiKeyToNativeQuery($apiKey, $allVisitsSubQuery);
|
||||||
|
$applyApiKeyToNativeQuery($apiKey, $nonBotVisitsSubQuery);
|
||||||
|
|
||||||
// A native query builder needs to be used here, because DQL and ORM query builders do not support
|
// A native query builder needs to be used here, because DQL and ORM query builders do not support
|
||||||
// sub-queries at "from" and "join" level.
|
// sub-queries at "from" and "join" level.
|
||||||
|
@ -71,29 +107,22 @@ class TagRepository extends EntitySpecificationRepository implements TagReposito
|
||||||
->select(
|
->select(
|
||||||
't.id_0 AS id',
|
't.id_0 AS id',
|
||||||
't.name_1 AS name',
|
't.name_1 AS name',
|
||||||
|
'COALESCE(v.visits, 0) AS visits', // COALESCE required for postgres to properly order
|
||||||
|
'COALESCE(v2.non_bot_visits, 0) AS non_bot_visits', // COALESCE required for postgres to properly order
|
||||||
'COUNT(DISTINCT s.id) AS short_urls_count',
|
'COUNT(DISTINCT s.id) AS short_urls_count',
|
||||||
'COUNT(DISTINCT v.id) AS visits', // Native queries require snake_case for cross-db compatibility
|
|
||||||
'COUNT(DISTINCT v2.id) AS non_bot_visits',
|
|
||||||
)
|
)
|
||||||
->from('(' . $subQb->getQuery()->getSQL() . ')', 't') // @phpstan-ignore-line
|
->from('(' . $subQb->getQuery()->getSQL() . ')', 't') // @phpstan-ignore-line
|
||||||
->leftJoin('t', 'short_urls_in_tags', 'st', $nativeQb->expr()->eq('t.id_0', 'st.tag_id'))
|
->leftJoin('t', 'short_urls_in_tags', 'st', $nativeQb->expr()->eq('t.id_0', 'st.tag_id'))
|
||||||
->leftJoin('st', 'short_urls', 's', $nativeQb->expr()->eq('s.id', 'st.short_url_id'))
|
->leftJoin('st', 'short_urls', 's', $nativeQb->expr()->eq('s.id', 'st.short_url_id'))
|
||||||
->leftJoin('st', 'visits', 'v', $nativeQb->expr()->eq('st.short_url_id', 'v.short_url_id'))
|
->leftJoin('t', '(' . $allVisitsSubQuery->getSQL() . ')', 'v', $nativeQb->expr()->eq('t.id_0', 'v.tag_id'))
|
||||||
->leftJoin('st', 'visits', 'v2', $nativeQb->expr()->and( // @phpstan-ignore-line
|
->leftJoin('t', '(' . $nonBotVisitsSubQuery->getSQL() . ')', 'v2', $nativeQb->expr()->eq(
|
||||||
$nativeQb->expr()->eq('st.short_url_id', 'v2.short_url_id'),
|
't.id_0',
|
||||||
$nativeQb->expr()->eq('v2.potential_bot', $conn->quote('0')),
|
'v2.tag_id',
|
||||||
))
|
))
|
||||||
->groupBy('t.id_0', 't.name_1');
|
->groupBy('t.id_0', 't.name_1', 'v.visits', 'v2.non_bot_visits');
|
||||||
|
|
||||||
// Apply API key role conditions to the native query too, as they will affect the amounts on the aggregates
|
// Apply API key role conditions to the native query too, as they will affect the amounts on the aggregates
|
||||||
$apiKey?->mapRoles(static fn (Role $role, array $meta) => match ($role) {
|
$applyApiKeyToNativeQuery($apiKey, $nativeQb);
|
||||||
Role::DOMAIN_SPECIFIC => $nativeQb->andWhere(
|
|
||||||
$nativeQb->expr()->eq('s.domain_id', $conn->quote(Role::domainIdFromMeta($meta))),
|
|
||||||
),
|
|
||||||
Role::AUTHORED_SHORT_URLS => $nativeQb->andWhere(
|
|
||||||
$nativeQb->expr()->eq('s.author_api_key_id', $conn->quote($apiKey->getId())),
|
|
||||||
),
|
|
||||||
});
|
|
||||||
|
|
||||||
if ($orderMainQuery) {
|
if ($orderMainQuery) {
|
||||||
$nativeQb
|
$nativeQb
|
||||||
|
@ -107,9 +136,9 @@ class TagRepository extends EntitySpecificationRepository implements TagReposito
|
||||||
|
|
||||||
$rsm = new ResultSetMappingBuilder($this->getEntityManager());
|
$rsm = new ResultSetMappingBuilder($this->getEntityManager());
|
||||||
$rsm->addScalarResult('name', 'tag');
|
$rsm->addScalarResult('name', 'tag');
|
||||||
$rsm->addScalarResult('short_urls_count', 'shortUrlsCount');
|
|
||||||
$rsm->addScalarResult('visits', 'visits');
|
$rsm->addScalarResult('visits', 'visits');
|
||||||
$rsm->addScalarResult('non_bot_visits', 'nonBotVisits');
|
$rsm->addScalarResult('non_bot_visits', 'nonBotVisits');
|
||||||
|
$rsm->addScalarResult('short_urls_count', 'shortUrlsCount');
|
||||||
|
|
||||||
return map(
|
return map(
|
||||||
$this->getEntityManager()->createNativeQuery($nativeQb->getSQL(), $rsm)->getResult(),
|
$this->getEntityManager()->createNativeQuery($nativeQb->getSQL(), $rsm)->getResult(),
|
||||||
|
|
|
@ -11,7 +11,7 @@ use Shlinkio\Shlink\Rest\Entity\ApiKey;
|
||||||
|
|
||||||
class WithInlinedApiKeySpecsEnsuringJoin extends BaseSpecification
|
class WithInlinedApiKeySpecsEnsuringJoin extends BaseSpecification
|
||||||
{
|
{
|
||||||
public function __construct(private ?ApiKey $apiKey, private string $fieldToJoin = 'shortUrls')
|
public function __construct(private readonly ?ApiKey $apiKey, private readonly string $fieldToJoin = 'shortUrls')
|
||||||
{
|
{
|
||||||
parent::__construct();
|
parent::__construct();
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue