MySQL-spezifische Funktionen in DQL nutzen

in  Zikula Apps , ,

MySQL-spezifische Funktionen in DQL nutzen

Im Kern von Doctrine 2 wird Wert darauf gelegt, dass die Doctrine Query Language (DQL) möglichst portabel bleibt, damit Projekte einfach mit unterschiedlichen Datenquellen arbeiten können. Bei Webanwendungen kommt es immer wieder einmal vor, dass man eine bestimmte Funktion benötigt, die zwar in MySQL existiert, jedoch nicht von DQL unterstützt wird.

Man kann in solchen Fällen eigene DQL-Funktionen erstellen, dies ist aber in den meisten Fällen zum Glück nicht nötig. Denn in den DoctrineExtensions von Benjamin Eberlei befinden sich bereits eine ganze Reihe von MySQL-Funktionen. Dieses Paket ist idealerweise auch im Zikula Core enthalten, so dass man sie für die eigenen Module direkt nutzen kann.

Zumindest in Zikula 1.3.x sind die Funktionen aber per Standard nicht via Symfony registriert (für 1.4 habe ich das nicht nachgeschaut, aber dort verhält es sich vermutlich ähnlich). Daher habe ich einfach die Funktionen, die ich benötigt habe, geladen und dem EntityManager von Doctrine bekannt gemacht. Das folgende Beispiel ist Code aus einer Repository-Methode, die eine gegebene QueryBuilder-Instanz erweitert:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
$config = $this->getEntityManager()->getConfiguration();
$config->addCustomNumericFunction('ACOS', 'DoctrineExtensions\\Query\\Mysql\\Acos');
$config->addCustomNumericFunction('COS', 'DoctrineExtensions\\Query\\Mysql\\Cos');
$config->addCustomNumericFunction('RADIANS', 'DoctrineExtensions\\Query\\Mysql\\Radians');
$config->addCustomNumericFunction('SIN', 'DoctrineExtensions\\Query\\Mysql\\Sin');
$qb->addSelect(
    '(6371 * ACOS( COS( RADIANS(\'' . $lat . '\') )
        * COS( RADIANS( tbl.latitude ) )
        * COS( RADIANS( tbl.longitude ) - RADIANS(\'' . $long . '\') )
        + SIN( RADIANS(\'' . $lat . '\') )
        * SIN( RADIANS( tbl.latitude ) ) ) ) AS ' . /*HIDDEN */ 'distance'
)
->having('distance <= :distance')
->setParameter('distance', $radius)
->orderBy('distance');

Das Keyword HIDDEN ist derzeit auskommentiert, da in Zikula 1.3 noch eine ältere Version von Doctrine 2 verwendet wird, die dieses Schlüsselwort noch nicht unterstützt. Bei der Umstellung des Modules auf 1.4 wird es dann aktiviert, was die weitere Behandlung des Abfrageergebnisses etwas vereinfacht. Denn mit dem zusätzlichen Feld distance liefert die Abfrage nicht mehr die selekterten Objekte alleine, sondern ein Array mit Paaren aus den Objekten und dem neuen Feld zurück.

Weitere Beiträge in Kategorie Zikula Apps

Symfony 6.2.7 korrigiert CSRF-Problem im SecurityBundle
- Am ersten Februar, also vor genau einem Monat, wurden mit Symfony 6.2.6 zwei Sicherheitsprobleme adressiert. Ein Patch im SecurityBundle hatte allerdings auch valide Use Cases gebrochen, wie zum …
Flexible Tabellenspalten im EasyAdminBundle
- Vor kurzer Zeit haben wir das EasyAdminBundle (EAB) in die Entwicklungsversion von Zikula 4 integriert um das alte Admin-Interface abzulösen. Seitdem beobachten wir natürlich stets, was sich bei …
Kommende Neuerungen in Symfony 6.2
- Gegenwärtig laufen die Arbeiten an der nächsten Symfony-Version 6.2. Wie immer gibt es regelmäßige Einblicke in die wichtigsten, zu erwartenden Features und Verbesserungen. Dieser Beitrag zeigt im …
Zikula 4 - Ansätze für ein leichtgewichtigeres User Management
- Im Rahmen der Schlankheitskur vom Zikula Core sind einige Altlasten bereits entfernt worden. Das Admin-Interface baut nun auf dem EasyAdminBundle auf. Ein größerer Knoten, den es noch zu entwirren …
Zikula 4 und ModuleStudio - Weitere Integration mit dem EasyAdminBundle
- Im letzten Beitrag wurde unter anderem dargestellt, dass Zikula 4 nun das EasyAdminBundle (EAB) integriert um das alte Admin-Interface abzulösen. Zwischenzeitlich sind die Arbeiten daran etwas …