forked from GNUsocial/gnu-social
[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:
parent
6c7f69dd94
commit
d4c77925d2
@ -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;
|
||||||
@ -72,7 +73,8 @@ class DB
|
|||||||
*/
|
*/
|
||||||
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];
|
||||||
|
@ -291,9 +291,8 @@ 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
|
||||||
@ -306,15 +305,12 @@ class Actor extends Entity
|
|||||||
['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
|
||||||
@ -333,8 +329,6 @@ class Actor extends Entity
|
|||||||
['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,
|
|
||||||
),
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user