[CORE][DB][ENTITY][Actor] Make DB::dql return a chunked array if selecting multiple entities, remove partitioning from callsite

`DB::dql('select a, b, from a join b')` would previously return `[a,
b, a, b, ...]` (or even `[b, a, b, a, ...]`), and now will return
`[[a, a, ...], [b, b, ...]]`. The issue would be further compounded
when selecting even more entities, where the order would be
unpredictable
This commit is contained in:
Hugo Sales 2021-12-01 00:42:56 +00:00 committed by Diogo Peralta Cordeiro
parent 6c7f69dd94
commit d4c77925d2
Signed by: diogo
GPG Key ID: 18D2D35001FBFAB0
2 changed files with 75 additions and 45 deletions

View File

@ -36,6 +36,7 @@ namespace App\Core\DB;
use App\Util\Exception\DuplicateFoundException; use App\Util\Exception\DuplicateFoundException;
use App\Util\Exception\NotFoundException; use App\Util\Exception\NotFoundException;
use Closure;
use Doctrine\Common\Collections\Criteria; use Doctrine\Common\Collections\Criteria;
use Doctrine\Common\Collections\ExpressionBuilder; use Doctrine\Common\Collections\ExpressionBuilder;
use Doctrine\ORM\EntityManager; use Doctrine\ORM\EntityManager;
@ -70,9 +71,10 @@ class DB
/** /**
* Table name to class map, used to allow specifying table names instead of classes in doctrine calls * Table name to class map, used to allow specifying table names instead of classes in doctrine calls
*/ */
private static array $table_map = []; private static array $table_map = [];
private static array $class_pk = []; private static array $class_pk = [];
private static ?string $table_entity_pattern = null; private static ?string $sql_table_entity_pattern = null;
private static ?array $dql_table_name_patterns = null;
public static function initTableMap() public static function initTableMap()
{ {
$all = self::$em->getMetadataFactory()->getAllMetadata(); $all = self::$em->getMetadataFactory()->getAllMetadata();
@ -81,7 +83,8 @@ class DB
self::$class_pk[$meta->getMetadataValue('name')] = $meta->getIdentifier(); self::$class_pk[$meta->getMetadataValue('name')] = $meta->getIdentifier();
} }
self::$table_entity_pattern = '/(' . implode('|', array_keys(self::$table_map)) . ')\s([^\s]+)/'; self::$sql_table_entity_pattern = '/(' . implode('|', array_keys(self::$table_map)) . ')\s([^\s]+)/';
self::$dql_table_name_patterns = F\map(self::$table_map, fn ($_, $s) => "/(?<!\\.)\\b{$s}\\b/");
} }
public static function getTableForClass(string $class) public static function getTableForClass(string $class)
@ -99,7 +102,7 @@ class DB
*/ */
public static function dql(string $query, array $params = [], array $options = []) public static function dql(string $query, array $params = [], array $options = [])
{ {
$query = preg_replace(F\map(self::$table_map, fn ($_, $s) => "/(?<!\\.)\\b{$s}\\b/"), self::$table_map, $query); $query = preg_replace(self::$dql_table_name_patterns, self::$table_map, $query);
$q = new Query(self::$em); $q = new Query(self::$em);
$q->setDQL($query); $q->setDQL($query);
@ -114,7 +117,40 @@ class DB
$q->setParameter($k, $v); $q->setParameter($k, $v);
} }
return $q->getResult(); $results = $q->getResult();
// So, Doctrine doesn't return 'select a, b from a join b' as [[a, a], [b, b]], but as [a, b, a, b] (or even [b,
// a, b, a]), so we do it ourselves. For whatever reason, neither the AST nor the ResultSetMapping have the
// entities in the correct order, so we need to "parse" the query ourselves. This only applies if there's no '.'
// in the select clause (i.e. we're selecting whole entities, not just a bunch of columns)
$matches = []; // v not a space in case of line breaks
if ($ret = preg_match('/SELECT.([^\.]*).FROM/is', $query, $matches)) {
// Grab the entities from the select clause and trim spaces
$entities = F\map(explode(',', $matches[1]), fn ($p) => trim($p));
if (\count($entities) > 1) { // If more than one entities in the select clause
// Call protected method getResultSetMapping, get the alias map (to avoid parsing it ourselves, or
// dealing with the AST)
$aliases = Closure::bind(fn ($q) => $q->getResultSetMapping(), null, $q)($q)->aliasMap;
// Since the order is not necessarily the correct one in the results (for whatever reason) (though it
// presumably is the same as in the AST, but just in case), use Functional\partition to chunk the
// results into groups of the same class
return F\partition(
$results,
...F\map(
// partition partitions into one more array than we want (those that don't pass any predicate),
// so drop the last
F\but_last($entities),
// Map into a list of callables that each check if the given object is an instance of the class
// in $aliases
fn ($p) => (fn ($o) => $o instanceof $aliases[$p]),
),
);
} else {
return $results;
}
} else {
return $results;
}
} }
/** /**
@ -130,7 +166,7 @@ class DB
$rsmb = new ResultSetMappingBuilder(self::$em, \is_null($entities) ? ResultSetMappingBuilder::COLUMN_RENAMING_INCREMENT : ResultSetMappingBuilder::COLUMN_RENAMING_NONE); $rsmb = new ResultSetMappingBuilder(self::$em, \is_null($entities) ? ResultSetMappingBuilder::COLUMN_RENAMING_INCREMENT : ResultSetMappingBuilder::COLUMN_RENAMING_NONE);
if (\is_null($entities)) { if (\is_null($entities)) {
$matches = []; $matches = [];
preg_match_all(self::$table_entity_pattern, $query, $matches); preg_match_all(self::$sql_table_entity_pattern, $query, $matches);
$entities = []; $entities = [];
foreach (F\zip($matches[1], $matches[2]) as [$table, $alias]) { foreach (F\zip($matches[1], $matches[2]) as [$table, $alias]) {
$entities[$alias] = self::$table_map[$table]; $entities[$alias] = self::$table_map[$table];

View File

@ -291,49 +291,43 @@ class Actor extends Entity
{ {
if (\is_null($scoped)) { if (\is_null($scoped)) {
return Cache::get( return Cache::get(
"othertags-{$this->getId()}", "actor-circles-and-tags-{$this->getId()}",
fn () => F\partition( fn () => DB::dql(
DB::dql( <<< 'EOQ'
<<< 'EOQ' SELECT circle, tag
SELECT circle, tag FROM actor_tag tag
FROM actor_tag tag JOIN actor_circle circle
JOIN actor_circle circle WITH tag.tagger = circle.tagger
WITH tag.tagger = circle.tagger AND tag.tag = circle.tag
AND tag.tag = circle.tag WHERE tag.tagged = :id
WHERE tag.tagged = :id ORDER BY tag.modified DESC, tag.tagged DESC
ORDER BY tag.modified DESC, tag.tagged DESC EOQ,
EOQ, ['id' => $this->getId()],
['id' => $this->getId()], options: ['offset' => $offset, 'limit' => $limit],
options: ['offset' => $offset, 'limit' => $limit],
),
fn ($o) => $o instanceof ActorCircle,
), ),
); );
} else { } else {
$scoped_id = \is_int($scoped) ? $scoped : $scoped->getId(); $scoped_id = \is_int($scoped) ? $scoped : $scoped->getId();
return Cache::get( return Cache::get(
"othertags-{$this->getId()}-by-{$scoped_id}", "actor-circles-and-tags-{$this->getId()}-by-{$scoped_id}",
fn () => F\partition( fn () => DB::dql(
DB::dql( <<< 'EOQ'
<<< 'EOQ' SELECT circle, tag
SELECT circle, tag FROM actor_tag tag
FROM actor_tag tag JOIN actor_circle circle
JOIN actor_circle circle WITH tag.tagger = circle.tagger
WITH tag.tagger = circle.tagger AND tag.tag = circle.tag
AND tag.tag = circle.tag WHERE
WHERE tag.tagged = :id
tag.tagged = :id AND (circle.private != true
AND (circle.private != true OR (circle.tagger = :scoped
OR (circle.tagger = :scoped AND circle.private = true
AND circle.private = true )
) )
) ORDER BY tag.modified DESC, tag.tagged DESC
ORDER BY tag.modified DESC, tag.tagged DESC EOQ,
EOQ, ['id' => $this->getId(), 'scoped' => $scoped_id],
['id' => $this->getId(), 'scoped' => $scoped_id], options: ['offset' => $offset, 'limit' => $limit],
options: ['offset' => $offset, 'limit' => $limit],
),
fn ($o) => $o instanceof ActorCircle,
), ),
); );
} }