mirror of
https://github.com/shlinkio/shlink.git
synced 2025-01-15 11:12:15 -06:00
Merge pull request #748 from acelaya-forks/feature/visits-perf-improvements
Feature/visits perf improvements
This commit is contained in:
commit
79b8834c61
@ -25,7 +25,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com), and this
|
|||||||
|
|
||||||
#### Changed
|
#### Changed
|
||||||
|
|
||||||
* *Nothing*
|
* [#692](https://github.com/shlinkio/shlink/issues/692) Drastically improved performance when loading visits. Specially noticeable when loading big result sets.
|
||||||
|
|
||||||
#### Deprecated
|
#### Deprecated
|
||||||
|
|
||||||
|
27
data/migrations/Version20200503170404.php
Normal file
27
data/migrations/Version20200503170404.php
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace ShlinkMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20200503170404 extends AbstractMigration
|
||||||
|
{
|
||||||
|
private const INDEX_NAME = 'IDX_visits_date';
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$visits = $schema->getTable('visits');
|
||||||
|
$this->skipIf($visits->hasIndex(self::INDEX_NAME));
|
||||||
|
$visits->addIndex(['date'], self::INDEX_NAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$visits = $schema->getTable('visits');
|
||||||
|
$this->skipIf(! $visits->hasIndex(self::INDEX_NAME));
|
||||||
|
$visits->dropIndex(self::INDEX_NAME);
|
||||||
|
}
|
||||||
|
}
|
@ -32,6 +32,8 @@ return static function (ClassMetadata $metadata, array $emConfig): void {
|
|||||||
->columnName('`date`')
|
->columnName('`date`')
|
||||||
->build();
|
->build();
|
||||||
|
|
||||||
|
$builder->addIndex(['date'], 'IDX_visits_date');
|
||||||
|
|
||||||
$builder->createField('remoteAddr', Types::STRING)
|
$builder->createField('remoteAddr', Types::STRING)
|
||||||
->columnName('remote_addr')
|
->columnName('remote_addr')
|
||||||
->length(Visitor::REMOTE_ADDRESS_MAX_LENGTH)
|
->length(Visitor::REMOTE_ADDRESS_MAX_LENGTH)
|
||||||
|
@ -15,6 +15,8 @@ class VisitsPaginatorAdapter implements AdapterInterface
|
|||||||
private ShortUrlIdentifier $identifier;
|
private ShortUrlIdentifier $identifier;
|
||||||
private VisitsParams $params;
|
private VisitsParams $params;
|
||||||
|
|
||||||
|
private ?int $count = null;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
VisitRepositoryInterface $visitRepository,
|
VisitRepositoryInterface $visitRepository,
|
||||||
ShortUrlIdentifier $identifier,
|
ShortUrlIdentifier $identifier,
|
||||||
@ -38,7 +40,17 @@ class VisitsPaginatorAdapter implements AdapterInterface
|
|||||||
|
|
||||||
public function count(): int
|
public function count(): int
|
||||||
{
|
{
|
||||||
return $this->visitRepository->countVisitsByShortCode(
|
// Since a new adapter instance is created every time visits are fetched, it is reasonably safe to internally
|
||||||
|
// cache the count value.
|
||||||
|
// The reason it is cached is because the Paginator is actually calling the method twice.
|
||||||
|
// An inconsistent value could be returned if between the first call and the second one, a new visit is created.
|
||||||
|
// However, it's almost instant, and then the adapter instance is discarded immediately after.
|
||||||
|
|
||||||
|
if ($this->count !== null) {
|
||||||
|
return $this->count;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->count = $this->visitRepository->countVisitsByShortCode(
|
||||||
$this->identifier->shortCode(),
|
$this->identifier->shortCode(),
|
||||||
$this->identifier->domain(),
|
$this->identifier->domain(),
|
||||||
$this->params->getDateRange(),
|
$this->params->getDateRange(),
|
||||||
|
@ -5,9 +5,17 @@ declare(strict_types=1);
|
|||||||
namespace Shlinkio\Shlink\Core\Repository;
|
namespace Shlinkio\Shlink\Core\Repository;
|
||||||
|
|
||||||
use Doctrine\ORM\EntityRepository;
|
use Doctrine\ORM\EntityRepository;
|
||||||
|
use Doctrine\ORM\Query\ResultSetMappingBuilder;
|
||||||
use Doctrine\ORM\QueryBuilder;
|
use Doctrine\ORM\QueryBuilder;
|
||||||
|
use Shlinkio\Shlink\Common\Doctrine\Type\ChronosDateTimeType;
|
||||||
use Shlinkio\Shlink\Common\Util\DateRange;
|
use Shlinkio\Shlink\Common\Util\DateRange;
|
||||||
|
use Shlinkio\Shlink\Core\Entity\ShortUrl;
|
||||||
use Shlinkio\Shlink\Core\Entity\Visit;
|
use Shlinkio\Shlink\Core\Entity\Visit;
|
||||||
|
use Shlinkio\Shlink\Core\Entity\VisitLocation;
|
||||||
|
|
||||||
|
use function preg_replace;
|
||||||
|
|
||||||
|
use const PHP_INT_MAX;
|
||||||
|
|
||||||
class VisitRepository extends EntityRepository implements VisitRepositoryInterface
|
class VisitRepository extends EntityRepository implements VisitRepositoryInterface
|
||||||
{
|
{
|
||||||
@ -81,24 +89,60 @@ class VisitRepository extends EntityRepository implements VisitRepositoryInterfa
|
|||||||
?int $limit = null,
|
?int $limit = null,
|
||||||
?int $offset = null
|
?int $offset = null
|
||||||
): array {
|
): array {
|
||||||
$qb = $this->createVisitsByShortCodeQueryBuilder($shortCode, $domain, $dateRange);
|
/**
|
||||||
$qb->select('v')
|
* @var QueryBuilder $qb
|
||||||
->orderBy('v.date', 'DESC');
|
* @var ShortUrl|int $shortUrl
|
||||||
|
*/
|
||||||
|
[$qb, $shortUrl] = $this->createVisitsByShortCodeQueryBuilder($shortCode, $domain, $dateRange);
|
||||||
|
$qb->select('v.id')
|
||||||
|
->orderBy('v.id', 'DESC')
|
||||||
|
// Falling back to values that will behave as no limit/offset, but will workaround MS SQL not allowing
|
||||||
|
// order on sub-queries without offset
|
||||||
|
->setMaxResults($limit ?? PHP_INT_MAX)
|
||||||
|
->setFirstResult($offset ?? 0);
|
||||||
|
|
||||||
if ($limit !== null) {
|
// FIXME Crappy way to resolve the params into the query. Best option would be to inject the sub-query with
|
||||||
$qb->setMaxResults($limit);
|
// placeholders and then pass params to the main query
|
||||||
|
$shortUrlId = $shortUrl instanceof ShortUrl ? $shortUrl->getId() : $shortUrl;
|
||||||
|
$subQuery = preg_replace('/\?/', $shortUrlId, $qb->getQuery()->getSQL(), 1);
|
||||||
|
if ($dateRange !== null && $dateRange->getStartDate() !== null) {
|
||||||
|
$subQuery = preg_replace(
|
||||||
|
'/\?/',
|
||||||
|
'\'' . $dateRange->getStartDate()->toDateTimeString() . '\'',
|
||||||
|
$subQuery,
|
||||||
|
1,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
if ($offset !== null) {
|
if ($dateRange !== null && $dateRange->getEndDate() !== null) {
|
||||||
$qb->setFirstResult($offset);
|
$subQuery = preg_replace('/\?/', '\'' . $dateRange->getEndDate()->toDateTimeString() . '\'', $subQuery, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $qb->getQuery()->getResult();
|
// A native query builder needs to be used here because DQL and ORM query builders do not accept
|
||||||
|
// sub-queries at "from" and "join" level.
|
||||||
|
// If no sub-query is used, then performance drops dramatically while the "offset" grows.
|
||||||
|
$nativeQb = $this->getEntityManager()->getConnection()->createQueryBuilder();
|
||||||
|
$nativeQb->select('v.id AS visit_id', 'v.*', 'vl.*')
|
||||||
|
->from('visits', 'v')
|
||||||
|
->join('v', '(' . $subQuery . ')', 'sq', $nativeQb->expr()->eq('sq.id_0', 'v.id'))
|
||||||
|
->leftJoin('v', 'visit_locations', 'vl', $nativeQb->expr()->eq('v.visit_location_id', 'vl.id'))
|
||||||
|
->orderBy('v.id', 'DESC');
|
||||||
|
|
||||||
|
$rsm = new ResultSetMappingBuilder($this->getEntityManager());
|
||||||
|
$rsm->addRootEntityFromClassMetadata(Visit::class, 'v', ['id' => 'visit_id']);
|
||||||
|
$rsm->addJoinedEntityFromClassMetadata(VisitLocation::class, 'vl', 'v', 'visitLocation', [
|
||||||
|
'id' => 'visit_location_id',
|
||||||
|
]);
|
||||||
|
|
||||||
|
$query = $this->getEntityManager()->createNativeQuery($nativeQb->getSQL(), $rsm);
|
||||||
|
|
||||||
|
return $query->getResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function countVisitsByShortCode(string $shortCode, ?string $domain = null, ?DateRange $dateRange = null): int
|
public function countVisitsByShortCode(string $shortCode, ?string $domain = null, ?DateRange $dateRange = null): int
|
||||||
{
|
{
|
||||||
$qb = $this->createVisitsByShortCodeQueryBuilder($shortCode, $domain, $dateRange);
|
/** @var QueryBuilder $qb */
|
||||||
$qb->select('COUNT(DISTINCT v.id)');
|
[$qb] = $this->createVisitsByShortCodeQueryBuilder($shortCode, $domain, $dateRange);
|
||||||
|
$qb->select('COUNT(v.id)');
|
||||||
|
|
||||||
return (int) $qb->getQuery()->getSingleScalarResult();
|
return (int) $qb->getQuery()->getSingleScalarResult();
|
||||||
}
|
}
|
||||||
@ -107,32 +151,26 @@ class VisitRepository extends EntityRepository implements VisitRepositoryInterfa
|
|||||||
string $shortCode,
|
string $shortCode,
|
||||||
?string $domain,
|
?string $domain,
|
||||||
?DateRange $dateRange
|
?DateRange $dateRange
|
||||||
): QueryBuilder {
|
): array {
|
||||||
|
/** @var ShortUrlRepositoryInterface $shortUrlRepo */
|
||||||
|
$shortUrlRepo = $this->getEntityManager()->getRepository(ShortUrl::class);
|
||||||
|
$shortUrl = $shortUrlRepo->findOne($shortCode, $domain) ?? -1;
|
||||||
|
|
||||||
$qb = $this->getEntityManager()->createQueryBuilder();
|
$qb = $this->getEntityManager()->createQueryBuilder();
|
||||||
$qb->from(Visit::class, 'v')
|
$qb->from(Visit::class, 'v')
|
||||||
->join('v.shortUrl', 'su')
|
->where($qb->expr()->eq('v.shortUrl', ':shortUrl'))
|
||||||
->where($qb->expr()->eq('su.shortCode', ':shortCode'))
|
->setParameter('shortUrl', $shortUrl);
|
||||||
->setParameter('shortCode', $shortCode);
|
|
||||||
|
|
||||||
// Apply domain filtering
|
|
||||||
if ($domain !== null) {
|
|
||||||
$qb->join('su.domain', 'd')
|
|
||||||
->andWhere($qb->expr()->eq('d.authority', ':domain'))
|
|
||||||
->setParameter('domain', $domain);
|
|
||||||
} else {
|
|
||||||
$qb->andWhere($qb->expr()->isNull('su.domain'));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Apply date range filtering
|
// Apply date range filtering
|
||||||
if ($dateRange !== null && $dateRange->getStartDate() !== null) {
|
if ($dateRange !== null && $dateRange->getStartDate() !== null) {
|
||||||
$qb->andWhere($qb->expr()->gte('v.date', ':startDate'))
|
$qb->andWhere($qb->expr()->gte('v.date', ':startDate'))
|
||||||
->setParameter('startDate', $dateRange->getStartDate());
|
->setParameter('startDate', $dateRange->getStartDate(), ChronosDateTimeType::CHRONOS_DATETIME);
|
||||||
}
|
}
|
||||||
if ($dateRange !== null && $dateRange->getEndDate() !== null) {
|
if ($dateRange !== null && $dateRange->getEndDate() !== null) {
|
||||||
$qb->andWhere($qb->expr()->lte('v.date', ':endDate'))
|
$qb->andWhere($qb->expr()->lte('v.date', ':endDate'))
|
||||||
->setParameter('endDate', $dateRange->getEndDate());
|
->setParameter('endDate', $dateRange->getEndDate(), ChronosDateTimeType::CHRONOS_DATETIME);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $qb;
|
return [$qb, $shortUrl];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user