Das NDK unterstützt bei der Entwicklung von PHP-Clients, um die Datenbank der Evangelisch-Lutherischen Kirche in Norddeutschland (Nordkirche) anzubinden. Dazu greift es auf die Nordkirche API (kurz "NAPI") zu, um z.B. Adress- und Veranstaltungsdaten zu beziehen.
Welchen Daten über die NAPI ausgelesen werden können, findet man in der Dokumentation zur NAPI. Die selben Filter und Objekteigenschaften, finden sich im NDK wieder. Daher kann diese Dokumentation zusammen mit IntelliSense/Autocomplete einer IDE oder eines passenden Editors benutzt werden, um das NDK und seine Objekte einzusetzen.
Das NDK ist nicht Vorausetzung, um mit der NAPI zu kommunizieren. Es stellt nur Hilfsmittel zur Verfügung, um eigene Anwendungen zu erstellen.
Die verwendeten Namespaces in diesem Projekt entsprechen dem PSR-4-Standard. Deshalb wird in dieser Dokumentation nicht direkt auf Dateien, sondern auf Namespaces und Klassenamen verwiesen.
Die neuste Version installieren:
$ composer require nordkirche/ndk
<?php
$configuration = new \Nordkirche\Ndk\Configuration(
getenv('NAPI_ID'),
getenv('NAPI_ACCESSTOKEN')
);
$configuration->setNapiHost('www.nordkirche.de')->setNapiPath('api/');
$api = new \Nordkirche\Ndk\Api($configuration);
Das NDK stützt sich auf ein Konfigurationsobjekt, welches minimal mit Zugangsdaten in Form einer ID und eines Accesstoken, sowie dem offiziellen NAPI-Endpunkt initialisiert werden muss.
Damit lässt sich die API des NDK initalisieren, um weitere Objekte im Rahmen der Konfiguration zu erhalten.
Daten lassen sich über Repository-Klassen abfragen. Alle verfügbaren Klassen befinden sich unter
\Nordkirche\Ndk\Domain\Repository
.
Filter und Parameter für Abfragen lassen sich über Query-Klassen definieren. Alle Query-Klassen befinden sich unter
\Nordkirche\Ndk\Domain\Query
.
Alle Repositories lassen sich mit den Queries \Nordkirche\Ndk\Domain\Query\PageQuery
oder
\Nordkirche\Ndk\Domain\Query\SimpleQuery
anfragen. Einige, aber nicht alle, Repositories haben spezifische
Query-Klassen, um besondere Filter anzuwenden. Diese folgenden dem Namenschema
\Nordkirche\Ndk\Domain\Repository\*Repository -> \Nordkirche\Ndk\Domain\Query\*Query
und sind immer von der Klasse \Nordkirche\Ndk\Domain\Query\PageQuery
abgeleitet.
Die Verfügbaren Repositories und Filter-Optionen in Queries sind equivalent zu den Endpunkten und Filter-Optionen in der Dokumentation der NAPI.
<?php
$api = new \Nordkirche\Ndk\Api($configuration);
$repository = $api->factory(\Nordkirche\Ndk\Domain\Repository\InstitutionRepository::class);
$result = $repository->get(new \Nordkirche\Ndk\Domain\Query\InstitutionQuery());
foreach ($result as $institution) {
echo $institution->getLabel() . ' (' . $institution->getName() . ')' . PHP_EOL;
}
Dieses Beispiel besteht aus zwei Teilen. Zum einen der Initialisierung der API und dem erstellen eine Repository-Objektes
mit dem Methode factory()
, zum anderen dem Abfragen der Daten mit Hilfe eines Query-Objektes.
<?php
$api = new \Nordkirche\Ndk\Api($configuration);
$repository = $api->factory(\Nordkirche\Ndk\Domain\Repository\InstitutionRepository::class);
Das NDK arbeitetet mit Repositories, welche die verschiedenen Endpunkte der NAPI repräsentieren. In diesem Beispiel
benutzten wir das InstitutionRepository
, um Institutionen abfragen zu können. Die Methode factory()
der API
instanziiert unser Repository mit der vorher definierten Konfiguration.
<?php
$result = $repository->get(new \Nordkirche\Ndk\Domain\Query\InstitutionQuery());
foreach ($result as $institution) {
echo $institution->getLabel() . ' (' . $institution->getName() . ')' . PHP_EOL;
}
Mit der Methode get()
und einem Query-Objekt fragen wir eine Liste der Institutionen ab. Das NDK verpackt diese in ein
Objekt der Klasse \Nordkirche\Ndk\Service\Result
, welches neben den Ressourcen-Objekten in Form von
\Nordkirche\Ndk\Domain\Model\Institution\Instituion
Objekten zusätzliche Meta-Daten zu einer Anfrage enthält.
Die NAPI liefert Ergebnisse paginiert. Deshalb erhalten wir mit dieser Anfrage nur die erste Seite.
Meta-Daten sind z.B. die Anzahl aller gefundenen Objekte, wieviel Objekte auf der aktuellen Seite sind, evtl. zugehörige Facetten zur Suchanfrage, um weitere Filter anwenden zu können etc.
Dieses Beispiel kann unter examples/2_simple_request.php
direkt ausgeführt werden.
Welche Seite man erhält, ob nach bestimmten Daten gesucht/gefiltert werden soll etc., wird über ein Query-Objekt
bestimmt. Im vorherigen Beispiel über die Klasse \Nordkirche\Ndk\Domain\Query\InstitutionQuery
.
Da die meisten Anfragen paginiert sind, bildet die Klasse \Nordkirche\Ndk\Domain\Query\PageQuery
oft die Elternklasse
für *Query
Klassen. Diese lässt sich auch für alle Anfrage nutzen, vermisst dann allerdings die speziellen
Filtermöglichkeiten z.B. nach Postleitzahl.
<?php
$query = new \Nordkirche\Ndk\Domain\Query\InstitutionQuery();
$query->setPageSize(30)->setPageNumber(2)->setZipCodes([24103]);
$result = $repository->get($query);
Objekte wie das InstitutionQuery
bieten verschiedene Optionen, um eine Anfrage zu modifizieren. Neben den allgemeinen
Optionen die gewünschte Seite setPageNumber()
und die Einträge pro Seite setPageSize()
zu definieren,
können wir bei Institutionen und Personen z.B. nach Postleitzahl filtern.
Hier empfiehlt es sich, einmal alle Setter einer *Query
Klasse anzusehen. Auch die
Dokumentation der NAPI bietet hier Aufschluss über die
möglichen Filter.
Im NDK unterscheiden wir zwischen zwei Arten von Models welche auf zwei abstrakten Klassen aufbauen:
\Nordkirche\Ndk\Domain\Model\AbstractModel
\Nordkirche\Ndk\Domain\Model\AbstractResourceObject
Das AbstractResourceObject
ist eine erweiterung des generischen AbstractModels
und zeichent ein Objekt als Resource
der NAPI aus. Diese Objekte haben eine ID, einen Typ und Relationen zu anderen Resourcen. Sie werden über Repositories bezogen.
Alle AbstractModels
sind fast auschließlich Unterobjekte einer Resource und gliedern die teilweise komplexen Daten
der Resourcen-Objekte.
In der Regel kann mit den jeweiligen get*()
-Methoden auf die Eigenschaften einer Resource zugegriffen werden.
Die dynamische Verarbeitung von Objekten, die mit method_exists()
prüft, ob ein Attribute holbar ist, ist allerdings
nicht möglich, da einige get*()
-Methoden magisch sind.
Um trotzdem eine dynamische Prüfung und Verarbeitung von Attributen zu ermöglichen, kann mit property_exists()
und
dem direkten Zugriff auf die Property via $object->property
gearbeitet werden. Dabei werden Features wie
ResourcePlaceholder
und ResolutionProxy
aus den folgenden Abschnitten berücksichtigt.
Die NAPI selbst basiert auf der JSON API 1.0 Spezifikation und arbeitet deshalb mit der Option, Relationen zu anderen Resourcen nicht unmittelbar zurück zu geben, sondern nur zu referenzieren.
Dies führt dazu, dass verwandte Resourcen-Objekte nicht automatisch inkludiert sind, sondern vom NDK bei Bedarf nachgeladen werden. Wir erklären dieses Verhalten einmal Anhand der Institutionen:
Instituionen haben Relationen zu anderen Ressourcen-Objekten, z.B. zu Ihrer Adresse oder ihrem Typ.
Ein Aufruf der Methode getAddress()
führt dazu, dass eine weitere Anfrage an die NAPI gestellt wird,
um die Adressdaten der Institution zu erhalten.
Kann eine Ressource nicht dynamisch nachgeladen werden, so wird ein ResourcePlaceholder
-Objekt zurück gegeben.
Dieses beinhaltet ein generiertes Label für die Ressource und ggf. den Grund für das fehlgeschlagene Nachladen.
Möchte man auf diese Weise eine Adressliste ausgeben, wird für jeden Eintrag in der Adressliste eine weitere Anfrage gegen die NAPI gestellt.
Ist bekannt, dass man zu allen Institutionen die Adresse benötigt, lässt sich die Relation in die Antwort der NAPI inkludieren und man verhindert unnötige Anfragen. Dazu wird dem Query-Objekt eine Liste mit Relationen mitgegeben:
<?php
use Nordkirche\Ndk\Domain\Model\Institution\Institution;
$query->setInclude([Institution::RELATION_ADDRESS, Institution::RELATION_INSTITUTION_TYPE]);
Eine Liste mit allen möglichen Relationen gibt es nicht, da diese spezifisch für das Resourcen-Objekt sind.
Alle Resourcen-Klassen mit Relationen enthalten Konstanten, beginnend mit RELATION_
, welche mögliche Relationen
ausweisen und mit denen ein Include-Array gebaut werden kann.
Aus dem Namen des Repositories lässt sich immer auf die zugehörige Resourcen-Klasse schließen:
\Nordkirche\Ndk\Domain\Repository\PersonRepository -> \Nordkirche\Ndk\Domain\Model\Person\Person
Includes können auch verschachtelt werden, um die Relationen von Relationen zu inkludieren:
<?php
use Nordkirche\Ndk\Domain\Model\Institution\Institution;
$query->setInclude([
Institution::RELATION_ADDRESS,
Institution::RELATION_PARENT_INSTITUTIONS => [
Institution::RELATION_ADDRESS
]
]);
Wir inkludieren in obigem Beispiel die Adressen für alle Institutionen die wir Angefragt haben, inkludieren die übergeordneten Institutionen und wiederum die dazu gehörigen Adressen.
Man könnte dazu tendieren, immer alle Relationen zu inkludieren, um unter allen Umständen keine unnötigen Anfragen gegen die NAPI zu schicken. Dabei sei angemerkt, dass das Inkludieren von Daten die Abfragen langsamer macht. Daher sollte ganz bewusst ausgewählt werden, welche Relationen für welche Abfragen benötigt werden.
Anhand des Beispiels examples/3_includes.php
lässt sich der Unterschied zwischen Anfragen mit und ohne setIncludes()
nachvollziehen.
Möchte man, dass Ressourcen niemals dynamisch nachgeladen werden, so besteht die Möglichkeit dies zu deaktiveren:
<?php
$configuration->setResolutionProxyDisabled(true);
Diese Einstellung führt dazu, dass nicht inkludierte Relationen null
zurück liefern. Der Zugriff auf nicht inkludierte
Relationen wird geloggt und kann ausgegeben werden. Dazu mehr im Abschnitt Logging.
Manchmal ist es gewünscht, Relationen z.B. zu Personen der Nordkirche für eigene Objekten in einer Datenbank herzustellen. Dazu ist es möglich ein Resourcen-Objekt des NDK in eine URI umzuwandeln, welche gespeichert werden kann.
Diese URIs erhält man beim Casten eines Ressourcen-Objektes zu einem String oder durch Aufruf der Methode __toString()
.
<?php
$uri = (string)$institution;
// oder
$uri = $institution->__toString();
Der String der daraus resultiert, ist im Format: napi://resource/type/1234
.
Diese URIs lassen sich komfortabel wieder auflösen:
<?php
$napi = $api->factory(\Nordkirche\Ndk\Service\NapiService::class);
$result = $napi->resolveUrl($uri);
Auch ganze Listen von URIs lassen sich mit dem NDK direkt auflösen und fehlende Objekte behandeln:
<?php
$uris = ['napi://resource/typeA/1234', 'napi://resource/typeB/4567', 'napi://resource/typeC/891011'];
$napi = $api->factory(\Nordkirche\Ndk\Service\NapiService::class);
foreach ($napi->resolveUrls($uris) as $result) {
if ($result instanceof \Nordkirche\Ndk\Domain\Model\ResourcePlaceholder) {
echo "Resource not found: " . $result->getLabel() . PHP_EOL;
} else {
echo "Found: " . $result->getName() . ' (' . $result->getId() . ')' . PHP_EOL;
}
}
In diesem Arrays können gemischt Personen, Institutionen, Verstanstaltungen etc. vorkommen.
Das Beispiel examples/5_napi_urls.php
führt die Umwandlung und Auflösung einmal praktisch aus.
Resourcen-Objekte können mit Bildern versehen sein. Zum Beispiel hat eine Institution ein Logo getLogo()
oder eine
Veranstaltung ein beschreibendes Bild getPicture()
. Ein Bild wird durch ein Objekt der Klasse
\Nordkirche\Ndk\Domain\Model\File\Image
repräsentiert.
<?php
$repository = $api->factory(\Nordkirche\Ndk\Domain\Repository\InstitutionRepository::class);
$instituion = $repository->getById(1930);
$imageUrl = $instituion->getLogo()->render(150);
echo $imageUrl . PHP_EOL;
Neben üblichen Bild- und Datei-Eigenschaften bietet es die Methode render($width, $height)
,
um die URL zu einer skalierte Version des Bildes zu erhalten. Die Argumente $width
und $height
sind beide optional.
Werden beide Argumente angegeben und eines (oder beide) mit dem Postifx c
versehen, so wird das Bild gecroppt:
<?php
$imageUrl = $event->getPicture()->render('100c', '100c');
Weiter Bild- und Datei-Eigenschaften erhält man über getDetails()
. Dazu gehören Bildunterschriften,
Copyright-Angaben usw.
Dem im ersten Abschnitt erstellten Konfigurations-Objekt \Nordkirche\Ndk\Configuration
können noch weitere
Optionen mitgegeben werden, um das Verhalten des NDK anzupassen.
Über PSR-3 kompatible Logger können Anfragen des NDK an die NAPI mitgeschnitten werden.
<?php
$logger = new \Monolog\Logger(
'stdout',
[new \Monolog\Handler\StreamHandler('php://stdout')]
);
$configuration->setLogger($logger);
Siehe examples/5_logging.php
für ein Anwendungsbeispiel mit Monolog.
Per Default werden HTTP-Anfragen an die NAPI und interne Daten flüchtig in einem Doctrine ArrayCache hinterlegt.
Es ist möglich PSR-6 kompatible Caches für bestimmte Aspekte der Anwendung einzurichten, z.B. mit dem diversen Treibern
aus doctrine/cache
.
Eine beispielhafte Konfiguration lässt sich unter 6_Caching.php
einsehen und ausführen.
Das Paket doctrine/cache
selbst bietet
Support für diverse Backends.
Für alle konfigurierbaren Caches ist ein InMemory-Cache als Backend zu empfehlen. Beispielsweise APCu, Redis oder Memcache.
<?php
$configuration->setHttpCacheProvider(new \Doctrine\Common\Cache\FilesystemCache('./cache/http'));
Konfiguriert das Backend für die HTTP-Caching-Middleware. Die NAPI prüft ob Anfragen im Cache neue Daten enthalten und nimmt ggf. die bestehenden Daten aus dem eigenen HTTP-Cache.
<?php
$configuration->setReflectionCacheProvider(new \Doctrine\Common\Cache\FilesystemCache('./cache/reflection'));
$configuration->setDependencyInjectionCacheProvider(new \Doctrine\Common\Cache\FilesystemCache('./cache/di'));
Diese internen Caches heben die allgemeine Ausführungsgeschwindikeit des NDK an.